class Selenium::WebDriver::Firefox::Profile
def []=(key, value)
def []=(key, value) unless VALID_PREFERENCE_TYPES.any? { |e| value.is_a? e } raise TypeError, "expected one of #{VALID_PREFERENCE_TYPES.inspect}, got #{value.inspect}:#{value.class}" end if value.is_a?(String) && Util.stringified?(value) raise ArgumentError, "preference values must be plain strings: #{key.inspect} => #{value.inspect}" end @additional_prefs[key.to_s] = value end
def add_extension(path, name = extension_name_for(path))
def add_extension(path, name = extension_name_for(path)) @extensions[name] = Extension.new(path) end
def decoded(json)
def decoded(json) JSON.parse(json) end
def delete_extensions_cache(directory)
def delete_extensions_cache(directory) FileUtils.rm_f File.join(directory, 'extensions.cache') end
def delete_lock_files(directory)
def delete_lock_files(directory) LOCK_FILES.each do |name| FileUtils.rm_f File.join(directory, name) end end
def extension_name_for(path)
def extension_name_for(path) File.basename(path, File.extname(path)) end
def from_name(name)
def from_name(name) profile = ini[name] return profile if profile raise Error::WebDriverError, "unable to find profile named: #{name.inspect}" end
def ini
def ini @ini ||= ProfilesIni.new end
def initialize(model = nil)
def initialize(model = nil) @model = verify_model(model) @additional_prefs = read_model_prefs @extensions = {} end
def install_extensions(directory)
def install_extensions(directory) destination = File.join(directory, 'extensions') @extensions.each do |name, extension| WebDriver.logger.debug({extension: name}.inspect, id: :firefox_profile) extension.write_to(destination) end end
def layout_on_disk
def layout_on_disk profile_dir = @model ? create_tmp_copy(@model) : Dir.mktmpdir('webdriver-profile') FileReaper << profile_dir install_extensions(profile_dir) delete_lock_files(profile_dir) delete_extensions_cache(profile_dir) update_user_prefs_in(profile_dir) profile_dir end
def log_file=(file)
def log_file=(file) @log_file = file self[WEBDRIVER_PREFS[:log_file]] = file end
def port=(port)
def port=(port) self[WEBDRIVER_PREFS[:port]] = port end
def proxy=(proxy)
def proxy=(proxy) raise TypeError, "expected #{Proxy.name}, got #{proxy.inspect}:#{proxy.class}" unless proxy.is_a? Proxy case proxy.type when :manual self['network.proxy.type'] = 1 set_manual_proxy_preference 'ftp', proxy.ftp set_manual_proxy_preference 'http', proxy.http set_manual_proxy_preference 'ssl', proxy.ssl set_manual_proxy_preference 'socks', proxy.socks self['network.proxy.no_proxies_on'] = proxy.no_proxy || '' when :pac self['network.proxy.type'] = 2 self['network.proxy.autoconfig_url'] = proxy.pac when :auto_detect self['network.proxy.type'] = 4 else raise ArgumentError, "unsupported proxy type #{proxy.type}" end end
def read_model_prefs
def read_model_prefs return {} unless @model read_user_prefs(File.join(@model, 'user.js')) end
def read_user_prefs(path)
def read_user_prefs(path) prefs = {} return prefs unless File.exist?(path) File.read(path).split("\n").each do |line| next unless line =~ /user_pref\("([^"]+)"\s*,\s*(.+?)\);/ key = Regexp.last_match(1)&.strip value = Regexp.last_match(2)&.strip # wrap the value in an array to make it a valid JSON string. prefs[key] = JSON.parse("[#{value}]").first end prefs end
def set_manual_proxy_preference(key, value)
def set_manual_proxy_preference(key, value) return unless value host, port = value.to_s.split(':', 2) self["network.proxy.#{key}"] = host self["network.proxy.#{key}_port"] = Integer(port) if port end
def update_user_prefs_in(directory)
def update_user_prefs_in(directory) path = File.join(directory, 'user.js') prefs = read_user_prefs(path) prefs.merge! self.class::DEFAULT_PREFERENCES prefs.merge!(@additional_prefs) # If the user sets the home page, we should also start up there prefs['startup.homepage_welcome_url'] ||= prefs['browser.startup.homepage'] write_prefs prefs, path end
def write_prefs(prefs, path)
def write_prefs(prefs, path) File.open(path, 'w') do |file| prefs.each do |key, value| file.puts %{user_pref("#{key}", #{value.to_json});} end end end