Profile.pm 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293
  1. package Selenium::Firefox::Profile;
  2. # ABSTRACT: Use custom profiles with Selenium::Remote::Driver
  3. use strict;
  4. use warnings;
  5. use Archive::Zip qw( :ERROR_CODES );
  6. use Archive::Extract;
  7. use Carp qw(croak);
  8. use Cwd qw(abs_path);
  9. use File::Copy qw(copy);
  10. use File::Temp;
  11. use File::Basename qw(dirname);
  12. use JSON qw/decode_json/;
  13. use MIME::Base64;
  14. use Scalar::Util qw(blessed looks_like_number);
  15. =head1 DESCRIPTION
  16. You can use this module to create a custom Firefox Profile for your
  17. Selenium tests. Currently, you can set browser preferences and add
  18. extensions to the profile before passing it in the constructor for a
  19. new Selenium::Remote::Driver.
  20. =head1 SYNPOSIS
  21. use Selenium::Remote::Driver;
  22. use Selenium::Remote::Driver::Firefox::Profile;
  23. my $profile = Selenium::Remote::Driver::Firefox::Profile->new;
  24. $profile->set_preference(
  25. 'browser.startup.homepage' => 'http://www.google.com',
  26. 'browser.cache.disk.capacity' => 358400
  27. );
  28. $profile->set_boolean_preference(
  29. 'browser.shell.checkDefaultBrowser' => 0
  30. );
  31. $profile->add_extension('t/www/redisplay.xpi');
  32. my $driver = Selenium::Remote::Driver->new(
  33. 'firefox_profile' => $profile
  34. );
  35. $driver->get('http://www.google.com');
  36. print $driver->get_title();
  37. =cut
  38. sub new {
  39. my $class = shift;
  40. # TODO: add handling for a pre-existing profile folder passed into
  41. # the constructor
  42. # TODO: accept user prefs, boolean prefs, and extensions in
  43. # constructor
  44. my $self = {
  45. profile_dir => File::Temp->newdir(),
  46. user_prefs => {},
  47. extensions => []
  48. };
  49. bless $self, $class or die "Can't bless $class: $!";
  50. return $self;
  51. }
  52. =method set_preference
  53. Set string and integer preferences on the profile object. You can set
  54. multiple preferences at once. If you need to set a boolean preference,
  55. either use JSON::true/JSON::false, or see C<set_boolean_preference()>.
  56. $profile->set_preference("quoted.integer.pref" => '"20140314220517"');
  57. # user_pref("quoted.integer.pref", "20140314220517");
  58. $profile->set_preference("plain.integer.pref" => 9005);
  59. # user_pref("plain.integer.pref", 9005);
  60. $profile->set_preference("string.pref" => "sample string value");
  61. # user_pref("string.pref", "sample string value");
  62. =cut
  63. sub set_preference {
  64. my ($self, %prefs) = @_;
  65. foreach (keys %prefs) {
  66. my $value = $prefs{$_};
  67. my $clean_value = '';
  68. if ( blessed($value) ) {
  69. $self->set_boolean_preference($_, $value );
  70. next;
  71. }
  72. elsif ($value =~ /^(['"]).*\1$/ or looks_like_number($value)) {
  73. # plain integers: 0, 1, 32768, or integers wrapped in strings:
  74. # "0", "1", "20140204". in either case, there's nothing for us
  75. # to do.
  76. $clean_value = $value;
  77. }
  78. else {
  79. # otherwise it's hopefully a string that we'll need to
  80. # quote on our own
  81. $clean_value = '"' . $value . '"';
  82. }
  83. $self->{user_prefs}->{$_} = $clean_value;
  84. }
  85. }
  86. =method set_boolean_preference
  87. Set preferences that require boolean values of 'true' or 'false'. You
  88. can set multiple preferences at once. For string or integer
  89. preferences, use C<set_preference()>.
  90. $profile->set_boolean_preference("false.pref" => 0);
  91. # user_pref("false.pref", false);
  92. $profile->set_boolean_preference("true.pref" => 1);
  93. # user_pref("true.pref", true);
  94. =cut
  95. sub set_boolean_preference {
  96. my ($self, %prefs) = @_;
  97. foreach (keys %prefs) {
  98. my $value = $prefs{$_};
  99. $self->{user_prefs}->{$_} = $value ? 'true' : 'false';
  100. }
  101. }
  102. =method get_preference
  103. Retrieve the computed value of a preference. Strings will be double
  104. quoted and boolean values will be single quoted as "true" or "false"
  105. accordingly.
  106. $profile->set_boolean_preference("true.pref" => 1);
  107. print $profile->get_preference("true.pref") # true
  108. $profile->set_preference("string.pref" => "an extra set of quotes");
  109. print $profile->get_preference("string.pref") # "an extra set of quotes"
  110. =cut
  111. sub get_preference {
  112. my ($self, $pref) = @_;
  113. return $self->{user_prefs}->{$pref};
  114. }
  115. =method add_extension
  116. Add an existing C<.xpi> to the profile by providing its path. This
  117. only works with packaged C<.xpi> files, not plain/un-packed extension
  118. directories.
  119. $profile->add_extension('t/www/redisplay.xpi');
  120. =cut
  121. sub add_extension {
  122. my ($self, $xpi) = @_;
  123. croak 'File not found: ' . $xpi unless -e $xpi;
  124. my $xpi_abs_path = abs_path($xpi);
  125. croak '$xpi_abs_path: extensions must be in .xpi format' unless $xpi_abs_path =~ /\.xpi$/;
  126. push (@{$self->{extensions}}, $xpi_abs_path);
  127. }
  128. =method add_webdriver
  129. Primarily for internal use, we add the webdriver extension to the
  130. current Firefox profile.
  131. =cut
  132. sub add_webdriver {
  133. my ($self, $port) = @_;
  134. my $this_dir = dirname(abs_path(__FILE__));
  135. my $webdriver_extension = $this_dir . '/webdriver.xpi';
  136. my $default_prefs_filename = $this_dir . '/webdriver_prefs.json';
  137. my $json;
  138. {
  139. local $/;
  140. open (my $fh, '<', $default_prefs_filename);
  141. $json = <$fh>;
  142. close ($fh);
  143. }
  144. my $webdriver_prefs = decode_json($json);
  145. $self->set_preference(%{ $webdriver_prefs->{mutable} });
  146. $self->set_preference(%{ $webdriver_prefs->{frozen} });
  147. $self->add_extension($webdriver_extension);
  148. $self->set_preference('webdriver_firefox_port', $port);
  149. }
  150. sub _encode {
  151. my $self = shift;
  152. # The remote webdriver accepts the Firefox profile as a base64
  153. # encoded zip file
  154. $self->_layout_on_disk();
  155. my $zip = Archive::Zip->new();
  156. my $dir_member = $zip->addTree( $self->{profile_dir} );
  157. my $string = "";
  158. open (my $fh, ">", \$string);
  159. binmode($fh);
  160. unless ( $zip->writeToFileHandle($fh) == AZ_OK ) {
  161. die 'write error';
  162. }
  163. return encode_base64($string);
  164. }
  165. sub _layout_on_disk {
  166. my $self = shift;
  167. $self->_write_preferences();
  168. $self->_install_extensions();
  169. return $self->{profile_dir};
  170. }
  171. sub _write_preferences {
  172. my $self = shift;
  173. my $userjs = $self->{profile_dir} . "/user.js";
  174. open (my $fh, ">>", $userjs)
  175. or die "Cannot open $userjs for writing preferences: $!";
  176. foreach (keys %{$self->{user_prefs}}) {
  177. print $fh 'user_pref("' . $_ . '", ' . $self->get_preference($_) . ');' . "\n";
  178. }
  179. close ($fh);
  180. }
  181. sub _install_extensions {
  182. my $self = shift;
  183. my $extension_dir = $self->{profile_dir} . "/extensions/";
  184. mkdir $extension_dir unless -d $extension_dir;
  185. # TODO: handle extensions that need to be unpacked
  186. foreach (@{$self->{extensions}}) {
  187. # For Firefox to recognize the extension, we have to put the
  188. # .xpi in the /extensions/ folder and change the filename to
  189. # its id, which is found in the install.rdf in the root of the
  190. # zip.
  191. my $ae = Archive::Extract->new(
  192. archive => $_,
  193. type => "zip"
  194. );
  195. $Archive::Extract::PREFER_BIN = 1;
  196. my $tempDir = File::Temp->newdir();
  197. $ae->extract( to => $tempDir );
  198. my $install = $ae->extract_path();
  199. $install .= '/install.rdf';
  200. open (my $fh, "<", $install)
  201. or croak "No install.rdf inside $_: $!";
  202. my (@file) = <$fh>;
  203. close ($fh);
  204. my @name = grep { chomp; $_ =~ /<em:id>[^{]/ } @file;
  205. $name[0] =~ s/.*<em:id>(.*)<\/em:id>.*/$1/;
  206. my $xpi_dest = $extension_dir . $name[0] . ".xpi";
  207. copy($_, $xpi_dest)
  208. or croak "Error copying $_ to $xpi_dest : $!";
  209. }
  210. }
  211. 1;
  212. __END__
  213. =head1 SEE ALSO
  214. http://kb.mozillazine.org/About:config_entries
  215. https://developer.mozilla.org/en-US/docs/Mozilla/Preferences/A_brief_guide_to_Mozilla_preferences