├── .document ├── .github ├── dependabot.yml └── workflows │ ├── release.yml │ └── ruby.yml ├── .gitignore ├── .rspec ├── .rubocop.yml ├── .rubocop_todo.yml ├── .ruby-version ├── CHANGELOG.md ├── CODE_OF_CONDUCT.md ├── Dockerfile ├── Gemfile ├── Guardfile ├── LICENSE.txt ├── README.md ├── Rakefile ├── bin ├── console └── setup ├── docker-compose.yml ├── exe └── image_scraper ├── image_scraper.gemspec ├── lib ├── image_scraper.rb └── image_scraper │ ├── client.rb │ ├── railtie.rb │ ├── util.rb │ └── version.rb ├── sig └── image_scraper.rbs └── spec ├── cassettes └── ImageScraper_Client │ ├── _image_urls │ ├── handles_url_with_unescaped_spaces.yml │ ├── scrapes_absolute_paths.yml │ └── scrapes_relative_paths.yml │ ├── _page_images │ ├── handldes_image_urls_that_include_square_brackets.yml │ └── handles_unescaped_urls.yml │ ├── _stylesheet_images │ ├── handles_404s.yml │ ├── handles_stylesheet_image_with_a_relative_url.yml │ └── scrapes_stylesheet_images.yml │ ├── _stylesheets │ └── lists_relative_path_stylesheets.yml │ └── foo │ └── something.yml ├── image_scraper ├── client_spec.rb └── util_spec.rb ├── image_scraper_spec.rb ├── spec_helper.rb └── support ├── extra_whitespace.html ├── relative_image_url.css ├── relative_image_url.html ├── space in url.html ├── stylesheet_test.html ├── stylesheet_unescaped_image.html └── unescaped_image.css /.document: -------------------------------------------------------------------------------- 1 | lib/**/*.rb 2 | bin/* 3 | - 4 | features/**/*.feature 5 | LICENSE.txt 6 | -------------------------------------------------------------------------------- /.github/dependabot.yml: -------------------------------------------------------------------------------- 1 | # To get started with Dependabot version updates, you'll need to specify which 2 | # package ecosystems to update and where the package manifests are located. 3 | # Please see the documentation for all configuration options: 4 | # https://docs.github.com/github/administering-a-repository/configuration-options-for-dependency-updates 5 | 6 | version: 2 7 | updates: 8 | - package-ecosystem: "bundler" # See documentation for possible values 9 | directory: "/" # Location of package manifests 10 | schedule: 11 | interval: "weekly" 12 | -------------------------------------------------------------------------------- /.github/workflows/release.yml: -------------------------------------------------------------------------------- 1 | name: release 2 | on: 3 | push: 4 | branches: [master] 5 | jobs: 6 | release: 7 | runs-on: ubuntu-latest 8 | env: 9 | GEM_NAME: image_scraper 10 | steps: 11 | - uses: google-github-actions/release-please-action@v3 12 | id: release 13 | with: 14 | bump-minor-pre-major: true 15 | package-name: image_scraper 16 | release-type: ruby 17 | version-file: "lib/image_scraper/version.rb" 18 | - uses: actions/checkout@v3 19 | - name: install ruby 20 | if: "${{ steps.release.outputs.release_created }}" 21 | uses: ruby/setup-ruby@v1 22 | with: 23 | bundler-cache: true 24 | - name: bundle 25 | if: "${{ steps.release.outputs.release_created }}" 26 | run: | 27 | bundle config unset --local deployment 28 | bundle 29 | - name: publish gem 30 | if: "${{ steps.release.outputs.release_created }}" 31 | uses: dawidd6/action-publish-gem@v1 32 | with: 33 | api_key: "${{secrets.RUBYGEMS_API_KEY}}" 34 | github_token: "${{secrets.GITHUB_TOKEN}}" 35 | -------------------------------------------------------------------------------- /.github/workflows/ruby.yml: -------------------------------------------------------------------------------- 1 | name: ruby 2 | 3 | on: 4 | push: 5 | branches: 6 | - master 7 | pull_request: 8 | 9 | jobs: 10 | build-and-run-tests: 11 | runs-on: ubuntu-latest 12 | 13 | steps: 14 | - uses: actions/checkout@v3 15 | - name: Set up Ruby 16 | uses: ruby/setup-ruby@v1 17 | with: 18 | bundler-cache: true 19 | - name: Run tests 20 | run: bundle exec rspec spec 21 | - name: Run rubocop 22 | run: bundle exec rubocop 23 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | # rspec failure tracking 2 | .rspec_status 3 | 4 | /.bundle/ 5 | /.yardoc 6 | /Gemfile*lock 7 | /_yardoc/ 8 | /coverage/ 9 | /doc/ 10 | /pkg/ 11 | /spec/reports/ 12 | /tmp/ 13 | -------------------------------------------------------------------------------- /.rspec: -------------------------------------------------------------------------------- 1 | --color 2 | --require spec_helper 3 | -------------------------------------------------------------------------------- /.rubocop.yml: -------------------------------------------------------------------------------- 1 | inherit_from: .rubocop_todo.yml 2 | 3 | AllCops: 4 | SuggestExtensions: false 5 | NewCops: enable 6 | 7 | require: 8 | - rubocop-rspec 9 | 10 | Layout/SpaceAroundMethodCallOperator: 11 | Enabled: true 12 | Lint/RaiseException: 13 | Enabled: True 14 | Lint/StructNewOverride: 15 | Enabled: True 16 | Style/ExponentialNotation: 17 | Enabled: True 18 | Style/HashEachMethods: 19 | Enabled: True 20 | Style/HashTransformKeys: 21 | Enabled: True 22 | Style/HashTransformValues: 23 | Enabled: True 24 | Layout/EmptyLinesAroundAttributeAccessor: 25 | Enabled: True 26 | Lint/DeprecatedOpenSSLConstant: 27 | Enabled: True 28 | Lint/MixedRegexpCaptureTypes: 29 | Enabled: True 30 | Style/RedundantRegexpCharacterClass: 31 | Enabled: False 32 | Style/RedundantRegexpEscape: 33 | Enabled: False 34 | Style/SlicingWithRange: 35 | Enabled: True 36 | Style/AccessorGrouping: 37 | Enabled: True 38 | Style/BisectedAttrAccessor: 39 | Enabled: True 40 | Style/RedundantAssignment: 41 | Enabled: True 42 | Style/RedundantFetchBlock: 43 | Enabled: True 44 | -------------------------------------------------------------------------------- /.rubocop_todo.yml: -------------------------------------------------------------------------------- 1 | # This configuration was generated by 2 | # `rubocop --auto-gen-config` 3 | # on 2021-12-21 21:23:53 UTC using RuboCop version 1.23.0. 4 | # The point is for the user to remove these configuration records 5 | # one by one as the offenses are removed from the code base. 6 | # Note that changes in the inspected code, or installation of new 7 | # versions of RuboCop, may require this file to be generated again. 8 | 9 | # Offense count: 1 10 | # Configuration parameters: Include. 11 | # Include: **/*.gemspec 12 | Gemspec/RequiredRubyVersion: 13 | Exclude: 14 | - 'image_scraper.gemspec' 15 | 16 | # Offense count: 1 17 | # Cop supports --auto-correct. 18 | # Configuration parameters: EnforcedStyle. 19 | # SupportedStyles: empty_lines, no_empty_lines 20 | Layout/EmptyLinesAroundBlockBody: 21 | Exclude: 22 | - 'spec/image_scraper/util_spec.rb' 23 | 24 | # Offense count: 1 25 | Lint/DuplicateRescueException: 26 | Exclude: 27 | - 'lib/image_scraper/client.rb' 28 | 29 | # Offense count: 1 30 | # Cop supports --auto-correct. 31 | Lint/OrderedMagicComments: 32 | Exclude: 33 | - 'image_scraper.gemspec' 34 | 35 | # Offense count: 2 36 | # Configuration parameters: IgnoredMethods, CountRepeatedAttributes. 37 | Metrics/AbcSize: 38 | Max: 19 39 | 40 | # Offense count: 5 41 | # Configuration parameters: CountComments, CountAsOne, ExcludedMethods, IgnoredMethods. 42 | # IgnoredMethods: refine 43 | Metrics/BlockLength: 44 | Max: 120 45 | 46 | # Offense count: 5 47 | # Configuration parameters: CountComments, CountAsOne, ExcludedMethods, IgnoredMethods. 48 | Metrics/MethodLength: 49 | Max: 18 50 | 51 | # Offense count: 2 52 | RSpec/AnyInstance: 53 | Exclude: 54 | - 'spec/image_scraper/client_spec.rb' 55 | 56 | # Offense count: 6 57 | # Configuration parameters: Max. 58 | RSpec/ExampleLength: 59 | Exclude: 60 | - 'spec/image_scraper/client_spec.rb' 61 | - 'spec/image_scraper/util_spec.rb' 62 | 63 | # Offense count: 4 64 | RSpec/MultipleExpectations: 65 | Max: 5 66 | 67 | # Offense count: 1 68 | Security/Open: 69 | Exclude: 70 | - 'lib/image_scraper/client.rb' 71 | 72 | # Offense count: 1 73 | # Cop supports --auto-correct. 74 | Style/CaseLikeIf: 75 | Exclude: 76 | - 'lib/image_scraper/client.rb' 77 | 78 | # Offense count: 2 79 | # Configuration parameters: AllowedConstants. 80 | Style/Documentation: 81 | Exclude: 82 | - 'spec/**/*' 83 | - 'test/**/*' 84 | - 'lib/image_scraper/client.rb' 85 | - 'lib/image_scraper/util.rb' 86 | 87 | # Offense count: 1 88 | # Cop supports --auto-correct. 89 | # Configuration parameters: EnforcedStyle. 90 | # SupportedStyles: always, always_true, never 91 | Style/FrozenStringLiteralComment: 92 | Exclude: 93 | - 'Guardfile' 94 | 95 | # Offense count: 1 96 | # Cop supports --auto-correct. 97 | Style/IfUnlessModifier: 98 | Exclude: 99 | - 'image_scraper.gemspec' 100 | 101 | # Offense count: 1 102 | # Cop supports --auto-correct. 103 | # Configuration parameters: PreferredDelimiters. 104 | Style/PercentLiteralDelimiters: 105 | Exclude: 106 | - 'Guardfile' 107 | 108 | # Offense count: 8 109 | # Cop supports --auto-correct. 110 | # Configuration parameters: EnforcedStyle, ConsistentQuotesInMultiline. 111 | # SupportedStyles: single_quotes, double_quotes 112 | Style/StringLiterals: 113 | Exclude: 114 | - 'Guardfile' 115 | - 'spec/image_scraper/client_spec.rb' 116 | 117 | # Offense count: 2 118 | # Cop supports --auto-correct. 119 | # Configuration parameters: AllowHeredoc, AllowURI, URISchemes, IgnoreCopDirectives, IgnoredPatterns. 120 | # URISchemes: http, https 121 | Layout/LineLength: 122 | Max: 142 123 | -------------------------------------------------------------------------------- /.ruby-version: -------------------------------------------------------------------------------- 1 | 3.2.0 2 | -------------------------------------------------------------------------------- /CHANGELOG.md: -------------------------------------------------------------------------------- 1 | ## [Unreleased] 2 | 3 | ## [0.1.14](https://github.com/charlotte-ruby/image_scraper/compare/v0.1.13...v0.1.14) (2022-12-09) 4 | 5 | 6 | ### Bug Fixes 7 | 8 | * testing release please ([c722746](https://github.com/charlotte-ruby/image_scraper/commit/c72274671065e8329f879b9640a39fc5652bbbd7)) 9 | 10 | ## [0.1.13](https://github.com/charlotte-ruby/image_scraper/compare/v0.1.12...v0.1.13) (2022-11-25) 11 | 12 | 13 | ### Bug Fixes 14 | 15 | * bump to gems. bump ruby to 3.1.3 ([7ee3877](https://github.com/charlotte-ruby/image_scraper/commit/7ee38775bbc0e9684fe512c698a9caa1ecb3c07b)) 16 | 17 | ## [0.1.12](https://github.com/charlotte-ruby/image_scraper/compare/v0.1.11...v0.1.12) (2022-11-25) 18 | 19 | 20 | ### Bug Fixes 21 | 22 | * trying again lol ([9f55af5](https://github.com/charlotte-ruby/image_scraper/commit/9f55af55bb34d61a4a6dbf141998212229bf6ac9)) 23 | 24 | ## [0.1.11](https://github.com/charlotte-ruby/image_scraper/compare/v0.1.10...v0.1.11) (2022-11-25) 25 | 26 | 27 | ### Bug Fixes 28 | 29 | * bump version ([936f850](https://github.com/charlotte-ruby/image_scraper/commit/936f850b0f8f3d87607d28a5f3a4a088975b2ada)) 30 | 31 | ### [0.1.10](https://www.github.com/charlotte-ruby/image_scraper/compare/v0.1.7...v0.1.10) (2022-04-17) 32 | 33 | ### Bug Fixes 34 | 35 | * bump ruby ([3519010](https://github.com/charlotte-ruby/image_scraper/commit/351901036ed4b4b9432814ce05bcd5c67ae0c332)) 36 | 37 | ## [0.1.9] - 2022-02-21 38 | 39 | - deprecate jeweler in favor of bundler and rake tasks 40 | -------------------------------------------------------------------------------- /CODE_OF_CONDUCT.md: -------------------------------------------------------------------------------- 1 | # Contributor Covenant Code of Conduct 2 | 3 | ## Our Pledge 4 | 5 | We as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, religion, or sexual identity and orientation. 6 | 7 | We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community. 8 | 9 | ## Our Standards 10 | 11 | Examples of behavior that contributes to a positive environment for our community include: 12 | 13 | * Demonstrating empathy and kindness toward other people 14 | * Being respectful of differing opinions, viewpoints, and experiences 15 | * Giving and gracefully accepting constructive feedback 16 | * Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience 17 | * Focusing on what is best not just for us as individuals, but for the overall community 18 | 19 | Examples of unacceptable behavior include: 20 | 21 | * The use of sexualized language or imagery, and sexual attention or 22 | advances of any kind 23 | * Trolling, insulting or derogatory comments, and personal or political attacks 24 | * Public or private harassment 25 | * Publishing others' private information, such as a physical or email 26 | address, without their explicit permission 27 | * Other conduct which could reasonably be considered inappropriate in a 28 | professional setting 29 | 30 | ## Enforcement Responsibilities 31 | 32 | Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful. 33 | 34 | Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate. 35 | 36 | ## Scope 37 | 38 | This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event. 39 | 40 | ## Enforcement 41 | 42 | Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to the community leaders responsible for enforcement at matt@invalid8.com. All complaints will be reviewed and investigated promptly and fairly. 43 | 44 | All community leaders are obligated to respect the privacy and security of the reporter of any incident. 45 | 46 | ## Enforcement Guidelines 47 | 48 | Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct: 49 | 50 | ### 1. Correction 51 | 52 | **Community Impact**: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community. 53 | 54 | **Consequence**: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested. 55 | 56 | ### 2. Warning 57 | 58 | **Community Impact**: A violation through a single incident or series of actions. 59 | 60 | **Consequence**: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban. 61 | 62 | ### 3. Temporary Ban 63 | 64 | **Community Impact**: A serious violation of community standards, including sustained inappropriate behavior. 65 | 66 | **Consequence**: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban. 67 | 68 | ### 4. Permanent Ban 69 | 70 | **Community Impact**: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals. 71 | 72 | **Consequence**: A permanent ban from any sort of public interaction within the community. 73 | 74 | ## Attribution 75 | 76 | This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.0, 77 | available at https://www.contributor-covenant.org/version/2/0/code_of_conduct.html. 78 | 79 | Community Impact Guidelines were inspired by [Mozilla's code of conduct enforcement ladder](https://github.com/mozilla/diversity). 80 | 81 | [homepage]: https://www.contributor-covenant.org 82 | 83 | For answers to common questions about this code of conduct, see the FAQ at 84 | https://www.contributor-covenant.org/faq. Translations are available at https://www.contributor-covenant.org/translations. 85 | -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | FROM ruby:3.2-alpine 2 | 3 | WORKDIR /usr/src/app 4 | 5 | RUN apk update && \ 6 | apk add gcc gcompat git \ 7 | libxml2-dev libxslt-dev \ 8 | make musl-dev 9 | 10 | RUN mkdir -p lib/image_scraper 11 | COPY lib/image_scraper/version.rb ./lib/image_scraper/version.rb 12 | COPY .ruby-version image_scraper.gemspec Gemfile Gemfile.lock ./ 13 | 14 | RUN bundle install 15 | COPY . . 16 | 17 | CMD ["bundle", "exec", "rspec"] 18 | -------------------------------------------------------------------------------- /Gemfile: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | source 'http://rubygems.org' 4 | 5 | ruby File.read('.ruby-version').chomp 6 | 7 | gemspec 8 | 9 | gem 'rake', '~> 13.0' 10 | gem 'rspec', '~> 3.4' 11 | gem 'rubocop', '~> 1.21' 12 | 13 | group :development do 14 | gem 'bundler', '~> 2.3' 15 | gem 'guard-rspec', require: false 16 | gem 'pry' 17 | gem 'rubocop-rspec', require: false 18 | gem 'test-unit' 19 | gem 'vcr', '~> 6.0' 20 | gem 'webmock' 21 | end 22 | -------------------------------------------------------------------------------- /Guardfile: -------------------------------------------------------------------------------- 1 | # A sample Guardfile 2 | # More info at https://github.com/guard/guard#readme 3 | 4 | ## Uncomment and set this to only include directories you want to watch 5 | # directories %w(app lib config test spec features) \ 6 | # .select{|d| Dir.exist?(d) ? d : UI.warning("Directory #{d} does not exist")} 7 | 8 | ## Note: if you are using the `directories` clause above and you are not 9 | ## watching the project directory ('.'), then you will want to move 10 | ## the Guardfile to a watched dir and symlink it back, e.g. 11 | # 12 | # $ mkdir config 13 | # $ mv Guardfile config/ 14 | # $ ln -s config/Guardfile . 15 | # 16 | # and, you'll have to watch "config/Guardfile" instead of "Guardfile" 17 | 18 | # NOTE: The cmd option is now required due to the increasing number of ways 19 | # rspec may be run, below are examples of the most common uses. 20 | # * bundler: 'bundle exec rspec' 21 | # * bundler binstubs: 'bin/rspec' 22 | # * spring: 'bin/rspec' (This will use spring if running and you have 23 | # installed the spring binstubs per the docs) 24 | # * zeus: 'zeus rspec' (requires the server to be started separately) 25 | # * 'just' rspec: 'rspec' 26 | 27 | guard :rspec, cmd: "bundle exec rspec" do 28 | require "guard/rspec/dsl" 29 | dsl = Guard::RSpec::Dsl.new(self) 30 | 31 | # Feel free to open issues for suggestions and improvements 32 | 33 | # RSpec files 34 | rspec = dsl.rspec 35 | watch(rspec.spec_helper) { rspec.spec_dir } 36 | watch(rspec.spec_support) { rspec.spec_dir } 37 | watch(rspec.spec_files) 38 | 39 | # Ruby files 40 | ruby = dsl.ruby 41 | dsl.watch_spec_files_for(ruby.lib_files) 42 | 43 | # Rails files 44 | rails = dsl.rails(view_extensions: %w(erb haml slim)) 45 | dsl.watch_spec_files_for(rails.app_files) 46 | dsl.watch_spec_files_for(rails.views) 47 | 48 | watch(rails.controllers) do |m| 49 | [ 50 | rspec.spec.call("routing/#{m[1]}_routing"), 51 | rspec.spec.call("controllers/#{m[1]}_controller"), 52 | rspec.spec.call("acceptance/#{m[1]}") 53 | ] 54 | end 55 | 56 | # Rails config changes 57 | watch(rails.spec_helper) { rspec.spec_dir } 58 | watch(rails.routes) { "#{rspec.spec_dir}/routing" } 59 | watch(rails.app_controller) { "#{rspec.spec_dir}/controllers" } 60 | 61 | # Capybara features specs 62 | watch(rails.view_dirs) { |m| rspec.spec.call("features/#{m[1]}") } 63 | watch(rails.layouts) { |m| rspec.spec.call("features/#{m[1]}") } 64 | 65 | # Turnip features and steps 66 | watch(%r{^spec/acceptance/(.+)\.feature$}) 67 | watch(%r{^spec/acceptance/steps/(.+)_steps\.rb$}) do |m| 68 | Dir[File.join("**/#{m[1]}.feature")][0] || "spec/acceptance" 69 | end 70 | end 71 | -------------------------------------------------------------------------------- /LICENSE.txt: -------------------------------------------------------------------------------- 1 | Copyright (c) 2011 John McAliley 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining 4 | a copy of this software and associated documentation files (the 5 | "Software"), to deal in the Software without restriction, including 6 | without limitation the rights to use, copy, modify, merge, publish, 7 | distribute, sublicense, and/or sell copies of the Software, and to 8 | permit persons to whom the Software is furnished to do so, subject to 9 | the following conditions: 10 | 11 | The above copyright notice and this permission notice shall be 12 | included in all copies or substantial portions of the Software. 13 | 14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE 18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION 19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION 20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 21 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # ImageScraper 2 | [![ruby](https://github.com/charlotte-ruby/image_scraper/actions/workflows/ruby.yml/badge.svg)](https://github.com/charlotte-ruby/image_scraper/actions/workflows/ruby.yml) 3 | 4 | Simple utility that pulls image URLs from web page 5 | ## Installation 6 | 7 | Install in your application's Gemfile or as a standalone gem: 8 | 9 | ```ruby 10 | gem 'image_scraper' 11 | ``` 12 | 13 | And then execute: 14 | 15 | ``` 16 | $ bundle install 17 | ``` 18 | 19 | Standalone install: 20 | 21 | ``` 22 | $ gem install image_scraper 23 | ``` 24 | 25 | ## Usage 26 | 27 | ```ruby 28 | options = { 29 | convert_to_absolute_url: true, 30 | include_css_images: true # convert any relative images to absolute urls. 31 | include_css_data_images: true # convert any data images (data:image/gif;base64....) 32 | } 33 | 34 | image_scraper = ImageScraper::Client.new("http://www.rubygems.org", options) 35 | image_scraper.image_urls 36 | 37 | # => ["https://rubygems.org/assets/github_icon.png"", "https://rubygems.org/sponsors.png"] 38 | ``` 39 | 40 | ### CLI 41 | 42 | ``` 43 | $ image_scraper https://unsplash.com | head -n 2 44 | https://images.unsplash.com/photo-1471897488648 45 | https://images.unsplash.com/photo-1590073242678 46 | ``` 47 | 48 | ## Development 49 | 50 | After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake spec` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment. 51 | 52 | To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`. Releases are done via Github Actions. 53 | 54 | If you prefer to use docker: 55 | 56 | ``` 57 | docker-compose build 58 | docker-compose run app 59 | ``` 60 | 61 | Once inside the container, run the tests and you'll see output similar to this: 62 | 63 | ``` 64 | /usr/src/app # bundle exec rspec 65 | ........................ 66 | 67 | Finished in 0.54303 seconds (files took 0.95976 seconds to load) 68 | 24 examples, 0 failures 69 | ``` 70 | 71 | ## Contributing 72 | 73 | Bug reports and pull requests are welcome on GitHub at https://github.com/charlotte-ruby/image_scraper. This project is intended to be a safe, welcoming space for collaboration, and contributors are expected to adhere to the [code of conduct](https://github.com/charlotte-ruby/image_scraper/blob/master/CODE_OF_CONDUCT.md). 74 | 75 | - Check out the latest master to make sure the feature hasn't been implemented or the bug hasn't been fixed yet 76 | - Check out the issue tracker to make sure someone already hasn't requested it and/or contributed it 77 | - Fork the project 78 | - Start a feature/bugfix branch 79 | - Commit and push until you are happy with your contribution 80 | - Make sure to add tests for it. This is important so I don't break it in a future version unintentionally. 81 | - Please try not to mess with the Rakefile, version, or history. If you want to have your own version, or is otherwise necessary, that is fine, but please isolate to its own commit so I can cherry-pick around it. 82 | 83 | ## Copyright 84 | 85 | Copyright (c) 2011 John McAliley. See LICENSE.txt for 86 | further details. 87 | 88 | ## Code of Conduct 89 | 90 | Everyone interacting in the ImageScraper project's codebases, issue trackers, chat rooms and mailing lists is expected to follow the [code of conduct](https://github.com/charlotte-ruby/image_scraper/blob/master/CODE_OF_CONDUCT.md). 91 | -------------------------------------------------------------------------------- /Rakefile: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require 'bundler/gem_tasks' 4 | require 'rspec/core/rake_task' 5 | 6 | RSpec::Core::RakeTask.new(:spec) 7 | 8 | require 'rubocop/rake_task' 9 | 10 | RuboCop::RakeTask.new 11 | 12 | task default: %i[spec rubocop] 13 | -------------------------------------------------------------------------------- /bin/console: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | require 'bundler/setup' 5 | require 'image_scraper' 6 | 7 | # You can add fixtures and/or initialization code here to make experimenting 8 | # with your gem easier. You can also use a different console, if you like. 9 | 10 | # (If you use this, don't forget to add pry to your Gemfile!) 11 | # require "pry" 12 | # Pry.start 13 | 14 | require 'irb' 15 | IRB.start(__FILE__) 16 | -------------------------------------------------------------------------------- /bin/setup: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env bash 2 | set -euo pipefail 3 | IFS=$'\n\t' 4 | set -vx 5 | 6 | bundle install 7 | 8 | # Do any other automated setup that you need to do here 9 | -------------------------------------------------------------------------------- /docker-compose.yml: -------------------------------------------------------------------------------- 1 | version: "3" 2 | services: 3 | app: 4 | build: . 5 | command: /bin/sh 6 | volumes: 7 | - .:/usr/src/app 8 | -------------------------------------------------------------------------------- /exe/image_scraper: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # frozen_string_literal: true 3 | 4 | require 'image_scraper' 5 | 6 | url = ARGV[0].to_s 7 | 8 | unless url.length.positive? 9 | puts 'usage: image_scraper ' 10 | exit 1 11 | end 12 | 13 | image_scraper = ImageScraper::Client.new(url) 14 | scraped_urls = image_scraper.image_urls 15 | 16 | puts scraped_urls * "\n" if scraped_urls.count.positive? 17 | -------------------------------------------------------------------------------- /image_scraper.gemspec: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require_relative 'lib/image_scraper/version' 4 | 5 | Gem::Specification.new do |spec| 6 | spec.name = 'image_scraper' 7 | spec.version = ImageScraper::VERSION 8 | spec.authors = ['John McAliley', 'Matt McMahand'] 9 | spec.email = ['john.mcaliley@gmail.com', 'matt@invalid8.com'] 10 | spec.summary = 'Simple utility to pull image urls from web page' 11 | spec.description = spec.summary 12 | spec.homepage = 'http://github.com/charlotte-ruby/image_scraper' 13 | spec.license = 'MIT' 14 | spec.required_ruby_version = '>= 2.6.0' 15 | 16 | spec.metadata['homepage_uri'] = spec.homepage 17 | spec.metadata['source_code_uri'] = spec.homepage 18 | spec.metadata['changelog_uri'] = File.join(spec.homepage, 'blob/master/CHANGELOG.md') 19 | 20 | spec.metadata['rubygems_mfa_required'] = 'true' 21 | 22 | begin 23 | files = (result = `git ls-files -z`.split "\0").empty? ? Dir['**/*'] : result 24 | rescue StandardError 25 | files = Dir['**/*'] 26 | end 27 | 28 | # Specify which files should be added to the gem when it is released. 29 | spec.files = files.grep_v(/^\A(?:(?:test|spec|features|.git|sig))/) 30 | 31 | spec.bindir = 'exe' 32 | spec.executables = spec.files.grep(%r{\Aexe/}) { |f| File.basename(f) } 33 | spec.require_paths = ['lib'] 34 | 35 | spec.add_dependency 'css_parser', '~> 1.11' 36 | spec.add_dependency 'nokogiri', '~> 1.13' 37 | end 38 | -------------------------------------------------------------------------------- /lib/image_scraper.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require 'open-uri' 4 | require 'nokogiri' 5 | 6 | require_relative 'image_scraper/client' 7 | require_relative 'image_scraper/railtie' if defined?(Rails::Railtie) 8 | require_relative 'image_scraper/util' 9 | require_relative 'image_scraper/version' 10 | 11 | module ImageScraper 12 | class Error < StandardError; end 13 | end 14 | -------------------------------------------------------------------------------- /lib/image_scraper/client.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require 'net/http' 4 | require 'uri' 5 | require 'cgi' 6 | 7 | module ImageScraper 8 | class Client 9 | USER_AGENT = 'Mozilla/5.0 (Macintosh)' 10 | 11 | attr_accessor :url, :convert_to_absolute_url, :include_css_images, :include_css_data_images, :doc 12 | attr_reader :uri, :error 13 | 14 | def initialize(url, options = {}) 15 | defaults = { convert_to_absolute_url: true, include_css_images: true, include_css_data_images: false } 16 | options.merge!(defaults) 17 | 18 | @url = url 19 | @uri = Util.convert_to_uri(url) 20 | 21 | @convert_to_absolute_url = options[:convert_to_absolute_url] 22 | @include_css_images = options[:include_css_images] 23 | @include_css_data_images = options[:include_css_data_images] 24 | 25 | begin 26 | html = fetch(@uri) 27 | rescue StandardError => e 28 | @error = e 29 | html = nil 30 | end 31 | 32 | @doc = html ? Nokogiri::HTML(html, nil, 'UTF-8') : nil 33 | end 34 | 35 | def fetch(url, limit = 10) 36 | raise ArgumentError, 'HTTP redirect too deep' if limit.zero? 37 | 38 | uri = Util.convert_to_uri(url) 39 | 40 | return false unless uri 41 | 42 | result = Net::HTTP.start(uri.host, uri.port, use_ssl: uri.port == 443) do |http| 43 | request = Net::HTTP::Get.new(uri, { 'User-Agent' => USER_AGENT }) 44 | response = http.request request 45 | 46 | case response 47 | when Net::HTTPSuccess then response 48 | when Net::HTTPRedirection then fetch(response['location'], limit - 1) 49 | else 50 | response.error! 51 | end 52 | end 53 | 54 | if result.is_a? Net::HTTPOK 55 | result.body 56 | elsif result.is_a? String 57 | result 58 | end 59 | end 60 | 61 | def image_urls 62 | images = page_images 63 | images += stylesheet_images if include_css_images 64 | images.sort.uniq 65 | end 66 | 67 | def cleanup_src_value(text) 68 | text.to_s.strip! 69 | text.gsub!(' ', '%20') 70 | 71 | # escape characters that CGI::escape doesn't get 72 | text.gsub(/([{}|\^\[\]\@`])/) { |s| s } 73 | end 74 | 75 | def page_images 76 | return [] if doc.to_s.empty? 77 | 78 | doc.xpath('//img').collect do |e| 79 | src = cleanup_src_value(e['src']) 80 | next if src.empty? 81 | 82 | if convert_to_absolute_url 83 | Util.absolute_url(@uri.to_s, src) 84 | else 85 | src 86 | end 87 | end.compact 88 | end 89 | 90 | def fetch_css(url) 91 | begin 92 | file = URI.open(url) 93 | rescue StandardError 94 | return '' 95 | end 96 | 97 | begin 98 | css = file.string 99 | rescue StandardError 100 | css = File.read(file) 101 | rescue StandardError 102 | return '' 103 | end 104 | 105 | css.unpack('C*').pack('U*') 106 | end 107 | 108 | def stylesheet_images 109 | images = [] 110 | 111 | stylesheets.each do |stylesheet| 112 | css = fetch_css(stylesheet) 113 | 114 | next unless css.to_s.length.positive? 115 | 116 | images += css.scan(/url\((.*?)\)/).collect do |image_url| 117 | image_url = Util.cleanup_url(image_url[0]) 118 | image_url = image_url.gsub(/([{}|\^\[\]\@`])/) { |s| CGI.escape(s) } # escape characters that URI.escape doesn't get 119 | if image_url.include?('data:image') && @include_css_data_images 120 | image_url 121 | else 122 | @convert_to_absolute_url ? Util.absolute_url(stylesheet, image_url) : image_url 123 | end 124 | end 125 | end 126 | images.compact 127 | end 128 | 129 | def stylesheets 130 | return [] if doc.to_s.empty? 131 | 132 | doc.xpath('//link[@rel="stylesheet"]').collect do |stylesheet| 133 | Util.absolute_url(@uri.to_s, Util.cleanup_url(stylesheet['href'])) 134 | end.compact 135 | end 136 | end 137 | end 138 | -------------------------------------------------------------------------------- /lib/image_scraper/railtie.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module ImageScraper 4 | class Railtie < Rails::Railtie 5 | end 6 | end 7 | -------------------------------------------------------------------------------- /lib/image_scraper/util.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module ImageScraper 4 | module Util 5 | def self.absolute_url(url, asset = nil) 6 | # TODO: - what happens when an index redirect occurs? 7 | # Example: 'http://example.com/about' specified as url 8 | # 'style.css' specified as asset 9 | # url redirects to 'http://example.com/about/' 10 | # and serves http://example.com/about/index.html 11 | # which then links to the relative asset path 'style.css' 12 | # based on original url (http://example.com/about), 13 | # self.absolute_url gives 14 | # 'http://example.com/style.css 15 | # but should get: 16 | # 'http://example.com/about/style.css 17 | 18 | URI.parse(url).merge(URI.parse(asset.to_s)).to_s 19 | rescue StandardError 20 | nil 21 | end 22 | 23 | def self.convert_to_uri(url) 24 | if url.is_a?(URI::HTTP) 25 | url 26 | else 27 | url = url.strip 28 | url = "http://#{url}" unless url.include?('://') 29 | url = url.gsub(' ', '%20') if url.include?(' ') 30 | 31 | begin 32 | URI.parse(url) 33 | rescue URI::InvalidURIError 34 | nil 35 | end 36 | end 37 | end 38 | 39 | def self.domain(url) 40 | uri = URI.parse(url) 41 | "#{uri.scheme}://#{uri.host}" 42 | rescue StandardError 43 | print('domain error') 44 | nil 45 | end 46 | 47 | def self.path(url) 48 | URI.parse(url).path 49 | rescue StandardError 50 | nil 51 | end 52 | 53 | def self.strip_backslashes(image_url) 54 | image_url.gsub('\\', '') 55 | end 56 | 57 | def self.strip_quotes(image_url) 58 | image_url.gsub("'", '').gsub('"', '') 59 | end 60 | 61 | def self.chomp(image_url) 62 | image_url.gsub(/\s/, '') 63 | end 64 | 65 | def self.cleanup_url(image_url) 66 | ImageScraper::Util.chomp( 67 | ImageScraper::Util.strip_quotes( 68 | ImageScraper::Util.strip_backslashes(image_url || '') 69 | ) 70 | ) 71 | end 72 | end 73 | end 74 | -------------------------------------------------------------------------------- /lib/image_scraper/version.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | module ImageScraper 4 | VERSION = '0.1.14' 5 | end 6 | -------------------------------------------------------------------------------- /sig/image_scraper.rbs: -------------------------------------------------------------------------------- 1 | module ImageScraper 2 | VERSION: String 3 | # See the writing guide of rbs: https://github.com/ruby/rbs#guides 4 | end 5 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_image_urls/handles_url_with_unescaped_spaces.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: https://raw.github.com/syoder/image_scraper/stylesheet_fix/test/resources/space%20in%20url.html 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | User-Agent: 11 | - Mozilla/5.0 (Macintosh) 12 | Accept-Encoding: 13 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 14 | Accept: 15 | - "*/*" 16 | Host: 17 | - raw.github.com 18 | response: 19 | status: 20 | code: 301 21 | message: Moved Permanently 22 | headers: 23 | Connection: 24 | - keep-alive 25 | Content-Length: 26 | - '0' 27 | Location: 28 | - https://raw.githubusercontent.com/syoder/image_scraper/stylesheet_fix/test/resources/space%20in%20url.html 29 | Accept-Ranges: 30 | - bytes 31 | Date: 32 | - Sat, 11 Jul 2020 02:48:32 GMT 33 | Via: 34 | - 1.1 varnish 35 | Age: 36 | - '0' 37 | X-Served-By: 38 | - cache-fty21351-FTY 39 | X-Cache: 40 | - MISS 41 | X-Cache-Hits: 42 | - '0' 43 | Vary: 44 | - Accept-Encoding 45 | X-Fastly-Request-Id: 46 | - 1938ad6b24faae4219882b22499a0884f3fcfac5 47 | body: 48 | encoding: UTF-8 49 | string: '' 50 | recorded_at: Sat, 11 Jul 2020 02:48:32 GMT 51 | - request: 52 | method: get 53 | uri: https://raw.githubusercontent.com/syoder/image_scraper/stylesheet_fix/test/resources/space%20in%20url.html 54 | body: 55 | encoding: US-ASCII 56 | string: '' 57 | headers: 58 | User-Agent: 59 | - Mozilla/5.0 (Macintosh) 60 | Accept-Encoding: 61 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 62 | Accept: 63 | - "*/*" 64 | Host: 65 | - raw.githubusercontent.com 66 | response: 67 | status: 68 | code: 200 69 | message: OK 70 | headers: 71 | Connection: 72 | - keep-alive 73 | Content-Length: 74 | - '66' 75 | Cache-Control: 76 | - max-age=300 77 | Content-Security-Policy: 78 | - default-src 'none'; style-src 'unsafe-inline'; sandbox 79 | Content-Type: 80 | - text/plain; charset=utf-8 81 | Etag: 82 | - W/"97a67c7d07a8aa3a39a54347e00f2eb4a7b9a18080d07f38fa16538b9187a65a" 83 | Strict-Transport-Security: 84 | - max-age=31536000 85 | X-Content-Type-Options: 86 | - nosniff 87 | X-Frame-Options: 88 | - deny 89 | X-Xss-Protection: 90 | - 1; mode=block 91 | Via: 92 | - 1.1 varnish 93 | - 1.1 varnish (Varnish/6.0) 94 | X-Github-Request-Id: 95 | - B330:6FFF:8A3AC:A38C4:5F09287F 96 | Accept-Ranges: 97 | - bytes 98 | Date: 99 | - Sat, 11 Jul 2020 02:48:32 GMT 100 | X-Served-By: 101 | - cache-fty21323-FTY 102 | X-Cache: 103 | - MISS, MISS 104 | X-Cache-Hits: 105 | - 0, 0 106 | X-Timer: 107 | - S1594435712.380265,VS0,VE118 108 | Vary: 109 | - Authorization,Accept-Encoding 110 | Access-Control-Allow-Origin: 111 | - "*" 112 | X-Fastly-Request-Id: 113 | - 3ce7f01ccb996f373840a16c0c3b1756eba30c9a 114 | Expires: 115 | - Sat, 11 Jul 2020 02:53:32 GMT 116 | Source-Age: 117 | - '0' 118 | body: 119 | encoding: ASCII-8BIT 120 | string: | 121 | 122 | 123 | 124 | 125 | 126 | recorded_at: Sat, 11 Jul 2020 02:48:32 GMT 127 | recorded_with: VCR 6.0.0 128 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_page_images/handldes_image_urls_that_include_square_brackets.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: http://google.com/ 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | User-Agent: 11 | - Mozilla/5.0 (Macintosh) 12 | Accept-Encoding: 13 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 14 | Accept: 15 | - "*/*" 16 | Host: 17 | - google.com 18 | response: 19 | status: 20 | code: 301 21 | message: Moved Permanently 22 | headers: 23 | Location: 24 | - http://www.google.com/ 25 | Content-Type: 26 | - text/html; charset=UTF-8 27 | Date: 28 | - Sat, 11 Jul 2020 03:19:08 GMT 29 | Expires: 30 | - Mon, 10 Aug 2020 03:19:08 GMT 31 | Cache-Control: 32 | - public, max-age=2592000 33 | Server: 34 | - gws 35 | Content-Length: 36 | - '219' 37 | X-Xss-Protection: 38 | - '0' 39 | X-Frame-Options: 40 | - SAMEORIGIN 41 | body: 42 | encoding: UTF-8 43 | string: "\n301 44 | Moved\n

301 Moved

\nThe document has moved\nhere.\r\n\r\n" 46 | recorded_at: Sat, 11 Jul 2020 03:19:08 GMT 47 | - request: 48 | method: get 49 | uri: http://www.google.com/ 50 | body: 51 | encoding: US-ASCII 52 | string: '' 53 | headers: 54 | User-Agent: 55 | - Mozilla/5.0 (Macintosh) 56 | Accept-Encoding: 57 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 58 | Accept: 59 | - "*/*" 60 | Host: 61 | - www.google.com 62 | response: 63 | status: 64 | code: 200 65 | message: OK 66 | headers: 67 | Date: 68 | - Sat, 11 Jul 2020 03:19:08 GMT 69 | Expires: 70 | - "-1" 71 | Cache-Control: 72 | - private, max-age=0 73 | Content-Type: 74 | - text/html; charset=UTF-8 75 | P3p: 76 | - CP="This is not a P3P policy! See g.co/p3phelp for more info." 77 | Server: 78 | - gws 79 | Content-Length: 80 | - '15676' 81 | X-Xss-Protection: 82 | - '0' 83 | X-Frame-Options: 84 | - SAMEORIGIN 85 | Set-Cookie: 86 | - 1P_JAR=2020-07-11-03; expires=Mon, 10-Aug-2020 03:19:08 GMT; path=/; domain=.google.com; 87 | Secure 88 | - NID=204=NKuzVtzV2dKPsOc7_z9bjNoIRhrN5kM9LQVZji5cqnUzPnDf6oMK-ritO8eMSwcgEJfqzbjCFUynwrJg1XMrJTjn_8t3hvFXqZhqvCCVAgBSQiN1wFE7ZP2JGvXkvahwYVo6QkfpnbwfRK3uMGL2Wk5tD4KbnopvxHAQ9qCHrGA; 89 | expires=Sun, 10-Jan-2021 03:19:08 GMT; path=/; domain=.google.com; HttpOnly 90 | body: 91 | encoding: ASCII-8BIT 92 | string: !binary |- 93 | <!doctype html><html itemscope="" itemtype="http://schema.org/WebPage" lang="en"><head><meta content="Search the world's information, including webpages, images, videos and more. Google has many special features to help you find exactly what you're looking for." name="description"><meta content="noodp" name="robots"><meta content="text/html; charset=UTF-8" http-equiv="Content-Type"><meta content="/images/branding/googleg/1x/googleg_standard_color_128dp.png" itemprop="image"><title>Google</title><script nonce="Z0rwtxMNtMtj5GlIvyrd2g==">(function(){window.google={kEI:'rC8JX_T-BcHl_Qb__rjoCA',kEXPI:'0,202123,3,4,32,1151585,5662,730,224,5105,206,3204,10,1226,364,1499,611,93,113,383,246,5,1184,170,407,96,145,194,486,5,361,2184,221,13,80,106,116,3,519,531,91,49,131,2,10,275,100,379,11,1121251,1197754,395,329118,1294,12383,4855,32691,15248,861,28690,9188,8384,4858,1362,283,9007,3022,4746,7,11026,1808,4020,978,4788,1,3144,5295,2676,298,873,37,1178,2384,3378,3645,1142,10164,3222,4516,2778,921,2275,10,2794,1593,1279,2212,530,149,561,542,840,517,1466,57,157,4100,312,1135,1,3,2063,606,1839,186,1775,143,377,1947,1030,1,1198,93,328,1284,24,2919,2247,473,1339,1795,3219,1989,857,6,6068,6286,4455,641,2449,2459,1226,1742,3654,1275,108,3408,907,2,941,2614,2398,7469,1,839,1337,666,432,3,346,1200,865,1,372,3545,706,149,189,3313,502,1,957,1028,158,1,2093,1992,1997,83,1018,73,570,4,1528,17,417,861,1009,224,1003,15,265,874,405,1259,580,24,174,69,1570,279,331,41,462,40,242,56,917,10,42,214,762,440,963,461,460,120,761,438,23,15,198,1232,538,2297,292,1973,86,3,1493,877,910,1426,69,94,211,1362,948,1305,209,1009,2,286,940,7,3,837,57,599,981,433,554,2,177,60,6,32,215,106,91,235,1,2,82,108,91,171,610,1556,44,72,298,200,2,402,193,160,1,756,67,175,35,6,281,689,97,276,57,31,177,169,114,618,552,122,106,254,105,126,3,407,1469,1,732,346,14,337,92,2,17,370,523,1,5791657,3377,5997322,2801217,549,333,444,1,2,80,1,900,896,1,9,2,2551,1,748,141,59,736,563,1,4265,1,1,1,1,137,1,879,9,305,2874,147,105,2,16,10,10,133,2,4,23961100',kBL:'hOv_'};google.sn='webhp';google.kHL='en';})();(function(){google.lc=[];google.li=0;google.getEI=function(a){for(var c;a&&(!a.getAttribute||!(c=a.getAttribute("eid")));)a=a.parentNode;return c||google.kEI};google.getLEI=function(a){for(var c=null;a&&(!a.getAttribute||!(c=a.getAttribute("leid")));)a=a.parentNode;return c};google.ml=function(){return null};google.time=function(){return Date.now()};google.log=function(a,c,b,d,g){if(b=google.logUrl(a,c,b,d,g)){a=new Image;var e=google.lc,f=google.li;e[f]=a;a.onerror=a.onload=a.onabort=function(){delete e[f]};google.vel&&google.vel.lu&&google.vel.lu(b);a.src=b;google.li=f+1}};google.logUrl=function(a,c,b,d,g){var e="",f=google.ls||"";b||-1!=c.search("&ei=")||(e="&ei="+google.getEI(d),-1==c.search("&lei=")&&(d=google.getLEI(d))&&(e+="&lei="+d));d="";!b&&google.cshid&&-1==c.search("&cshid=")&&"slh"!=a&&(d="&cshid="+google.cshid);b=b||"/"+(g||"gen_204")+"?atyp=i&ct="+a+"&cad="+c+e+f+"&zx="+google.time()+d;/^http:/i.test(b)&&"https:"==window.location.protocol&&(google.ml(Error("a"),!1,{src:b,glmm:1}),b="");return b};}).call(this);(function(){google.y={};google.x=function(a,b){if(a)var c=a.id;else{do c=Math.random();while(google.y[c])}google.y[c]=[a,b];return!1};google.lm=[];google.plm=function(a){google.lm.push.apply(google.lm,a)};google.lq=[];google.load=function(a,b,c){google.lq.push([[a],b,c])};google.loadAll=function(a,b){google.lq.push([a,b])};}).call(this);google.f={};(function(){
document.documentElement.addEventListener("submit",function(b){var a;if(a=b.target){var c=a.getAttribute("data-submitfalse");a="1"==c||"q"==c&&!a.elements.q.value?!0:!1}else a=!1;a&&(b.preventDefault(),b.stopPropagation())},!0);document.documentElement.addEventListener("click",function(b){var a;a:{for(a=b.target;a&&a!=document.documentElement;a=a.parentElement)if("A"==a.tagName){a="1"==a.getAttribute("data-nohref");break a}a=!1}a&&b.preventDefault()},!0);}).call(this);
var a=window.location,b=a.href.indexOf("#");if(0<=b){var c=a.href.substring(b+1);/(^|&)q=/.test(c)&&-1==c.indexOf("#")&&a.replace("/search?"+c.replace(/(^|&)fp=[^&]*/g,"")+"&cad=h")};</script><style>#gb{font:13px/27px Arial,sans-serif;height:30px}#gbz,#gbg{position:absolute;white-space:nowrap;top:0;height:30px;z-index:1000}#gbz{left:0;padding-left:4px}#gbg{right:0;padding-right:5px}#gbs{background:transparent;position:absolute;top:-999px;visibility:hidden;z-index:998;right:0}.gbto #gbs{background:#fff}#gbx3,#gbx4{background-color:#2d2d2d;background-image:none;_background-image:none;background-position:0 -138px;background-repeat:repeat-x;border-bottom:1px solid #000;font-size:24px;height:29px;_height:30px;opacity:1;filter:alpha(opacity=100);position:absolute;top:0;width:100%;z-index:990}#gbx3{left:0}#gbx4{right:0}#gbb{position:relative}#gbbw{left:0;position:absolute;top:30px;width:100%}.gbtcb{position:absolute;visibility:hidden}#gbz .gbtcb{right:0}#gbg .gbtcb{left:0}.gbxx{display:none !important}.gbxo{opacity:0 !important;filter:alpha(opacity=0) !important}.gbm{position:absolute;z-index:999;top:-999px;visibility:hidden;text-align:left;border:1px solid #bebebe;background:#fff;-moz-box-shadow:-1px 1px 1px rgba(0,0,0,.2);-webkit-box-shadow:0 2px 4px rgba(0,0,0,.2);box-shadow:0 2px 4px rgba(0,0,0,.2)}.gbrtl .gbm{-moz-box-shadow:1px 1px 1px rgba(0,0,0,.2)}.gbto .gbm,.gbto #gbs{top:29px;visibility:visible}#gbz .gbm{left:0}#gbg .gbm{right:0}.gbxms{background-color:#ccc;display:block;position:absolute;z-index:1;top:-1px;left:-2px;right:-2px;bottom:-2px;opacity:.4;-moz-border-radius:3px;filter:progid:DXImageTransform.Microsoft.Blur(pixelradius=5);*opacity:1;*top:-2px;*left:-5px;*right:5px;*bottom:4px;-ms-filter:"progid:DXImageTransform.Microsoft.Blur(pixelradius=5)";opacity:1\0/;top:-4px\0/;left:-6px\0/;right:5px\0/;bottom:4px\0/}.gbma{position:relative;top:-1px;border-style:solid dashed dashed;border-color:transparent;border-top-color:#c0c0c0;display:-moz-inline-box;display:inline-block;font-size:0;height:0;line-height:0;width:0;border-width:3px 3px 0;padding-top:1px;left:4px}#gbztms1,#gbi4m1,#gbi4s,#gbi4t{zoom:1}.gbtc,.gbmc,.gbmcc{display:block;list-style:none;margin:0;padding:0}.gbmc{background:#fff;padding:10px 0;position:relative;z-index:2;zoom:1}.gbt{position:relative;display:-moz-inline-box;display:inline-block;line-height:27px;padding:0;vertical-align:top}.gbt{*display:inline}.gbto{box-shadow:0 2px 4px rgba(0,0,0,.2);-moz-box-shadow:0 2px 4px rgba(0,0,0,.2);-webkit-box-shadow:0 2px 4px rgba(0,0,0,.2)}.gbzt,.gbgt{cursor:pointer;display:block;text-decoration:none !important}span#gbg6,span#gbg4{cursor:default}.gbts{border-left:1px solid transparent;border-right:1px solid transparent;display:block;*display:inline-block;padding:0 5px;position:relative;z-index:1000}.gbts{*display:inline}.gbzt .gbts{display:inline;zoom:1}.gbto .gbts{background:#fff;border-color:#bebebe;color:#36c;padding-bottom:1px;padding-top:2px}.gbz0l .gbts{color:#fff;font-weight:bold}.gbtsa{padding-right:9px}#gbz .gbzt,#gbz .gbgt,#gbg .gbgt{color:#ccc!important}.gbtb2{display:block;border-top:2px solid transparent}.gbto .gbzt .gbtb2,.gbto .gbgt .gbtb2{border-top-width:0}.gbtb .gbts{background:url(https://ssl.gstatic.com/gb/images/b_8d5afc09.png);_background:url(https://ssl.gstatic.com/gb/images/b8_3615d64d.png);background-position:-27px -22px;border:0;font-size:0;padding:29px 0 0;*padding:27px 0 0;width:1px}.gbzt:hover,.gbzt:focus,.gbgt-hvr,.gbgt:focus{background-color:#4c4c4c;background-image:none;_background-image:none;background-position:0 -102px;background-repeat:repeat-x;outline:none;text-decoration:none !important}.gbpdjs .gbto .gbm{min-width:99%}.gbz0l .gbtb2{border-top-color:#dd4b39!important}#gbi4s,#gbi4s1{font-weight:bold}#gbg6.gbgt-hvr,#gbg6.gbgt:focus{background-color:transparent;background-image:none}.gbg4a{font-size:0;line-height:0}.gbg4a .gbts{padding:27px 5px 0;*padding:25px 5px 0}.gbto .gbg4a .gbts{padding:29px 5px 1px;*padding:27px 5px 1px}#gbi4i,#gbi4id{left:5px;border:0;height:24px;position:absolute;top:1px;width:24px}.gbto #gbi4i,.gbto #gbi4id{top:3px}.gbi4p{display:block;width:24px}#gbi4id{background-position:-44px -101px}#gbmpid{background-position:0 0}#gbmpi,#gbmpid{border:none;display:inline-block;height:48px;width:48px}#gbmpiw{display:inline-block;line-height:9px;padding-left:20px;margin-top:10px;position:relative}#gbmpi,#gbmpid,#gbmpiw{*display:inline}#gbg5{font-size:0}#gbgs5{padding:5px !important}.gbto #gbgs5{padding:7px 5px 6px !important}#gbi5{background:url(https://ssl.gstatic.com/gb/images/b_8d5afc09.png);_background:url(https://ssl.gstatic.com/gb/images/b8_3615d64d.png);background-position:0 0;display:block;font-size:0;height:17px;width:16px}.gbto #gbi5{background-position:-6px -22px}.gbn .gbmt,.gbn .gbmt:visited,.gbnd .gbmt,.gbnd .gbmt:visited{color:#dd8e27 !important}.gbf .gbmt,.gbf .gbmt:visited{color:#900 !important}.gbmt,.gbml1,.gbmlb,.gbmt:visited,.gbml1:visited,.gbmlb:visited{color:#36c !important;text-decoration:none !important}.gbmt,.gbmt:visited{display:block}.gbml1,.gbmlb,.gbml1:visited,.gbmlb:visited{display:inline-block;margin:0 10px}.gbml1,.gbmlb,.gbml1:visited,.gbmlb:visited{*display:inline}.gbml1,.gbml1:visited{padding:0 10px}.gbml1-hvr,.gbml1:focus{outline:none;text-decoration:underline !important}#gbpm .gbml1{display:inline;margin:0;padding:0;white-space:nowrap}.gbmlb,.gbmlb:visited{line-height:27px}.gbmlb-hvr,.gbmlb:focus{outline:none;text-decoration:underline !important}.gbmlbw{color:#ccc;margin:0 10px}.gbmt{padding:0 20px}.gbmt:hover,.gbmt:focus{background:#eee;cursor:pointer;outline:0 solid black;text-decoration:none !important}.gbm0l,.gbm0l:visited{color:#000 !important;font-weight:bold}.gbmh{border-top:1px solid #bebebe;font-size:0;margin:10px 0}#gbd4 .gbmc{background:#f5f5f5;padding-top:0}#gbd4 .gbsbic::-webkit-scrollbar-track:vertical{background-color:#f5f5f5;margin-top:2px}#gbmpdv{background:#fff;border-bottom:1px solid #bebebe;-moz-box-shadow:0 2px 4px rgba(0,0,0,.12);-o-box-shadow:0 2px 4px rgba(0,0,0,.12);-webkit-box-shadow:0 2px 4px rgba(0,0,0,.12);box-shadow:0 2px 4px rgba(0,0,0,.12);position:relative;z-index:1}#gbd4 .gbmh{margin:0}.gbmtc{padding:0;margin:0;line-height:27px}.GBMCC:last-child:after,#GBMPAL:last-child:after{content:'\0A\0A';white-space:pre;position:absolute}#gbmps{*zoom:1}#gbd4 .gbpc,#gbmpas .gbmt{line-height:17px}#gbd4 .gbpgs .gbmtc{line-height:27px}#gbd4 .gbmtc{border-bottom:1px solid #bebebe}#gbd4 .gbpc{display:inline-block;margin:16px 0 10px;padding-right:50px;vertical-align:top}#gbd4 .gbpc{*display:inline}.gbpc .gbps,.gbpc .gbps2{display:block;margin:0 20px}#gbmplp.gbps{margin:0 10px}.gbpc .gbps{color:#000;font-weight:bold}.gbpc .gbpd{margin-bottom:5px}.gbpd .gbmt,.gbpd .gbps{color:#666 !important}.gbpd .gbmt{opacity:.4;filter:alpha(opacity=40)}.gbps2{color:#666;display:block}.gbp0{display:none}.gbp0 .gbps2{font-weight:bold}#gbd4 .gbmcc{margin-top:5px}.gbpmc{background:#fef9db}.gbpmc .gbpmtc{padding:10px 20px}#gbpm{border:0;*border-collapse:collapse;border-spacing:0;margin:0;white-space:normal}#gbpm .gbpmtc{border-top:none;color:#000 !important;font:11px Arial,sans-serif}#gbpms{*white-space:nowrap}.gbpms2{font-weight:bold;white-space:nowrap}#gbmpal{*border-collapse:collapse;border-spacing:0;border:0;margin:0;white-space:nowrap;width:100%}.gbmpala,.gbmpalb{font:13px Arial,sans-serif;line-height:27px;padding:10px 20px 0;white-space:nowrap}.gbmpala{padding-left:0;text-align:left}.gbmpalb{padding-right:0;text-align:right}#gbmpasb .gbps{color:#000}#gbmpal .gbqfbb{margin:0 20px}.gbp0 .gbps{*display:inline}a.gbiba{margin:8px 20px 10px}.gbmpiaw{display:inline-block;padding-right:10px;margin-bottom:6px;margin-top:10px}.gbxv{visibility:hidden}.gbmpiaa{display:block;margin-top:10px}.gbmpia{border:none;display:block;height:48px;width:48px}.gbmpnw{display:inline-block;height:auto;margin:10px 0;vertical-align:top}
.gbqfb,.gbqfba,.gbqfbb{-moz-border-radius:2px;-webkit-border-radius:2px;border-radius:2px;cursor:default !important;display:inline-block;font-weight:bold;height:29px;line-height:29px;min-width:54px;*min-width:70px;padding:0 8px;text-align:center;text-decoration:none !important;-moz-user-select:none;-webkit-user-select:none}.gbqfb:focus,.gbqfba:focus,.gbqfbb:focus{border:1px solid #4d90fe;-moz-box-shadow:inset 0 0 0 1px rgba(255, 255, 255, 0.5);-webkit-box-shadow:inset 0 0 0 1px rgba(255, 255, 255, 0.5);box-shadow:inset 0 0 0 1px rgba(255, 255, 255, 0.5);outline:none}.gbqfb-hvr:focus,.gbqfba-hvr:focus,.gbqfbb-hvr:focus{-webkit-box-shadow:inset 0 0 0 1px #fff,0 1px 1px rgba(0,0,0,.1);-moz-box-shadow:inset 0 0 0 1px #fff,0 1px 1px rgba(0,0,0,.1);box-shadow:inset 0 0 0 1px #fff,0 1px 1px rgba(0,0,0,.1)}.gbqfb-no-focus:focus{border:1px solid #3079ed;-moz-box-shadow:none;-webkit-box-shadow:none;box-shadow:none}.gbqfb-hvr,.gbqfba-hvr,.gbqfbb-hvr{-webkit-box-shadow:0 1px 1px rgba(0,0,0,.1);-moz-box-shadow:0 1px 1px rgba(0,0,0,.1);box-shadow:0 1px 1px rgba(0,0,0,.1)}.gbqfb::-moz-focus-inner,.gbqfba::-moz-focus-inner,.gbqfbb::-moz-focus-inner{border:0}.gbqfba,.gbqfbb{border:1px solid #dcdcdc;border-color:rgba(0,0,0,.1);color:#444 !important;font-size:11px}.gbqfb{background-color:#4d90fe;background-image:-webkit-gradient(linear,left top,left bottom,from(#4d90fe),to(#4787ed));background-image:-webkit-linear-gradient(top,#4d90fe,#4787ed);background-image:-moz-linear-gradient(top,#4d90fe,#4787ed);background-image:-ms-linear-gradient(top,#4d90fe,#4787ed);background-image:-o-linear-gradient(top,#4d90fe,#4787ed);background-image:linear-gradient(top,#4d90fe,#4787ed);filter:progid:DXImageTransform.Microsoft.gradient(startColorStr='#4d90fe',EndColorStr='#4787ed');border:1px solid #3079ed;color:#fff!important;margin:0 0}.gbqfb-hvr{border-color:#2f5bb7}.gbqfb-hvr:focus{border-color:#2f5bb7}.gbqfb-hvr,.gbqfb-hvr:focus{background-color:#357ae8;background-image:-webkit-gradient(linear,left top,left bottom,from(#4d90fe),to(#357ae8));background-image:-webkit-linear-gradient(top,#4d90fe,#357ae8);background-image:-moz-linear-gradient(top,#4d90fe,#357ae8);background-image:-ms-linear-gradient(top,#4d90fe,#357ae8);background-image:-o-linear-gradient(top,#4d90fe,#357ae8);background-image:linear-gradient(top,#4d90fe,#357ae8)}.gbqfb:active{background-color:inherit;-webkit-box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.3);-moz-box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.3);box-shadow:inset 0 1px 2px rgba(0, 0, 0, 0.3)}.gbqfba{background-color:#f5f5f5;background-image:-webkit-gradient(linear,left top,left bottom,from(#f5f5f5),to(#f1f1f1));background-image:-webkit-linear-gradient(top,#f5f5f5,#f1f1f1);background-image:-moz-linear-gradient(top,#f5f5f5,#f1f1f1);background-image:-ms-linear-gradient(top,#f5f5f5,#f1f1f1);background-image:-o-linear-gradient(top,#f5f5f5,#f1f1f1);background-image:linear-gradient(top,#f5f5f5,#f1f1f1);filter:progid:DXImageTransform.Microsoft.gradient(startColorStr='#f5f5f5',EndColorStr='#f1f1f1')}.gbqfba-hvr,.gbqfba-hvr:active{background-color:#f8f8f8;background-image:-webkit-gradient(linear,left top,left bottom,from(#f8f8f8),to(#f1f1f1));background-image:-webkit-linear-gradient(top,#f8f8f8,#f1f1f1);background-image:-moz-linear-gradient(top,#f8f8f8,#f1f1f1);background-image:-ms-linear-gradient(top,#f8f8f8,#f1f1f1);background-image:-o-linear-gradient(top,#f8f8f8,#f1f1f1);background-image:linear-gradient(top,#f8f8f8,#f1f1f1);filter:progid:DXImageTransform.Microsoft.gradient(startColorStr='#f8f8f8',EndColorStr='#f1f1f1')}.gbqfbb{background-color:#fff;background-image:-webkit-gradient(linear,left top,left bottom,from(#fff),to(#fbfbfb));background-image:-webkit-linear-gradient(top,#fff,#fbfbfb);background-image:-moz-linear-gradient(top,#fff,#fbfbfb);background-image:-ms-linear-gradient(top,#fff,#fbfbfb);background-image:-o-linear-gradient(top,#fff,#fbfbfb);background-image:linear-gradient(top,#fff,#fbfbfb);filter:progid:DXImageTransform.Microsoft.gradient(startColorStr='#ffffff',EndColorStr='#fbfbfb')}.gbqfbb-hvr,.gbqfbb-hvr:active{background-color:#fff;background-image:-webkit-gradient(linear,left top,left bottom,from(#fff),to(#f8f8f8));background-image:-webkit-linear-gradient(top,#fff,#f8f8f8);background-image:-moz-linear-gradient(top,#fff,#f8f8f8);background-image:-ms-linear-gradient(top,#fff,#f8f8f8);background-image:-o-linear-gradient(top,#fff,#f8f8f8);background-image:linear-gradient(top,#fff,#f8f8f8);filter:progid:DXImageTransform.Microsoft.gradient(startColorStr='#ffffff',EndColorStr='#f8f8f8')}.gbqfba-hvr,.gbqfba-hvr:active,.gbqfbb-hvr,.gbqfbb-hvr:active{border-color:#c6c6c6;-webkit-box-shadow:0 1px 1px rgba(0,0,0,.1);-moz-box-shadow:0 1px 1px rgba(0,0,0,.1);box-shadow:0 1px 1px rgba(0,0,0,.1);color:#222 !important}.gbqfba:active,.gbqfbb:active{-webkit-box-shadow:inset 0 1px 2px rgba(0,0,0,.1);-moz-box-shadow:inset 0 1px 2px rgba(0,0,0,.1);box-shadow:inset 0 1px 2px rgba(0,0,0,.1)}
#gbmpas{max-height:220px}#gbmm{max-height:530px}.gbsb{-webkit-box-sizing:border-box;display:block;position:relative;*zoom:1}.gbsbic{overflow:auto}.gbsbis .gbsbt,.gbsbis .gbsbb{-webkit-mask-box-image:-webkit-gradient(linear,left top,right top,color-stop(0,rgba(0,0,0,.1)),color-stop(.5,rgba(0,0,0,.8)),color-stop(1,rgba(0,0,0,.1)));left:0;margin-right:0;opacity:0;position:absolute;width:100%}.gbsb .gbsbt:after,.gbsb .gbsbb:after{content:"";display:block;height:0;left:0;position:absolute;width:100%}.gbsbis .gbsbt{background:-webkit-gradient(linear,left top,left bottom,from(rgba(0,0,0,.2)),to(rgba(0,0,0,0)));background-image:-webkit-linear-gradient(top,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:-moz-linear-gradient(top,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:-ms-linear-gradient(top,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:-o-linear-gradient(top,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:linear-gradient(top,rgba(0,0,0,.2),rgba(0,0,0,0));height:6px;top:0}.gbsb .gbsbt:after{border-top:1px solid #ebebeb;border-color:rgba(0,0,0,.3);top:0}.gbsb .gbsbb{-webkit-mask-box-image:-webkit-gradient(linear,left top,right top,color-stop(0,rgba(0,0,0,.1)),color-stop(.5,rgba(0,0,0,.8)),color-stop(1,rgba(0,0,0,.1)));background:-webkit-gradient(linear,left bottom,left top,from(rgba(0,0,0,.2)),to(rgba(0,0,0,0)));background-image:-webkit-linear-gradient(bottom,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:-moz-linear-gradient(bottom,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:-ms-linear-gradient(bottom,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:-o-linear-gradient(bottom,rgba(0,0,0,.2),rgba(0,0,0,0));background-image:linear-gradient(bottom,rgba(0,0,0,.2),rgba(0,0,0,0));bottom:0;height:4px}.gbsb .gbsbb:after{border-bottom:1px solid #ebebeb;border-color:rgba(0,0,0,.3);bottom:0}
</style><style>body,td,a,p,.h{font-family:arial,sans-serif}body{margin:0;overflow-y:scroll}#gog{padding:3px 8px 0}td{line-height:.8em}.gac_m td{line-height:17px}form{margin-bottom:20px}.h{color:#36c}.q{color:#00c}.ts td{padding:0}.ts{border-collapse:collapse}em{font-weight:bold;font-style:normal}.lst{height:25px;width:496px}.gsfi,.lst{font:18px arial,sans-serif}.gsfs{font:17px arial,sans-serif}.ds{display:inline-box;display:inline-block;margin:3px 0 4px;margin-left:4px}input{font-family:inherit}body{background:#fff;color:#000}a{color:#11c;text-decoration:none}a:hover,a:active{text-decoration:underline}.fl a{color:#36c}a:visited{color:#551a8b}.sblc{padding-top:5px}.sblc a{display:block;margin:2px 0;margin-left:13px;font-size:11px}.lsbb{background:#eee;border:solid 1px;border-color:#ccc #999 #999 #ccc;height:30px}.lsbb{display:block}#fll a{display:inline-block;margin:0 12px}.lsb{background:url(/images/nav_logo229.png) 0 -261px repeat-x;border:none;color:#000;cursor:pointer;height:30px;margin:0;outline:0;font:15px arial,sans-serif;vertical-align:top}.lsb:active{background:#ccc}.lst:focus{outline:none}</style><script nonce="Z0rwtxMNtMtj5GlIvyrd2g==">(function(){try{/*

 Copyright The Closure Library Authors.
 SPDX-License-Identifier: Apache-2.0
*/
var e=this||self;var aa=function(a,b,c,d){d=d||{};d._sn=["cfg",b,c].join(".");window.gbar.logger.ml(a,d)};var g=window.gbar=window.gbar||{},h=window.gbar.i=window.gbar.i||{},ba;function _tvn(a,b){a=parseInt(a,10);return isNaN(a)?b:a}function _tvf(a,b){a=parseFloat(a);return isNaN(a)?b:a}function _tvv(a){return!!a}function p(a,b,c){(c||g)[a]=b}g.bv={n:_tvn("2",0),r:"",f:".66.41.",e:"1300102,3700302,3700697",m:_tvn("1",1)};
function ca(a,b,c){var d="on"+b;if(a.addEventListener)a.addEventListener(b,c,!1);else if(a.attachEvent)a.attachEvent(d,c);else{var f=a[d];a[d]=function(){var k=f.apply(this,arguments),m=c.apply(this,arguments);return void 0==k?m:void 0==m?k:m&&k}}}var da=function(a){return function(){return g.bv.m==a}},ea=da(1),fa=da(2);p("sb",ea);p("kn",fa);h.a=_tvv;h.b=_tvf;h.c=_tvn;h.i=aa;var r=window.gbar.i.i;var t=function(){},ha=function(){},ka=function(a){var b=new Image,c=ia;b.onerror=b.onload=b.onabort=function(){try{delete ja[c]}catch(d){}};ja[c]=b;b.src=a;ia=c+1},ja=[],ia=0;p("logger",{il:ha,ml:t,log:ka});var u=window.gbar.logger;var v={},la={},w=[],ma=h.b("0.1",.1),na=h.a("1",!0),oa=function(a,b){w.push([a,b])},pa=function(a,b){v[a]=b},qa=function(a){return a in v},x={},A=function(a,b){x[a]||(x[a]=[]);x[a].push(b)},B=function(a){A("m",a)},ra=function(a,b){var c=document.createElement("script");c.src=a;c.async=na;Math.random()<ma&&(c.onerror=function(){c.onerror=null;t(Error("Bundle load failed: name="+(b||"UNK")+" url="+a))});(document.getElementById("xjsc")||document.getElementsByTagName("body")[0]||
document.getElementsByTagName("head")[0]).appendChild(c)},D=function(a){for(var b=0,c;(c=w[b])&&c[0]!=a;++b);!c||c[1].l||c[1].s||(c[1].s=!0,sa(2,a),c[1].url&&ra(c[1].url,a),c[1].libs&&C&&C(c[1].libs))},ta=function(a){A("gc",a)},ua=null,va=function(a){ua=a},sa=function(a,b,c){if(ua){a={t:a,b:b};if(c)for(var d in c)a[d]=c[d];try{ua(a)}catch(f){}}};p("mdc",v);p("mdi",la);p("bnc",w);p("qGC",ta);p("qm",B);p("qd",x);p("lb",D);p("mcf",pa);p("bcf",oa);p("aq",A);p("mdd","");
p("has",qa);p("trh",va);p("tev",sa);if(h.a("m;/_/scs/abc-static/_/js/k=gapi.gapi.en.yyhByYeMTAc.O/am=AAY/d=1/ct=zgms/rs=AHpOoo-O470EQdZ-4tpWpppyTQmeOEUv-g/m=__features__")){var F=function(a,b){return wa?a||b:b},xa=h.a("1"),ya=h.a(""),za=h.a(""),wa=h.a(""),Aa=window.gapi=F(window.gapi,{}),Ba=function(a,b){var c=function(){g.dgl(a,b)};xa?B(c):(A("gl",c),D("gl"))},Ca={},Da=function(a){a=a.split(":");for(var b;(b=a.pop())&&Ca[b];);return!b},C=function(a){function b(){for(var c=a.split(":"),d=0,f;f=c[d];++d)Ca[f]=1;for(c=0;d=w[c];++c)d=d[1],(f=d.libs)&&!d.l&&d.i&&
Da(f)&&d.i()}g.dgl(a,b)},G=window.___jsl=F(window.___jsl,{});G.h=F(G.h,"m;/_/scs/abc-static/_/js/k=gapi.gapi.en.yyhByYeMTAc.O/am=AAY/d=1/ct=zgms/rs=AHpOoo-O470EQdZ-4tpWpppyTQmeOEUv-g/m=__features__");G.ms=F(G.ms,"https://apis.google.com");G.m=F(G.m,"");G.l=F(G.l,[]);G.dpo=F(G.dpo,"");xa||w.push(["gl",{url:"//ssl.gstatic.com/gb/js/abc/glm_e7bb39a7e1a24581ff4f8d199678b1b9.js"}]);var Ea={pu:ya,sh:"",si:za,hl:"en"};v.gl=Ea;wa?Aa.load||p("load",Ba,Aa):p("load",Ba,Aa);p("dgl",Ba);p("agl",Da);h.o=xa};var Fa=h.b("0.1",.001),Ga=0;
function _mlToken(a,b){try{if(1>Ga){Ga++;var c=a;b=b||{};var d=encodeURIComponent,f=["//www.google.com/gen_204?atyp=i&zx=",(new Date).getTime(),"&jexpid=",d("28834"),"&srcpg=",d("prop=1"),"&jsr=",Math.round(1/Fa),"&ogev=",d("rC8JX7mSB-aa_Qa1nYGACw"),"&ogf=",g.bv.f,"&ogrp=",d(""),"&ogv=",d("319703744.0"),"&oggv="+d("es_plusone_gc_20200601.0_p0"),"&ogd=",d("com"),"&ogc=",d("USA"),"&ogl=",d("en")];b._sn&&(b._sn=
"og."+b._sn);for(var k in b)f.push("&"),f.push(d(k)),f.push("="),f.push(d(b[k]));f.push("&emsg=");f.push(d(c.name+":"+c.message));var m=f.join("");Ha(m)&&(m=m.substr(0,2E3));var n=m;var l=window.gbar.logger._aem(a,n);ka(l)}}catch(q){}}var Ha=function(a){return 2E3<=a.length},Ia=function(a,b){return b};function Ja(a){t=a;p("_itl",Ha,u);p("_aem",Ia,u);p("ml",t,u);a={};v.er=a}h.a("")?Ja(function(a){throw a;}):h.a("1")&&Math.random()<Fa&&Ja(_mlToken);var _E="left",Ka=h.a(""),J=function(a,b){var c=a.className;H(a,b)||(a.className+=(""!=c?" ":"")+b)},K=function(a,b){var c=a.className;b=new RegExp("\\s?\\b"+b+"\\b");c&&c.match(b)&&(a.className=c.replace(b,""))},H=function(a,b){b=new RegExp("\\b"+b+"\\b");a=a.className;return!(!a||!a.match(b))},La=function(a,b){H(a,b)?K(a,b):J(a,b)},Ma=function(a,b){a[b]=function(c){var d=arguments;g.qm(function(){a[b].apply(this,d)})}},Na=function(){return"1"},
Oa=function(a){a=["//www.gstatic.com","/og/_/js/d=1/k=","og.og2.en_US.b5xEmr8PQlw.O","/rt=j/m=",a,"/rs=","AA2YrTuV-r3qg-4QMlf6a49KSE-5MJf7bw"];Ka&&a.push("?host=www.gstatic.com&bust=og.og2.en_US.mEGIp6pG9H4.DU");a=a.join("");ra(a)};p("ca",J);p("cr",K);p("cc",H);h.k=J;h.l=K;h.m=H;h.n=La;h.p=Oa;h.q=Ma;h.r=Na;var Pa=["gb_71","gb_155"],Qa;function Ra(a){Qa=a}function Sa(a){var b=Qa&&!a.href.match(/.*\/accounts\/ClearSID[?]/)&&encodeURIComponent(Qa());b&&(a.href=a.href.replace(/([?&]continue=)[^&]*/,"$1"+b))}function Ta(a){window.gApplication&&(a.href=window.gApplication.getTabUrl(a.href))}function Ua(a){try{var b=(document.forms[0].q||"").value;b&&(a.href=a.href.replace(/([?&])q=[^&]*|$/,function(c,d){return(d||"&")+"q="+encodeURIComponent(b)}))}catch(c){r(c,"sb","pq")}}
var Va=function(){for(var a=[],b=0,c;c=Pa[b];++b)(c=document.getElementById(c))&&a.push(c);return a},Wa=function(){var a=Va();return 0<a.length?a[0]:null},Xa=function(){return document.getElementById("gb_70")},L={},M={},Ya={},N={},O=void 0,cb=function(a,b){try{var c=document.getElementById("gb");J(c,"gbpdjs");P();Za(document.getElementById("gb"))&&J(c,"gbrtl");if(b&&b.getAttribute){var d=b.getAttribute("aria-owns");if(d.length){var f=document.getElementById(d);if(f){var k=b.parentNode;if(O==d)O=void 0,
K(k,"gbto");else{if(O){var m=document.getElementById(O);if(m&&m.getAttribute){var n=m.getAttribute("aria-owner");if(n.length){var l=document.getElementById(n);l&&l.parentNode&&K(l.parentNode,"gbto")}}}$a(f)&&ab(f);O=d;J(k,"gbto")}}}}B(function(){g.tg(a,b,!0)});bb(a)}catch(q){r(q,"sb","tg")}},db=function(a){B(function(){g.close(a)})},eb=function(a){B(function(){g.rdd(a)})},Za=function(a){var b,c=document.defaultView;c&&c.getComputedStyle?(a=c.getComputedStyle(a,""))&&(b=a.direction):b=a.currentStyle?
a.currentStyle.direction:a.style.direction;return"rtl"==b},gb=function(a,b,c){if(a)try{var d=document.getElementById("gbd5");if(d){var f=d.firstChild,k=f.firstChild,m=document.createElement("li");m.className=b+" gbmtc";m.id=c;a.className="gbmt";m.appendChild(a);if(k.hasChildNodes()){c=[["gbkc"],["gbf","gbe","gbn"],["gbkp"],["gbnd"]];d=0;var n=k.childNodes.length;f=!1;for(var l=-1,q=0,E;E=c[q];q++){for(var U=0,I;I=E[U];U++){for(;d<n&&H(k.childNodes[d],I);)d++;if(I==b){k.insertBefore(m,k.childNodes[d]||
null);f=!0;break}}if(f){if(d+1<k.childNodes.length){var V=k.childNodes[d+1];H(V.firstChild,"gbmh")||fb(V,E)||(l=d+1)}else if(0<=d-1){var W=k.childNodes[d-1];H(W.firstChild,"gbmh")||fb(W,E)||(l=d)}break}0<d&&d+1<n&&d++}if(0<=l){var y=document.createElement("li"),z=document.createElement("div");y.className="gbmtc";z.className="gbmt gbmh";y.appendChild(z);k.insertBefore(y,k.childNodes[l])}g.addHover&&g.addHover(a)}else k.appendChild(m)}}catch(Eb){r(Eb,"sb","al")}},fb=function(a,b){for(var c=b.length,
d=0;d<c;d++)if(H(a,b[d]))return!0;return!1},hb=function(a,b,c){gb(a,b,c)},ib=function(a,b){gb(a,"gbe",b)},jb=function(){B(function(){g.pcm&&g.pcm()})},kb=function(){B(function(){g.pca&&g.pca()})},lb=function(a,b,c,d,f,k,m,n,l,q){B(function(){g.paa&&g.paa(a,b,c,d,f,k,m,n,l,q)})},mb=function(a,b){L[a]||(L[a]=[]);L[a].push(b)},nb=function(a,b){M[a]||(M[a]=[]);M[a].push(b)},ob=function(a,b){Ya[a]=b},pb=function(a,b){N[a]||(N[a]=[]);N[a].push(b)},bb=function(a){a.preventDefault&&a.preventDefault();a.returnValue=
!1;a.cancelBubble=!0},qb=null,ab=function(a,b){P();if(a){rb(a,"Opening&hellip;");Q(a,!0);b="undefined"!=typeof b?b:1E4;var c=function(){sb(a)};qb=window.setTimeout(c,b)}},tb=function(a){P();a&&(Q(a,!1),rb(a,""))},sb=function(a){try{P();var b=a||document.getElementById(O);b&&(rb(b,"This service is currently unavailable.%1$sPlease try again later.","%1$s"),Q(b,!0))}catch(c){r(c,"sb","sdhe")}},rb=function(a,b,c){if(a&&b){var d=$a(a);if(d){if(c){d.textContent="";b=b.split(c);c=0;for(var f;f=b[c];c++){var k=document.createElement("div");
k.innerHTML=f;d.appendChild(k)}}else d.innerHTML=b;Q(a,!0)}}},Q=function(a,b){(b=void 0!==b?b:!0)?J(a,"gbmsgo"):K(a,"gbmsgo")},$a=function(a){for(var b=0,c;c=a.childNodes[b];b++)if(H(c,"gbmsg"))return c},P=function(){qb&&window.clearTimeout(qb)},ub=function(a){var b="inner"+a;a="offset"+a;return window[b]?window[b]:document.documentElement&&document.documentElement[a]?document.documentElement[a]:0},vb=function(){return!1},wb=function(){return!!O};p("so",Wa);p("sos",Va);p("si",Xa);p("tg",cb);
p("close",db);p("rdd",eb);p("addLink",hb);p("addExtraLink",ib);p("pcm",jb);p("pca",kb);p("paa",lb);p("ddld",ab);p("ddrd",tb);p("dderr",sb);p("rtl",Za);p("op",wb);p("bh",L);p("abh",mb);p("dh",M);p("adh",nb);p("ch",N);p("ach",pb);p("eh",Ya);p("aeh",ob);ba=h.a("")?Ta:Ua;p("qs",ba);p("setContinueCb",Ra);p("pc",Sa);p("bsy",vb);h.d=bb;h.j=ub;var xb={};v.base=xb;w.push(["m",{url:"//ssl.gstatic.com/gb/js/sem_68cc05bdcad222c35f3fcebdb7ead60d.js"}]);g.sg={c:"1"};p("wg",{rg:{}});var yb={tiw:h.c("15000",0),tie:h.c("30000",0)};v.wg=yb;var zb={thi:h.c("10000",0),thp:h.c("180000",0),tho:h.c("5000",0),tet:h.b("0.5",0)};v.wm=zb;if(h.a("1")){var Ab=h.a("");w.push(["gc",{auto:Ab,url:"//ssl.gstatic.com/gb/js/abc/gci_91f30755d6a6b787dcc2a4062e6e9824.js",libs:"googleapis.client:plusone:gapi.iframes"}]);var Bb={version:"gci_91f30755d6a6b787dcc2a4062e6e9824.js",index:"",lang:"en"};v.gc=Bb;var Cb=function(a){window.googleapis&&window.iframes?a&&a():(a&&ta(a),D("gc"))};p("lGC",Cb);h.a("1")&&p("lPWF",Cb)};window.__PVT="";if(h.a("1")&&h.a("1")){var Db=function(a){Cb(function(){A("pw",a);D("pw")})};p("lPW",Db);w.push(["pw",{url:"//ssl.gstatic.com/gb/js/abc/pwm_45f73e4df07a0e388b0fa1f3d30e7280.js"}]);var Fb=[],Gb=function(a){Fb[0]=a},Hb=function(a,b){b=b||{};b._sn="pw";t(a,b)},Ib={signed:Fb,elog:Hb,base:"https://plusone.google.com/u/0",loadTime:(new Date).getTime()};v.pw=Ib;var Jb=function(a,b){var c=b.split(".");b=function(){var m=arguments;a(function(){for(var n=g,l=0,q=c.length-1;l<q;++l)n=n[c[l]];n[c[l]].apply(n,m)})};for(var d=g,f=0,k=c.length-1;f<
k;++f)d=d[c[f]]=d[c[f]]||{};return d[c[f]]=b};Jb(Db,"pw.clk");Jb(Db,"pw.hvr");p("su",Gb,g.pw)};var Kb=[1,2,3,4,5,6,9,10,11,13,14,28,29,30,34,35,37,38,39,40,41,42,43,48,49,500];var Lb=h.b("0.001",1E-4),Mb=h.b("1",1),Nb=!1,Ob=!1;if(h.a("1")){var Pb=Math.random();Pb<Lb&&(Nb=!0);Pb<Mb&&(Ob=!0)}var R=null;
function Qb(a,b){var c=Lb,d=Nb;var f=a;if(!R){R={};for(var k=0;k<Kb.length;k++){var m=Kb[k];R[m]=!0}}if(f=!!R[f])c=Mb,d=Ob;if(d){d=encodeURIComponent;if(g.rp){var n=g.rp();n="-1"!=n?n:""}else n="";f=(new Date).getTime();k=d("28834");m=d("rC8JX7mSB-aa_Qa1nYGACw");var l=g.bv.f,q=d("1");n=d(n);c=Math.round(1/c);var E=d("319703744.0"),U="&oggv="+d("es_plusone_gc_20200601.0_p0"),I=d("com"),V=d("en"),W=
d("USA");var y=0;h.a("")&&(y|=1);h.a("")&&(y|=2);h.a("")&&(y|=4);a=["//www.google.com/gen_204?atyp=i&zx=",f,"&oge=",a,"&ogex=",k,"&ogev=",m,"&ogf=",l,"&ogp=",q,"&ogrp=",n,"&ogsr=",c,"&ogv=",E,U,"&ogd=",I,"&ogl=",V,"&ogc=",W,"&ogus=",y];if(b){"ogw"in b&&(a.push("&ogw="+b.ogw),delete b.ogw);f=[];for(z in b)0!=f.length&&f.push(","),f.push(Rb(z)),f.push("."),f.push(Rb(b[z]));var z=f.join("");""!=z&&(a.push("&ogad="),a.push(d(z)))}ka(a.join(""))}}
function Rb(a){"number"==typeof a&&(a+="");return"string"==typeof a?a.replace(".","%2E").replace(",","%2C"):a}ha=Qb;p("il",ha,u);var Sb={};v.il=Sb;var Tb=function(a,b,c,d,f,k,m,n,l,q){B(function(){g.paa(a,b,c,d,f,k,m,n,l,q)})},Ub=function(){B(function(){g.prm()})},Vb=function(a){B(function(){g.spn(a)})},Wb=function(a){B(function(){g.sps(a)})},Xb=function(a){B(function(){g.spp(a)})},Yb={"27":"https://lh3.googleusercontent.com/ogw/default-user=s24","27":"https://lh3.googleusercontent.com/ogw/default-user=s24","27":"https://lh3.googleusercontent.com/ogw/default-user=s24"},Zb=function(a){return(a=Yb[a])||"https://lh3.googleusercontent.com/ogw/default-user=s24"},
$b=function(){B(function(){g.spd()})};p("spn",Vb);p("spp",Xb);p("sps",Wb);p("spd",$b);p("paa",Tb);p("prm",Ub);mb("gbd4",Ub);
if(h.a("")){var ac={d:h.a(""),e:"",sanw:h.a(""),p:"https://lh3.googleusercontent.com/ogw/default-user=s96",cp:"1",xp:h.a("1"),mg:"%1$s (delegated)",md:"%1$s (default)",mh:"220",s:"1",pp:Zb,ppl:h.a(""),ppa:h.a(""),
ppm:"Google+ page"};v.prf=ac};var S,bc,T,cc,X=0,dc=function(a,b,c){if(a.indexOf)return a.indexOf(b,c);if(Array.indexOf)return Array.indexOf(a,b,c);for(c=null==c?0:0>c?Math.max(0,a.length+c):c;c<a.length;c++)if(c in a&&a[c]===b)return c;return-1},Y=function(a,b){return-1==dc(a,X)?(r(Error(X+"_"+b),"up","caa"),!1):!0},fc=function(a,b){Y([1,2],"r")&&(S[a]=S[a]||[],S[a].push(b),2==X&&window.setTimeout(function(){b(ec(a))},0))},gc=function(a,b,c){if(Y([1],"nap")&&c){for(var d=0;d<c.length;d++)bc[c[d]]=!0;g.up.spl(a,b,"nap",c)}},hc=
function(a,b,c){if(Y([1],"aop")&&c){if(T)for(var d in T)T[d]=T[d]&&-1!=dc(c,d);else for(T={},d=0;d<c.length;d++)T[c[d]]=!0;g.up.spl(a,b,"aop",c)}},ic=function(){try{if(X=2,!cc){cc=!0;for(var a in S)for(var b=S[a],c=0;c<b.length;c++)try{b[c](ec(a))}catch(d){r(d,"up","tp")}}}catch(d){r(d,"up","mtp")}},ec=function(a){if(Y([2],"ssp")){var b=!bc[a];T&&(b=b&&!!T[a]);return b}};cc=!1;S={};bc={};T=null;X=1;
var jc=function(a){var b=!1;try{b=a.cookie&&a.cookie.match("PREF")}catch(c){}return!b},kc=function(){try{return!!e.localStorage&&"object"==typeof e.localStorage}catch(a){return!1}},lc=function(a){return a&&a.style&&a.style.behavior&&"undefined"!=typeof a.load},mc=function(a,b,c,d){try{jc(document)||(d||(b="og-up-"+b),kc()?e.localStorage.setItem(b,c):lc(a)&&(a.setAttribute(b,c),a.save(a.id)))}catch(f){f.code!=DOMException.QUOTA_EXCEEDED_ERR&&r(f,"up","spd")}},nc=function(a,b,c){try{if(jc(document))return"";
c||(b="og-up-"+b);if(kc())return e.localStorage.getItem(b);if(lc(a))return a.load(a.id),a.getAttribute(b)}catch(d){d.code!=DOMException.QUOTA_EXCEEDED_ERR&&r(d,"up","gpd")}return""},oc=function(a,b,c){a.addEventListener?a.addEventListener(b,c,!1):a.attachEvent&&a.attachEvent("on"+b,c)},pc=function(a){for(var b=0,c;c=a[b];b++){var d=g.up;c=c in d&&d[c];if(!c)return!1}return!0},qc=function(a,b){try{if(jc(a))return-1;var c=a.cookie.match(/OGPC=([^;]*)/);if(c&&c[1]){var d=c[1].match(new RegExp("\\b"+
b+"-([0-9]+):"));if(d&&d[1])return parseInt(d[1],10)}}catch(f){f.code!=DOMException.QUOTA_EXCEEDED_ERR&&r(f,"up","gcc")}return-1};p("up",{r:fc,nap:gc,aop:hc,tp:ic,ssp:ec,spd:mc,gpd:nc,aeh:oc,aal:pc,gcc:qc});var Z=function(a,b){a[b]=function(c){var d=arguments;g.qm(function(){a[b].apply(this,d)})}};Z(g.up,"sl");Z(g.up,"si");Z(g.up,"spl");Z(g.up,"dpc");Z(g.up,"iic");g.mcf("up",{sp:h.b("0.01",1),tld:"com",prid:"1"});function rc(){function a(){for(var l;(l=k[m++])&&"m"!=l[0]&&!l[1].auto;);l&&(sa(2,l[0]),l[1].url&&ra(l[1].url,l[0]),l[1].libs&&C&&C(l[1].libs));m<k.length&&setTimeout(a,0)}function b(){0<f--?setTimeout(b,0):a()}var c=h.a("1"),d=h.a(""),f=3,k=w,m=0,n=window.gbarOnReady;if(n)try{n()}catch(l){r(l,"ml","or")}d?p("ldb",a):c?ca(window,"load",b):b()}p("rdl",rc);}catch(e){window.gbar&&gbar.logger&&gbar.logger.ml(e,{"_sn":"cfg.init"});}})();
(function(){try{/*

 Copyright The Closure Library Authors.
 SPDX-License-Identifier: Apache-2.0
*/
var a=window.gbar;a.mcf("pm",{p:""});}catch(e){window.gbar&&gbar.logger&&gbar.logger.ml(e,{"_sn":"cfg.init"});}})();
(function(){try{/*

 Copyright The Closure Library Authors.
 SPDX-License-Identifier: Apache-2.0
*/
var a=window.gbar;a.mcf("mm",{s:"1"});}catch(e){window.gbar&&gbar.logger&&gbar.logger.ml(e,{"_sn":"cfg.init"});}})();
(function(){try{/*

 Copyright The Closure Library Authors.
 SPDX-License-Identifier: Apache-2.0
*/
var d=window.gbar.i.i;var e=window.gbar;var f=e.i;var g=f.c("1",0),h=/\bgbmt\b/,k=function(a){try{var b=document.getElementById("gb_"+g),c=document.getElementById("gb_"+a);b&&f.l(b,h.test(b.className)?"gbm0l":"gbz0l");c&&f.k(c,h.test(c.className)?"gbm0l":"gbz0l")}catch(l){d(l,"sj","ssp")}g=a},m=e.qs,n=function(a){var b=a.href;var c=window.location.href.match(/.*?:\/\/[^\/]*/)[0];c=new RegExp("^"+c+"/search\\?");(b=c.test(b))&&!/(^|\\?|&)ei=/.test(a.href)&&(b=window.google)&&b.kEXPI&&(a.href+="&ei="+b.kEI)},p=function(a){m(a);
n(a)},q=function(){if(window.google&&window.google.sn){var a=/.*hp$/;return a.test(window.google.sn)?"":"1"}return"-1"};e.rp=q;e.slp=k;e.qs=p;e.qsi=n;}catch(e){window.gbar&&gbar.logger&&gbar.logger.ml(e,{"_sn":"cfg.init"});}})();
(function(){try{/*

 Copyright The Closure Library Authors.
 SPDX-License-Identifier: Apache-2.0
*/
var a=this||self;var b=window.gbar;var c=b.i;var d=c.a,e=c.c,f={cty:"USA",cv:"319703744",dbg:d(""),ecv:"0",ei:e("rC8JX7mSB-aa_Qa1nYGACw"),ele:d("1"),esr:e("0.1"),evts:["mousedown","touchstart","touchmove","wheel","keydown"],gbl:"es_plusone_gc_20200601.0_p0",hd:"com",hl:"en",irp:d(""),pid:e("1"),
snid:e("28834"),to:e("300000"),u:e(""),vf:".66.41."},g=f,h=["bndcfg"],k=a;h[0]in k||"undefined"==typeof k.execScript||k.execScript("var "+h[0]);for(var l;h.length&&(l=h.shift());)h.length||void 0===g?k=k[l]&&k[l]!==Object.prototype[l]?k[l]:k[l]={}:k[l]=g;}catch(e){window.gbar&&gbar.logger&&gbar.logger.ml(e,{"_sn":"cfg.init"});}})();
(function(){try{/*

 Copyright The Closure Library Authors.
 SPDX-License-Identifier: Apache-2.0
*/
window.gbar.rdl();}catch(e){window.gbar&&gbar.logger&&gbar.logger.ml(e,{"_sn":"cfg.init"});}})();
</script></head><body bgcolor="#fff"><script nonce="Z0rwtxMNtMtj5GlIvyrd2g==">(function(){var src='/images/nav_logo229.png';var iesg=false;document.body.onload = function(){window.n && window.n();if (document.images){new Image().src=src;}
if (!iesg){document.f&&document.f.q.focus();document.gbqf&&document.gbqf.q.focus();}
}
})();</script><div id="mngb"><div id=gb><script>window.gbar&&gbar.eli&&gbar.eli()</script><div id=gbw><div id=gbz><span class=gbtcb></span><ol id=gbzc class=gbtc><li class=gbt><a onclick=gbar.logger.il(1,{t:1}); class="gbzt gbz0l gbp1" id=gb_1 href="https://www.google.com/webhp?tab=ww"><span class=gbtb2></span><span class=gbts>Search</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:2}); class=gbzt id=gb_2 href="http://www.google.com/imghp?hl=en&tab=wi"><span class=gbtb2></span><span class=gbts>Images</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:8}); class=gbzt id=gb_8 href="http://maps.google.com/maps?hl=en&tab=wl"><span class=gbtb2></span><span class=gbts>Maps</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:78}); class=gbzt id=gb_78 href="https://play.google.com/?hl=en&tab=w8"><span class=gbtb2></span><span class=gbts>Play</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:36}); class=gbzt id=gb_36 href="http://www.youtube.com/?gl=US&tab=w1"><span class=gbtb2></span><span class=gbts>YouTube</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:5}); class=gbzt id=gb_5 href="http://news.google.com/nwshp?hl=en&tab=wn"><span class=gbtb2></span><span class=gbts>News</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:23}); class=gbzt id=gb_23 href="https://mail.google.com/mail/?tab=wm"><span class=gbtb2></span><span class=gbts>Gmail</span></a></li><li class=gbt><a onclick=gbar.logger.il(1,{t:49}); class=gbzt id=gb_49 href="https://drive.google.com/?tab=wo"><span class=gbtb2></span><span class=gbts>Drive</span></a></li><li class=gbt><a class=gbgt id=gbztm href="https://www.google.com/intl/en/about/products?tab=wh" onclick="gbar.tg(event,this)" aria-haspopup=true aria-owns=gbd><span class=gbtb2></span><span id=gbztms class="gbts gbtsa"><span id=gbztms1>More</span><span class=gbma></span></span></a><div class=gbm id=gbd aria-owner=gbztm><div id=gbmmb class="gbmc gbsb gbsbis"><ol id=gbmm class="gbmcc gbsbic"><li class=gbmtc><a onclick=gbar.logger.il(1,{t:24}); class=gbmt id=gb_24 href="https://www.google.com/calendar?tab=wc">Calendar</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:51}); class=gbmt id=gb_51 href="http://translate.google.com/?hl=en&tab=wT">Translate</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:17}); class=gbmt id=gb_17 href="http://www.google.com/mobile/?hl=en&tab=wD">Mobile</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:10}); class=gbmt id=gb_10 href="https://books.google.com/bkshp?hl=en&tab=wp">Books</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:6}); class=gbmt id=gb_6 href="https://www.google.com/shopping?hl=en&source=og&tab=wf">Shopping</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:30}); class=gbmt id=gb_30 href="http://www.blogger.com/?tab=wj">Blogger</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:27}); class=gbmt id=gb_27 href="http://www.google.com/finance?tab=we">Finance</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:31}); class=gbmt id=gb_31 href="https://photos.google.com/?tab=wq&pageId=none">Photos</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:12}); class=gbmt id=gb_12 href="http://video.google.com/?hl=en&tab=wv">Videos</a></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:25}); class=gbmt id=gb_25 href="https://docs.google.com/document/?usp=docs_alc">Docs</a></li><li class=gbmtc><div class="gbmt gbmh"></div></li><li class=gbmtc><a onclick=gbar.logger.il(1,{t:66}); href="https://www.google.com/intl/en/about/products?tab=wh" class=gbmt>Even more &raquo;</a></li></ol><div class=gbsbt></div><div class=gbsbb></div></div></div></li></ol></div><div id=gbg><h2 class=gbxx>Account Options</h2><span class=gbtcb></span><ol class=gbtc><li class=gbt><a target=_top href="https://accounts.google.com/ServiceLogin?hl=en&passive=true&continue=http://www.google.com/" onclick="gbar.logger.il(9,{l:'i'})" id=gb_70 class=gbgt><span class=gbtb2></span><span id=gbgs4 class=gbts><span id=gbi4s1>Sign in</span></span></a></li><li class="gbt gbtb"><span class=gbts></span></li><li class=gbt><a class=gbgt id=gbg5 href="http://www.google.com/preferences?hl=en" title="Options" onclick="gbar.tg(event,this)" aria-haspopup=true aria-owns=gbd5><span class=gbtb2></span><span id=gbgs5 class=gbts><span id=gbi5></span></span></a><div class=gbm id=gbd5 aria-owner=gbg5><div class=gbmc><ol id=gbom class=gbmcc><li class="gbkc gbmtc"><a  class=gbmt href="/preferences?hl=en">Search settings</a></li><li class=gbmtc><div class="gbmt gbmh"></div></li><li class="gbkp gbmtc"><a class=gbmt href="http://www.google.com/history/optout?hl=en">Web History</a></li></ol></div></div></li></ol></div></div><div id=gbx3></div><div id=gbx4></div><script>window.gbar&&gbar.elp&&gbar.elp()</script></div></div><center><br clear="all" id="lgpd"><div id="lga"><img alt="Google" height="92" src="/images/branding/googlelogo/1x/googlelogo_white_background_color_272x92dp.png" style="padding:28px 0 14px" width="272" id="hplogo"><br><br></div><form action="/search" name="f"><table cellpadding="0" cellspacing="0"><tr valign="top"><td width="25%">&nbsp;</td><td align="center" nowrap=""><input value="en" name="hl" type="hidden"><input name="source" type="hidden" value="hp"><input name="biw" type="hidden"><input name="bih" type="hidden"><div class="ds" style="height:32px;margin:4px 0"><input class="lst" style="margin:0;padding:5px 8px 0 6px;vertical-align:top;color:#000" autocomplete="off" value="" title="Google Search" maxlength="2048" name="q" size="57"></div><br style="line-height:0"><span class="ds"><span class="lsbb"><input class="lsb" value="Google Search" name="btnG" type="submit"></span></span><span class="ds"><span class="lsbb"><input class="lsb" id="tsuid1" value="I'm Feeling Lucky" name="btnI" type="submit"><script nonce="Z0rwtxMNtMtj5GlIvyrd2g==">(function(){var id='tsuid1';document.getElementById(id).onclick = function(){if (this.form.q.value){this.checked = 1;if (this.form.iflsig)this.form.iflsig.disabled = false;}
else top.location='/doodles/';};})();</script><input value="AINFCbYAAAAAXwk9vAYkuR7B56KqhSXVi0SqVmIkqPwd" name="iflsig" type="hidden"></span></span></td><td class="fl sblc" align="left" nowrap="" width="25%"><a href="/advanced_search?hl=en&amp;authuser=0">Advanced search</a></td></tr></table><input id="gbv" name="gbv" type="hidden" value="1"><script nonce="Z0rwtxMNtMtj5GlIvyrd2g==">(function(){var a,b="1";if(document&&document.getElementById)if("undefined"!=typeof XMLHttpRequest)b="2";else if("undefined"!=typeof ActiveXObject){var c,d,e=["MSXML2.XMLHTTP.6.0","MSXML2.XMLHTTP.3.0","MSXML2.XMLHTTP","Microsoft.XMLHTTP"];for(c=0;d=e[c++];)try{new ActiveXObject(d),b="2"}catch(h){}}a=b;if("2"==a&&-1==location.search.indexOf("&gbv=2")){var f=google.gbvu,g=document.getElementById("gbv");g&&(g.value=a);f&&window.setTimeout(function(){location.href=f},0)};}).call(this);</script></form><div id="gac_scont"></div><div style="font-size:83%;min-height:3.5em"><br></div><span id="footer"><div style="font-size:10pt"><div style="margin:19px auto;text-align:center" id="fll"><a href="/intl/en/ads/">Advertising Programs</a><a href="/services/">Business Solutions</a><a href="/intl/en/about.html">About Google</a></div></div><p style="font-size:8pt;color:#767676">&copy; 2020 - <a href="/intl/en/policies/privacy/">Privacy</a> - <a href="/intl/en/policies/terms/">Terms</a></p></span></center><script nonce="Z0rwtxMNtMtj5GlIvyrd2g==">(function(){window.google.cdo={height:0,width:0};(function(){var a=window.innerWidth,b=window.innerHeight;if(!a||!b){var c=window.document,d="CSS1Compat"==c.compatMode?c.documentElement:c.body;a=d.clientWidth;b=d.clientHeight}a&&b&&(a!=google.cdo.width||b!=google.cdo.height)&&google.log("","","/client_204?&atyp=i&biw="+a+"&bih="+b+"&ei="+google.kEI);}).call(this);})();(function(){var u='/xjs/_/js/k\x3dxjs.hp.en_US.7dwu1JjXsWU.O/m\x3dsb_he,d/am\x3dAE-wOQ/d\x3d1/rs\x3dACT90oFUt_m2yoS8JKUxpROnr7ueve8SiA';
setTimeout(function(){var b=document;var a="SCRIPT";"application/xhtml+xml"===b.contentType&&(a=a.toLowerCase());a=b.createElement(a);a.src=u;google.timers&&google.timers.load&&google.tick&&google.tick("load","xjsls");document.body.appendChild(a)},0);})();(function(){window.google.xjsu='/xjs/_/js/k\x3dxjs.hp.en_US.7dwu1JjXsWU.O/m\x3dsb_he,d/am\x3dAE-wOQ/d\x3d1/rs\x3dACT90oFUt_m2yoS8JKUxpROnr7ueve8SiA';})();function _DumpException(e){throw e;}
function _F_installCss(c){}
(function(){google.jl={dw:false,em:[],emw:false,lls:'default',pdt:0,snet:true,uwp:true};})();(function(){var pmc='{\x22d\x22:{},\x22sb_he\x22:{\x22agen\x22:false,\x22cgen\x22:false,\x22client\x22:\x22heirloom-hp\x22,\x22dh\x22:true,\x22dhqt\x22:true,\x22ds\x22:\x22\x22,\x22ffql\x22:\x22en\x22,\x22fl\x22:true,\x22host\x22:\x22google.com\x22,\x22isbh\x22:28,\x22jsonp\x22:true,\x22msgs\x22:{\x22cibl\x22:\x22Clear Search\x22,\x22dym\x22:\x22Did you mean:\x22,\x22lcky\x22:\x22I\\u0026#39;m Feeling Lucky\x22,\x22lml\x22:\x22Learn more\x22,\x22oskt\x22:\x22Input tools\x22,\x22psrc\x22:\x22This search was removed from your \\u003Ca href\x3d\\\x22/history\\\x22\\u003EWeb History\\u003C/a\\u003E\x22,\x22psrl\x22:\x22Remove\x22,\x22sbit\x22:\x22Search by image\x22,\x22srch\x22:\x22Google Search\x22},\x22ovr\x22:{},\x22pq\x22:\x22\x22,\x22refpd\x22:true,\x22rfs\x22:[],\x22sbpl\x22:16,\x22sbpr\x22:16,\x22scd\x22:10,\x22stok\x22:\x224opZcZpkFtwsjGHfQupwrqlnWrA\x22,\x22uhde\x22:false}}';google.pmc=JSON.parse(pmc);})();</script>        </body></html> 94 | recorded_at: Sat, 11 Jul 2020 03:19:08 GMT 95 | recorded_with: VCR 6.0.0 96 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_page_images/handles_unescaped_urls.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: http://test.com/ 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | User-Agent: 11 | - Mozilla/5.0 (Macintosh) 12 | Accept-Encoding: 13 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 14 | Accept: 15 | - "*/*" 16 | Host: 17 | - test.com 18 | response: 19 | status: 20 | code: 301 21 | message: Moved Permanently 22 | headers: 23 | Server: 24 | - nginx/1.16.1 25 | Date: 26 | - Sat, 11 Jul 2020 03:19:00 GMT 27 | Content-Type: 28 | - text/html; charset=UTF-8 29 | Transfer-Encoding: 30 | - chunked 31 | Connection: 32 | - keep-alive 33 | Keep-Alive: 34 | - timeout=20 35 | X-Dis-Request-Id: 36 | - a2838aa187355b40c449c1d4341cc5d8 37 | Location: 38 | - http://www.test.com/ 39 | body: 40 | encoding: UTF-8 41 | string: "301 Moved Permanently

301 42 | Moved Permanently

Object moved to here.


DOSarrest 43 | Internet Security
\n" 44 | recorded_at: Sat, 11 Jul 2020 03:19:00 GMT 45 | - request: 46 | method: get 47 | uri: http://www.test.com/ 48 | body: 49 | encoding: US-ASCII 50 | string: '' 51 | headers: 52 | User-Agent: 53 | - Mozilla/5.0 (Macintosh) 54 | Accept-Encoding: 55 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 56 | Accept: 57 | - "*/*" 58 | Host: 59 | - www.test.com 60 | response: 61 | status: 62 | code: 301 63 | message: Moved Permanently 64 | headers: 65 | Server: 66 | - nginx/1.16.1 67 | Date: 68 | - Sat, 11 Jul 2020 03:19:00 GMT 69 | Content-Type: 70 | - text/html; charset=UTF-8 71 | Transfer-Encoding: 72 | - chunked 73 | Connection: 74 | - keep-alive 75 | Keep-Alive: 76 | - timeout=20 77 | X-Dis-Request-Id: 78 | - 8b09747e816f16714b25596e80293886 79 | Location: 80 | - https://www.test.com/ 81 | body: 82 | encoding: UTF-8 83 | string: "301 Moved Permanently

301 84 | Moved Permanently

Object moved to here.


DOSarrest 85 | Internet Security
\n" 86 | recorded_at: Sat, 11 Jul 2020 03:19:01 GMT 87 | - request: 88 | method: get 89 | uri: https://www.test.com/ 90 | body: 91 | encoding: US-ASCII 92 | string: '' 93 | headers: 94 | User-Agent: 95 | - Mozilla/5.0 (Macintosh) 96 | Accept-Encoding: 97 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 98 | Accept: 99 | - "*/*" 100 | Host: 101 | - www.test.com 102 | response: 103 | status: 104 | code: 200 105 | message: OK 106 | headers: 107 | Server: 108 | - nginx/1.16.1 109 | Date: 110 | - Sat, 11 Jul 2020 03:19:01 GMT 111 | Content-Type: 112 | - text/html 113 | Transfer-Encoding: 114 | - chunked 115 | Connection: 116 | - keep-alive 117 | Keep-Alive: 118 | - timeout=20 119 | X-Dis-Request-Id: 120 | - ab80d220c4e964a267b7edb970c710d0 121 | P3p: 122 | - CP="NON DSP COR ADMa OUR IND UNI COM NAV INT" 123 | Cache-Control: 124 | - no-cache 125 | body: 126 | encoding: ASCII-8BIT 127 | string: "\n\n\n\n\n\n\n\n\n\n\n" 146 | recorded_at: Sat, 11 Jul 2020 03:19:01 GMT 147 | recorded_with: VCR 6.0.0 148 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_stylesheet_images/handles_404s.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: http://google.com/does_not_exist.css 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | Accept-Encoding: 11 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 12 | Accept: 13 | - "*/*" 14 | User-Agent: 15 | - Ruby 16 | response: 17 | status: 18 | code: 404 19 | message: Not Found 20 | headers: 21 | Content-Type: 22 | - text/html; charset=UTF-8 23 | Referrer-Policy: 24 | - no-referrer 25 | Content-Length: 26 | - '1579' 27 | Date: 28 | - Sat, 11 Jul 2020 02:47:46 GMT 29 | body: 30 | encoding: ASCII-8BIT 31 | string: !binary |- 32 | PCFET0NUWVBFIGh0bWw+CjxodG1sIGxhbmc9ZW4+CiAgPG1ldGEgY2hhcnNldD11dGYtOD4KICA8bWV0YSBuYW1lPXZpZXdwb3J0IGNvbnRlbnQ9ImluaXRpYWwtc2NhbGU9MSwgbWluaW11bS1zY2FsZT0xLCB3aWR0aD1kZXZpY2Utd2lkdGgiPgogIDx0aXRsZT5FcnJvciA0MDQgKE5vdCBGb3VuZCkhITE8L3RpdGxlPgogIDxzdHlsZT4KICAgICp7bWFyZ2luOjA7cGFkZGluZzowfWh0bWwsY29kZXtmb250OjE1cHgvMjJweCBhcmlhbCxzYW5zLXNlcmlmfWh0bWx7YmFja2dyb3VuZDojZmZmO2NvbG9yOiMyMjI7cGFkZGluZzoxNXB4fWJvZHl7bWFyZ2luOjclIGF1dG8gMDttYXgtd2lkdGg6MzkwcHg7bWluLWhlaWdodDoxODBweDtwYWRkaW5nOjMwcHggMCAxNXB4fSogPiBib2R5e2JhY2tncm91bmQ6dXJsKC8vd3d3Lmdvb2dsZS5jb20vaW1hZ2VzL2Vycm9ycy9yb2JvdC5wbmcpIDEwMCUgNXB4IG5vLXJlcGVhdDtwYWRkaW5nLXJpZ2h0OjIwNXB4fXB7bWFyZ2luOjExcHggMCAyMnB4O292ZXJmbG93OmhpZGRlbn1pbnN7Y29sb3I6Izc3Nzt0ZXh0LWRlY29yYXRpb246bm9uZX1hIGltZ3tib3JkZXI6MH1AbWVkaWEgc2NyZWVuIGFuZCAobWF4LXdpZHRoOjc3MnB4KXtib2R5e2JhY2tncm91bmQ6bm9uZTttYXJnaW4tdG9wOjA7bWF4LXdpZHRoOm5vbmU7cGFkZGluZy1yaWdodDowfX0jbG9nb3tiYWNrZ3JvdW5kOnVybCgvL3d3dy5nb29nbGUuY29tL2ltYWdlcy9icmFuZGluZy9nb29nbGVsb2dvLzF4L2dvb2dsZWxvZ29fY29sb3JfMTUweDU0ZHAucG5nKSBuby1yZXBlYXQ7bWFyZ2luLWxlZnQ6LTVweH1AbWVkaWEgb25seSBzY3JlZW4gYW5kIChtaW4tcmVzb2x1dGlvbjoxOTJkcGkpeyNsb2dve2JhY2tncm91bmQ6dXJsKC8vd3d3Lmdvb2dsZS5jb20vaW1hZ2VzL2JyYW5kaW5nL2dvb2dsZWxvZ28vMngvZ29vZ2xlbG9nb19jb2xvcl8xNTB4NTRkcC5wbmcpIG5vLXJlcGVhdCAwJSAwJS8xMDAlIDEwMCU7LW1vei1ib3JkZXItaW1hZ2U6dXJsKC8vd3d3Lmdvb2dsZS5jb20vaW1hZ2VzL2JyYW5kaW5nL2dvb2dsZWxvZ28vMngvZ29vZ2xlbG9nb19jb2xvcl8xNTB4NTRkcC5wbmcpIDB9fUBtZWRpYSBvbmx5IHNjcmVlbiBhbmQgKC13ZWJraXQtbWluLWRldmljZS1waXhlbC1yYXRpbzoyKXsjbG9nb3tiYWNrZ3JvdW5kOnVybCgvL3d3dy5nb29nbGUuY29tL2ltYWdlcy9icmFuZGluZy9nb29nbGVsb2dvLzJ4L2dvb2dsZWxvZ29fY29sb3JfMTUweDU0ZHAucG5nKSBuby1yZXBlYXQ7LXdlYmtpdC1iYWNrZ3JvdW5kLXNpemU6MTAwJSAxMDAlfX0jbG9nb3tkaXNwbGF5OmlubGluZS1ibG9jaztoZWlnaHQ6NTRweDt3aWR0aDoxNTBweH0KICA8L3N0eWxlPgogIDxhIGhyZWY9Ly93d3cuZ29vZ2xlLmNvbS8+PHNwYW4gaWQ9bG9nbyBhcmlhLWxhYmVsPUdvb2dsZT48L3NwYW4+PC9hPgogIDxwPjxiPjQwNC48L2I+IDxpbnM+VGhhdOKAmXMgYW4gZXJyb3IuPC9pbnM+CiAgPHA+VGhlIHJlcXVlc3RlZCBVUkwgPGNvZGU+L2RvZXNfbm90X2V4aXN0LmNzczwvY29kZT4gd2FzIG5vdCBmb3VuZCBvbiB0aGlzIHNlcnZlci4gIDxpbnM+VGhhdOKAmXMgYWxsIHdlIGtub3cuPC9pbnM+Cg== 33 | recorded_at: Sat, 11 Jul 2020 02:47:46 GMT 34 | recorded_with: VCR 6.0.0 35 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_stylesheet_images/handles_stylesheet_image_with_a_relative_url.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: https://raw.github.com/charlotte-ruby/image_scraper/master/spec/support/relative_image_url.html 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | User-Agent: 11 | - Mozilla/5.0 (Macintosh) 12 | Accept-Encoding: 13 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 14 | Accept: 15 | - "*/*" 16 | Host: 17 | - raw.github.com 18 | response: 19 | status: 20 | code: 301 21 | message: Moved Permanently 22 | headers: 23 | Connection: 24 | - keep-alive 25 | Content-Length: 26 | - '0' 27 | Location: 28 | - https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/relative_image_url.html 29 | Accept-Ranges: 30 | - bytes 31 | Date: 32 | - Sat, 11 Jul 2020 02:47:46 GMT 33 | Via: 34 | - 1.1 varnish 35 | Age: 36 | - '0' 37 | X-Served-By: 38 | - cache-fty21334-FTY 39 | X-Cache: 40 | - MISS 41 | X-Cache-Hits: 42 | - '0' 43 | Vary: 44 | - Accept-Encoding 45 | X-Fastly-Request-Id: 46 | - 42defefcb229163cf23f39d4a0e7bb09dc2346c7 47 | body: 48 | encoding: UTF-8 49 | string: '' 50 | recorded_at: Sat, 11 Jul 2020 02:47:46 GMT 51 | - request: 52 | method: get 53 | uri: https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/relative_image_url.html 54 | body: 55 | encoding: US-ASCII 56 | string: '' 57 | headers: 58 | User-Agent: 59 | - Mozilla/5.0 (Macintosh) 60 | Accept-Encoding: 61 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 62 | Accept: 63 | - "*/*" 64 | Host: 65 | - raw.githubusercontent.com 66 | response: 67 | status: 68 | code: 200 69 | message: OK 70 | headers: 71 | Connection: 72 | - keep-alive 73 | Content-Length: 74 | - '361' 75 | Cache-Control: 76 | - max-age=300 77 | Content-Security-Policy: 78 | - default-src 'none'; style-src 'unsafe-inline'; sandbox 79 | Content-Type: 80 | - text/plain; charset=utf-8 81 | Etag: 82 | - W/"5d863c870c1499165ae27fc405308b04c53be69e32f5d1a80e1e1265c1165454" 83 | Strict-Transport-Security: 84 | - max-age=31536000 85 | X-Content-Type-Options: 86 | - nosniff 87 | X-Frame-Options: 88 | - deny 89 | X-Xss-Protection: 90 | - 1; mode=block 91 | Via: 92 | - 1.1 varnish 93 | - 1.1 varnish (Varnish/6.0) 94 | X-Github-Request-Id: 95 | - 8CE0:4149:2B2888:32CDBC:5F092851 96 | Accept-Ranges: 97 | - bytes 98 | Date: 99 | - Sat, 11 Jul 2020 02:47:47 GMT 100 | X-Served-By: 101 | - cache-fty21353-FTY 102 | X-Cache: 103 | - MISS, MISS 104 | X-Cache-Hits: 105 | - 0, 0 106 | X-Timer: 107 | - S1594435667.903284,VS0,VE105 108 | Vary: 109 | - Authorization,Accept-Encoding 110 | Access-Control-Allow-Origin: 111 | - "*" 112 | X-Fastly-Request-Id: 113 | - 53c0ddf688d229bb9907820b623a2f696ff20874 114 | Expires: 115 | - Sat, 11 Jul 2020 02:52:47 GMT 116 | Source-Age: 117 | - '0' 118 | body: 119 | encoding: ASCII-8BIT 120 | string: | 121 | 122 | 123 | 124 | 125 | 126 | stylesheet_test 127 | 128 | 129 | 137 | 138 | 139 | 140 | 141 | recorded_at: Sat, 11 Jul 2020 02:47:47 GMT 142 | - request: 143 | method: get 144 | uri: https://raw.github.com/charlotte-ruby/image_scraper/master/spec/support/relative_image_url.css 145 | body: 146 | encoding: US-ASCII 147 | string: '' 148 | headers: 149 | Accept-Encoding: 150 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 151 | Accept: 152 | - "*/*" 153 | User-Agent: 154 | - Ruby 155 | response: 156 | status: 157 | code: 301 158 | message: Moved Permanently 159 | headers: 160 | Connection: 161 | - keep-alive 162 | Content-Length: 163 | - '0' 164 | Location: 165 | - https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/relative_image_url.css 166 | Accept-Ranges: 167 | - bytes 168 | Date: 169 | - Sat, 11 Jul 2020 02:47:47 GMT 170 | Via: 171 | - 1.1 varnish 172 | Age: 173 | - '0' 174 | X-Served-By: 175 | - cache-fty21382-FTY 176 | X-Cache: 177 | - MISS 178 | X-Cache-Hits: 179 | - '0' 180 | Vary: 181 | - Accept-Encoding 182 | X-Fastly-Request-Id: 183 | - c4d0c44089a9cd9f259ba5dfba01b82886c5bdd3 184 | body: 185 | encoding: UTF-8 186 | string: '' 187 | recorded_at: Sat, 11 Jul 2020 02:47:47 GMT 188 | - request: 189 | method: get 190 | uri: https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/relative_image_url.css 191 | body: 192 | encoding: US-ASCII 193 | string: '' 194 | headers: 195 | Accept-Encoding: 196 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 197 | Accept: 198 | - "*/*" 199 | User-Agent: 200 | - Ruby 201 | response: 202 | status: 203 | code: 200 204 | message: OK 205 | headers: 206 | Connection: 207 | - keep-alive 208 | Content-Length: 209 | - '75' 210 | Cache-Control: 211 | - max-age=300 212 | Content-Security-Policy: 213 | - default-src 'none'; style-src 'unsafe-inline'; sandbox 214 | Content-Type: 215 | - text/plain; charset=utf-8 216 | Etag: 217 | - W/"2c5ae7c2bb86ae5983019ef6e4fdd989bb35f94f20f3fd73e86ed5d9a953c9d1" 218 | Strict-Transport-Security: 219 | - max-age=31536000 220 | X-Content-Type-Options: 221 | - nosniff 222 | X-Frame-Options: 223 | - deny 224 | X-Xss-Protection: 225 | - 1; mode=block 226 | Via: 227 | - 1.1 varnish 228 | - 1.1 varnish (Varnish/6.0) 229 | X-Github-Request-Id: 230 | - A436:5FF4:137A64:1747FA:5F092852 231 | Accept-Ranges: 232 | - bytes 233 | Date: 234 | - Sat, 11 Jul 2020 02:47:47 GMT 235 | X-Served-By: 236 | - cache-fty21326-FTY 237 | X-Cache: 238 | - MISS, MISS 239 | X-Cache-Hits: 240 | - 0, 0 241 | X-Timer: 242 | - S1594435667.266851,VS0,VE115 243 | Vary: 244 | - Authorization,Accept-Encoding 245 | Access-Control-Allow-Origin: 246 | - "*" 247 | X-Fastly-Request-Id: 248 | - d8d1068470ba9ef99c8926302b7cfd66c55e3673 249 | Expires: 250 | - Sat, 11 Jul 2020 02:52:47 GMT 251 | Source-Age: 252 | - '0' 253 | body: 254 | encoding: ASCII-8BIT 255 | string: | 256 | .test { 257 | background-image: url('../images/some_image.png') 258 | } 259 | recorded_at: Sat, 11 Jul 2020 02:47:47 GMT 260 | recorded_with: VCR 6.0.0 261 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_stylesheet_images/scrapes_stylesheet_images.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: https://raw.github.com/charlotte-ruby/image_scraper/master/spec/support/stylesheet_unescaped_image.html 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | User-Agent: 11 | - Mozilla/5.0 (Macintosh) 12 | Accept-Encoding: 13 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 14 | Accept: 15 | - "*/*" 16 | Host: 17 | - raw.github.com 18 | response: 19 | status: 20 | code: 301 21 | message: Moved Permanently 22 | headers: 23 | Connection: 24 | - keep-alive 25 | Content-Length: 26 | - '0' 27 | Location: 28 | - https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/stylesheet_unescaped_image.html 29 | Accept-Ranges: 30 | - bytes 31 | Date: 32 | - Sat, 11 Jul 2020 02:47:46 GMT 33 | Via: 34 | - 1.1 varnish 35 | Age: 36 | - '0' 37 | X-Served-By: 38 | - cache-fty21330-FTY 39 | X-Cache: 40 | - MISS 41 | X-Cache-Hits: 42 | - '0' 43 | Vary: 44 | - Accept-Encoding 45 | X-Fastly-Request-Id: 46 | - 67cf4e16a67c4ff9dc5bc784d87f623f506e97b6 47 | body: 48 | encoding: UTF-8 49 | string: '' 50 | recorded_at: Sat, 11 Jul 2020 02:47:46 GMT 51 | - request: 52 | method: get 53 | uri: https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/stylesheet_unescaped_image.html 54 | body: 55 | encoding: US-ASCII 56 | string: '' 57 | headers: 58 | User-Agent: 59 | - Mozilla/5.0 (Macintosh) 60 | Accept-Encoding: 61 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 62 | Accept: 63 | - "*/*" 64 | Host: 65 | - raw.githubusercontent.com 66 | response: 67 | status: 68 | code: 200 69 | message: OK 70 | headers: 71 | Connection: 72 | - keep-alive 73 | Content-Length: 74 | - '393' 75 | Cache-Control: 76 | - max-age=300 77 | Content-Security-Policy: 78 | - default-src 'none'; style-src 'unsafe-inline'; sandbox 79 | Content-Type: 80 | - text/plain; charset=utf-8 81 | Etag: 82 | - W/"92596301aa24cd3a793a9a17d1d618bcc5b1e7072ce62d94abbf449f3f44e523" 83 | Strict-Transport-Security: 84 | - max-age=31536000 85 | X-Content-Type-Options: 86 | - nosniff 87 | X-Frame-Options: 88 | - deny 89 | X-Xss-Protection: 90 | - 1; mode=block 91 | Via: 92 | - 1.1 varnish 93 | - 1.1 varnish (Varnish/6.0) 94 | X-Github-Request-Id: 95 | - B256:5FF4:137A5F:1747CE:5F09284B 96 | Accept-Ranges: 97 | - bytes 98 | Date: 99 | - Sat, 11 Jul 2020 02:47:46 GMT 100 | X-Served-By: 101 | - cache-fty21365-FTY 102 | X-Cache: 103 | - MISS, MISS 104 | X-Cache-Hits: 105 | - 0, 0 106 | X-Timer: 107 | - S1594435666.161183,VS0,VE114 108 | Vary: 109 | - Authorization,Accept-Encoding 110 | Access-Control-Allow-Origin: 111 | - "*" 112 | X-Fastly-Request-Id: 113 | - b43101a4b597ca7c347a93cff934973b3d17990c 114 | Expires: 115 | - Sat, 11 Jul 2020 02:52:46 GMT 116 | Source-Age: 117 | - '0' 118 | body: 119 | encoding: ASCII-8BIT 120 | string: | 121 | z 122 | 123 | 124 | 125 | 126 | stylesheet_test 127 | 128 | 129 | 137 | 138 | 139 | 140 | 141 | recorded_at: Sat, 11 Jul 2020 02:47:46 GMT 142 | - request: 143 | method: get 144 | uri: https://raw.github.com/charlotte-ruby/image_scraper/master/spec/support/unescaped_image.css 145 | body: 146 | encoding: US-ASCII 147 | string: '' 148 | headers: 149 | Accept-Encoding: 150 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 151 | Accept: 152 | - "*/*" 153 | User-Agent: 154 | - Ruby 155 | response: 156 | status: 157 | code: 301 158 | message: Moved Permanently 159 | headers: 160 | Connection: 161 | - keep-alive 162 | Content-Length: 163 | - '0' 164 | Location: 165 | - https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/unescaped_image.css 166 | Accept-Ranges: 167 | - bytes 168 | Date: 169 | - Sat, 11 Jul 2020 02:47:46 GMT 170 | Via: 171 | - 1.1 varnish 172 | Age: 173 | - '0' 174 | X-Served-By: 175 | - cache-fty21378-FTY 176 | X-Cache: 177 | - MISS 178 | X-Cache-Hits: 179 | - '0' 180 | Vary: 181 | - Accept-Encoding 182 | X-Fastly-Request-Id: 183 | - c798264dfc2e5d7861e0abbe1696d6b1694c6882 184 | body: 185 | encoding: UTF-8 186 | string: '' 187 | recorded_at: Sat, 11 Jul 2020 02:47:46 GMT 188 | - request: 189 | method: get 190 | uri: https://raw.githubusercontent.com/charlotte-ruby/image_scraper/master/spec/support/unescaped_image.css 191 | body: 192 | encoding: US-ASCII 193 | string: '' 194 | headers: 195 | Accept-Encoding: 196 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 197 | Accept: 198 | - "*/*" 199 | User-Agent: 200 | - Ruby 201 | response: 202 | status: 203 | code: 200 204 | message: OK 205 | headers: 206 | Connection: 207 | - keep-alive 208 | Content-Length: 209 | - '117' 210 | Cache-Control: 211 | - max-age=300 212 | Content-Security-Policy: 213 | - default-src 'none'; style-src 'unsafe-inline'; sandbox 214 | Content-Type: 215 | - text/plain; charset=utf-8 216 | Etag: 217 | - W/"a372d59f8b5e966084cad7f1c3389d218c1056d1a6b0cc1f78ce7766543521cd" 218 | Strict-Transport-Security: 219 | - max-age=31536000 220 | X-Content-Type-Options: 221 | - nosniff 222 | X-Frame-Options: 223 | - deny 224 | X-Xss-Protection: 225 | - 1; mode=block 226 | Via: 227 | - 1.1 varnish 228 | - 1.1 varnish (Varnish/6.0) 229 | X-Github-Request-Id: 230 | - 95FC:6E25:146840:18350B:5F092848 231 | Accept-Ranges: 232 | - bytes 233 | Date: 234 | - Sat, 11 Jul 2020 02:47:46 GMT 235 | X-Served-By: 236 | - cache-fty21349-FTY 237 | X-Cache: 238 | - MISS, MISS 239 | X-Cache-Hits: 240 | - 0, 0 241 | X-Timer: 242 | - S1594435667.501147,VS0,VE115 243 | Vary: 244 | - Authorization,Accept-Encoding 245 | Access-Control-Allow-Origin: 246 | - "*" 247 | X-Fastly-Request-Id: 248 | - 05d7f78a7df9449f89a871c44ebb8caf1b56dfbe 249 | Expires: 250 | - Sat, 11 Jul 2020 02:52:46 GMT 251 | Source-Age: 252 | - '0' 253 | body: 254 | encoding: ASCII-8BIT 255 | string: | 256 | .test { 257 | background-image: url('https://raw.github.com/charlotte-ruby/image_scraper/master/some image.png') 258 | } 259 | recorded_at: Sat, 11 Jul 2020 02:47:46 GMT 260 | recorded_with: VCR 6.0.0 261 | -------------------------------------------------------------------------------- /spec/cassettes/ImageScraper_Client/_stylesheets/lists_relative_path_stylesheets.yml: -------------------------------------------------------------------------------- 1 | --- 2 | http_interactions: 3 | - request: 4 | method: get 5 | uri: http://test.com/ 6 | body: 7 | encoding: US-ASCII 8 | string: '' 9 | headers: 10 | User-Agent: 11 | - Mozilla/5.0 (Macintosh) 12 | Accept-Encoding: 13 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 14 | Accept: 15 | - "*/*" 16 | Host: 17 | - test.com 18 | response: 19 | status: 20 | code: 301 21 | message: Moved Permanently 22 | headers: 23 | Server: 24 | - nginx/1.16.1 25 | Date: 26 | - Sat, 11 Jul 2020 03:17:21 GMT 27 | Content-Type: 28 | - text/html; charset=UTF-8 29 | Transfer-Encoding: 30 | - chunked 31 | Connection: 32 | - keep-alive 33 | Keep-Alive: 34 | - timeout=20 35 | X-Dis-Request-Id: 36 | - 903727ea6a57471f663fbe8dabe1c00f 37 | Location: 38 | - http://www.test.com/ 39 | body: 40 | encoding: UTF-8 41 | string: "301 Moved Permanently

301 42 | Moved Permanently

Object moved to here.


DOSarrest 43 | Internet Security
\n" 44 | recorded_at: Sat, 11 Jul 2020 03:17:21 GMT 45 | - request: 46 | method: get 47 | uri: http://www.test.com/ 48 | body: 49 | encoding: US-ASCII 50 | string: '' 51 | headers: 52 | User-Agent: 53 | - Mozilla/5.0 (Macintosh) 54 | Accept-Encoding: 55 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 56 | Accept: 57 | - "*/*" 58 | Host: 59 | - www.test.com 60 | response: 61 | status: 62 | code: 301 63 | message: Moved Permanently 64 | headers: 65 | Server: 66 | - nginx/1.16.1 67 | Date: 68 | - Sat, 11 Jul 2020 03:17:21 GMT 69 | Content-Type: 70 | - text/html; charset=UTF-8 71 | Transfer-Encoding: 72 | - chunked 73 | Connection: 74 | - keep-alive 75 | Keep-Alive: 76 | - timeout=20 77 | X-Dis-Request-Id: 78 | - b654c84e0b7623155267ef12e5fe8d60 79 | Location: 80 | - https://www.test.com/ 81 | body: 82 | encoding: UTF-8 83 | string: "301 Moved Permanently

301 84 | Moved Permanently

Object moved to here.


DOSarrest 85 | Internet Security
\n" 86 | recorded_at: Sat, 11 Jul 2020 03:17:21 GMT 87 | - request: 88 | method: get 89 | uri: https://www.test.com/ 90 | body: 91 | encoding: US-ASCII 92 | string: '' 93 | headers: 94 | User-Agent: 95 | - Mozilla/5.0 (Macintosh) 96 | Accept-Encoding: 97 | - gzip;q=1.0,deflate;q=0.6,identity;q=0.3 98 | Accept: 99 | - "*/*" 100 | Host: 101 | - www.test.com 102 | response: 103 | status: 104 | code: 200 105 | message: OK 106 | headers: 107 | Server: 108 | - nginx/1.16.1 109 | Date: 110 | - Sat, 11 Jul 2020 03:17:22 GMT 111 | Content-Type: 112 | - text/html 113 | Transfer-Encoding: 114 | - chunked 115 | Connection: 116 | - keep-alive 117 | Keep-Alive: 118 | - timeout=20 119 | X-Dis-Request-Id: 120 | - df0d48077779cd58216d5edc81a47f69 121 | P3p: 122 | - CP="NON DSP COR ADMa OUR IND UNI COM NAV INT" 123 | Cache-Control: 124 | - no-cache 125 | body: 126 | encoding: ASCII-8BIT 127 | string: "\n\n\n\n\n\n\n\n\n\n\n" 146 | recorded_at: Sat, 11 Jul 2020 03:17:22 GMT 147 | recorded_with: VCR 6.0.0 148 | -------------------------------------------------------------------------------- /spec/image_scraper/client_spec.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require 'spec_helper' 4 | 5 | describe ImageScraper::Client, :vcr do 6 | let(:repo_url) { "https://raw.github.com/charlotte-ruby/image_scraper" } 7 | 8 | describe "foo" do 9 | it "something" do 10 | url = "http://www.amazon.com/Planet-Two-Disc-Digital-Combo-Blu-ray/dp/B004LWZW4W/ref=sr_1_1?s=movies-tv&ie=UTF8&qid=1324771542&sr=1-1" 11 | 12 | client = described_class.new(url) 13 | 14 | expect(client.page_images).not_to be_empty 15 | end 16 | end 17 | 18 | describe "#initialize" do 19 | it 'works with invalid URLs' do 20 | allow_any_instance_of(described_class).to receive(:fetch).and_return(nil) 21 | 22 | scraper = described_class.new('bogusurl4444.com') 23 | 24 | expect(scraper.doc).to be_nil 25 | end 26 | 27 | it 'has empty data if URL is invalid' do 28 | allow_any_instance_of(described_class).to receive(:fetch).and_return(nil) 29 | 30 | scraper = described_class.new('bogusurl4444.com') 31 | 32 | expect(scraper.image_urls).to be_empty 33 | expect(scraper.stylesheets).to be_empty 34 | expect(scraper.stylesheet_images).to be_empty 35 | expect(scraper.page_images).to be_empty 36 | end 37 | end 38 | 39 | describe '#image_urls' do 40 | it 'scrapes absolute paths' do 41 | images = [ 42 | 'http://en.wikipedia.org/static/images/poweredby_mediawiki_88x31.png', 43 | 'http://en.wikipedia.org/static/images/wikimedia-button.png', 44 | 'http://en.wikipedia.org/wiki/Special:CentralAutoLogin/start?type=1x1', 45 | 'http://upload.wikimedia.org/wikipedia/commons/thumb/b/b6/SIPI_Jelly_Beans_4.1.07.tiff/lossy-page1-220px-SIPI_Jelly_Beans_4.1.07.tiff.jpg', 46 | 'http://upload.wikimedia.org/wikipedia/en/thumb/5/5c/Symbol_template_class.svg/16px-Symbol_template_class.svg.png' 47 | ] 48 | 49 | url = 'http://en.wikipedia.org/wiki/Standard_test_image' 50 | 51 | client = described_class.new(url, include_css_images: false) 52 | 53 | expect(client.image_urls).to eq(images) 54 | end 55 | 56 | it 'scrapes with whitespace stripped' do 57 | file = 'spec/support/extra_whitespace.html' 58 | 59 | client = described_class.new('') 60 | client.doc = File.open(file) { |f| Nokogiri::HTML(f) } 61 | 62 | images = [ 63 | 'http://g-ecx.images-amazon.com/images/G/01/SIMON/IsaacsonWalter._V164348457_.jpg', 64 | 'http://g-ecx.images-amazon.com/images/G/01/SIMON/IsaacsonWalter.jpg' 65 | ] 66 | 67 | expect(client.image_urls).to eq(images) 68 | end 69 | 70 | it 'scrapes relative paths' do 71 | scraper = described_class.new('http://en.wikipedia.org/wiki/Standard_test_image', 72 | convert_to_absolute_url: false, 73 | include_css_images: false) 74 | 75 | images = [ 76 | 'http://en.wikipedia.org/static/images/poweredby_mediawiki_88x31.png', 77 | 'http://en.wikipedia.org/static/images/wikimedia-button.png', 78 | 'http://en.wikipedia.org/wiki/Special:CentralAutoLogin/start?type=1x1', 79 | 'http://upload.wikimedia.org/wikipedia/commons/thumb/b/b6/SIPI_Jelly_Beans_4.1.07.tiff/lossy-page1-220px-SIPI_Jelly_Beans_4.1.07.tiff.jpg', 80 | 'http://upload.wikimedia.org/wikipedia/en/thumb/5/5c/Symbol_template_class.svg/16px-Symbol_template_class.svg.png' 81 | ] 82 | 83 | expect(scraper.image_urls).to eq(images) 84 | end 85 | 86 | it 'handles url with unescaped spaces' do 87 | url = 'https://raw.github.com/syoder/image_scraper/stylesheet_fix/test/resources/space in url.html' 88 | 89 | scraper = described_class.new(url, include_css_images: false) 90 | 91 | expected_url = 'https://raw.github.com/syoder/image_scraper/stylesheet_fix/test/resources/image1.png' 92 | 93 | expect(scraper.image_urls.length).to eq(1) 94 | expect(scraper.image_urls.first).to eq(expected_url) 95 | end 96 | end 97 | 98 | describe '#stylesheets' do 99 | it 'lists relative path stylesheets' do 100 | file = 'spec/support/stylesheet_test.html' 101 | 102 | client = described_class.new('http://test.com') 103 | client.doc = File.open(file) { |f| Nokogiri::HTML(f) } 104 | 105 | stylesheets = [ 106 | 'http://test.com/css/master.css', 107 | 'http://test.com/css/master2.css' 108 | ] 109 | 110 | expect(client.stylesheets).to eq(stylesheets) 111 | end 112 | 113 | it 'handles stylesheet with an unescaped url' do 114 | scraper = described_class.new('') 115 | scraper.url = 'http://test.com' 116 | scraper.doc = Nokogiri::HTML("") 117 | 118 | expect(scraper.stylesheets).to include('http://test.com/unescapedpath.css') 119 | end 120 | end 121 | 122 | describe '#page_images' do 123 | it 'handles unescaped urls' do 124 | scraper = described_class.new('http://test.com') 125 | scraper.doc = Nokogiri::HTML("") 126 | 127 | expect(scraper.page_images.length).to eq(1) 128 | expect(scraper.page_images).to include('http://test.com/unescaped%20path') 129 | end 130 | 131 | it 'handldes image urls that include square brackets' do 132 | scraper = described_class.new('http://google.com') 133 | scraper.doc = Nokogiri::HTML("") 134 | 135 | expect(scraper.page_images).to be_empty 136 | end 137 | end 138 | 139 | describe '#stylesheet_images' do 140 | it 'scrapes stylesheet images' do 141 | url = "#{repo_url}/master/spec/support/stylesheet_unescaped_image.html" 142 | stylesheet_path = "#{repo_url}/master/someimage.png" 143 | # /charlotte-ruby/image_scraper/master/spec/support/unescaped_image.css 144 | scraper = described_class.new(url, include_css_images: true) 145 | 146 | expect(scraper.stylesheet_images).to include(stylesheet_path) 147 | end 148 | 149 | it 'handles 404s' do 150 | scraper = described_class.new('') 151 | scraper.url = 'http://google.com' 152 | scraper.doc = Nokogiri::HTML("") 153 | 154 | expect(scraper.stylesheet_images).to be_empty 155 | end 156 | 157 | it 'handles stylesheet image with a relative url' do 158 | url = "#{repo_url}/master/spec/support/relative_image_url.html" 159 | image_url = "#{repo_url}/master/spec/images/some_image.png" 160 | 161 | scraper = described_class.new(url, include_css_images: true) 162 | 163 | expect(scraper.stylesheet_images).to include(image_url) 164 | end 165 | end 166 | end 167 | -------------------------------------------------------------------------------- /spec/image_scraper/util_spec.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require 'spec_helper' 4 | 5 | describe ImageScraper::Util do 6 | 7 | describe 'absolute_url' do 8 | it 'parses asset' do 9 | url = 'http://www.test.com' 10 | asset = 'image.gif' 11 | 12 | result = described_class.absolute_url(url, asset) 13 | 14 | expect(result).to eq('http://www.test.com/image.gif') 15 | end 16 | 17 | it 'parses relative asset' do 18 | url = 'http://www.test.com' 19 | asset = 'images/image.gif' 20 | 21 | result = described_class.absolute_url(url, asset) 22 | 23 | expect(result).to eq('http://www.test.com/images/image.gif') 24 | end 25 | 26 | it 'parses absolute asset' do 27 | url = 'http://www.test.com' 28 | asset = '/images/image.gif' 29 | 30 | result = described_class.absolute_url(url, asset) 31 | 32 | expect(result).to eq('http://www.test.com/images/image.gif') 33 | end 34 | 35 | it 'parses root url with no asset' do 36 | result = described_class.absolute_url('http://www.test.com') 37 | 38 | expect(result).to eq('http://www.test.com') 39 | end 40 | 41 | it 'parses url with no asset' do 42 | result = described_class.absolute_url('http://www.test.com/a/test.html') 43 | 44 | expect(result).to eq('http://www.test.com/a/test.html') 45 | end 46 | end 47 | 48 | describe 'strip_quotes' do 49 | it 'parses paths' do 50 | result = described_class.strip_quotes("'/images/test.png'") 51 | 52 | expect(result).to eq('/images/test.png') 53 | end 54 | 55 | it 'parses a full url' do 56 | str = "'http://www.somsite.com/images/test.png'" 57 | 58 | result = described_class.strip_quotes(str) 59 | 60 | expect(result).to eq('http://www.somsite.com/images/test.png') 61 | end 62 | 63 | it 'parses emptyness' do 64 | result = described_class.strip_quotes('') 65 | 66 | expect(result).to be_empty 67 | end 68 | end 69 | 70 | describe 'domain' do 71 | it 'parses the domain of a url' do 72 | u = described_class 73 | 74 | expect(u.domain('http://ug.ly')).to eq('http://ug.ly') 75 | expect(u.domain('http://ug.ly/what')).to eq('http://ug.ly') 76 | expect(u.domain('http://ug.ly/what/is/this/')).to eq('http://ug.ly') 77 | expect(u.domain('http://www.ug.ly/what/is/this/')).to eq('http://www.ug.ly') 78 | expect(u.domain('http://ug.ly/what/is/this.html')).to eq('http://ug.ly') 79 | end 80 | end 81 | end 82 | -------------------------------------------------------------------------------- /spec/image_scraper_spec.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | RSpec.describe ImageScraper do 4 | it 'has a version number' do 5 | expect(ImageScraper::VERSION).not_to be_nil 6 | end 7 | end 8 | -------------------------------------------------------------------------------- /spec/spec_helper.rb: -------------------------------------------------------------------------------- 1 | # frozen_string_literal: true 2 | 3 | require 'webmock/rspec' 4 | require 'vcr' 5 | require 'image_scraper' 6 | 7 | # FIXME: remove when fixed in vcr 8 | # https://github.com/vcr/vcr/pull/907/files 9 | module VCR 10 | class LibraryHooks 11 | # @private 12 | module WebMock 13 | module_function 14 | 15 | def with_global_hook_disabled(request) 16 | global_hook_disabled_requests << request 17 | 18 | begin 19 | yield 20 | ensure 21 | global_hook_disabled_requests.delete(request) 22 | end 23 | end 24 | 25 | def global_hook_disabled?(request) 26 | requests = Thread.current[:_vcr_webmock_disabled_requests] 27 | requests&.include?(request) 28 | end 29 | 30 | def global_hook_disabled_requests 31 | Thread.current[:_vcr_webmock_disabled_requests] ||= [] 32 | end 33 | end 34 | end 35 | end 36 | 37 | VCR.configure do |c| 38 | c.cassette_library_dir = 'spec/cassettes' 39 | c.hook_into :webmock 40 | c.configure_rspec_metadata! 41 | end 42 | 43 | # This file was generated by the `rspec --init` command. Conventionally, all 44 | # specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`. 45 | # The generated `.rspec` file contains `--require spec_helper` which will cause 46 | # this file to always be loaded, without a need to explicitly require it in any 47 | # files. 48 | # 49 | # Given that it is always loaded, you are encouraged to keep this file as 50 | # light-weight as possible. Requiring heavyweight dependencies from this file 51 | # will add to the boot time of your test suite on EVERY test run, even for an 52 | # individual file that may not need all of that loaded. Instead, consider making 53 | # a separate helper file that requires the additional dependencies and performs 54 | # the additional setup, and require it from the spec files that actually need 55 | # it. 56 | # 57 | # See http://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration 58 | RSpec.configure do |config| 59 | # rspec-expectations config goes here. You can use an alternate 60 | # assertion/expectation library such as wrong or the stdlib/minitest 61 | # assertions if you prefer. 62 | config.expect_with :rspec do |expectations| 63 | # This option will default to `true` in RSpec 4. It makes the `description` 64 | # and `failure_message` of custom matchers include text for helper methods 65 | # defined using `chain`, e.g.: 66 | # be_bigger_than(2).and_smaller_than(4).description 67 | # # => "be bigger than 2 and smaller than 4" 68 | # ...rather than: 69 | # # => "be bigger than 2" 70 | expectations.include_chain_clauses_in_custom_matcher_descriptions = true 71 | end 72 | 73 | # rspec-mocks config goes here. You can use an alternate test double 74 | # library (such as bogus or mocha) by changing the `mock_with` option here. 75 | config.mock_with :rspec do |mocks| 76 | # Prevents you from mocking or stubbing a method that does not exist on 77 | # a real object. This is generally recommended, and will default to 78 | # `true` in RSpec 4. 79 | mocks.verify_partial_doubles = true 80 | end 81 | 82 | # This option will default to `:apply_to_host_groups` in RSpec 4 (and will 83 | # have no way to turn it off -- the option exists only for backwards 84 | # compatibility in RSpec 3). It causes shared context metadata to be 85 | # inherited by the metadata hash of host groups and examples, rather than 86 | # triggering implicit auto-inclusion in groups with matching metadata. 87 | config.shared_context_metadata_behavior = :apply_to_host_groups 88 | 89 | # The settings below are suggested to provide a good initial experience 90 | # with RSpec, but feel free to customize to your heart's content. 91 | # # This allows you to limit a spec run to individual examples or groups 92 | # # you care about by tagging them with `:focus` metadata. When nothing 93 | # # is tagged with `:focus`, all examples get run. RSpec also provides 94 | # # aliases for `it`, `describe`, and `context` that include `:focus` 95 | # # metadata: `fit`, `fdescribe` and `fcontext`, respectively. 96 | config.filter_run_when_matching :focus 97 | # 98 | # # Allows RSpec to persist some state between runs in order to support 99 | # # the `--only-failures` and `--next-failure` CLI options. We recommend 100 | # # you configure your source control system to ignore this file. 101 | # config.example_status_persistence_file_path = "spec/examples.txt" 102 | # 103 | # # Limits the available syntax to the non-monkey patched syntax that is 104 | # # recommended. For more details, see: 105 | # # - http://rspec.info/blog/2012/06/rspecs-new-expectation-syntax/ 106 | # # - http://www.teaisaweso.me/blog/2013/05/27/rspecs-new-message-expectation-syntax/ 107 | # # - http://rspec.info/blog/2014/05/notable-changes-in-rspec-3/#zero-monkey-patching-mode 108 | # config.disable_monkey_patching! 109 | # 110 | # # This setting enables warnings. It's recommended, but in some cases may 111 | # # be too noisy due to issues in dependencies. 112 | # config.warnings = true 113 | # 114 | # # Many RSpec users commonly either run the entire suite or an individual 115 | # # file, and it's useful to allow more verbose output when running an 116 | # # individual spec file. 117 | # if config.files_to_run.one? 118 | # # Use the documentation formatter for detailed output, 119 | # # unless a formatter has already been configured 120 | # # (e.g. via a command-line flag). 121 | # config.default_formatter = "doc" 122 | # end 123 | # 124 | # # Print the 10 slowest examples and example groups at the 125 | # # end of the spec run, to help surface which specs are running 126 | # # particularly slow. 127 | # config.profile_examples = 10 128 | # 129 | # # Run specs in random order to surface order dependencies. If you find an 130 | # # order dependency and want to debug it, you can fix the order by providing 131 | # # the seed, which is printed after each run. 132 | # # --seed 1234 133 | # config.order = :random 134 | # 135 | # # Seed global randomization in this process using the `--seed` CLI option. 136 | # # Setting this allows you to use `--seed` to deterministically reproduce 137 | # # test failures related to randomization by passing the same `--seed` value 138 | # # as the one that triggered the failure. 139 | # Kernel.srand config.seed 140 | end 141 | -------------------------------------------------------------------------------- /spec/support/extra_whitespace.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | -------------------------------------------------------------------------------- /spec/support/relative_image_url.css: -------------------------------------------------------------------------------- 1 | .test { 2 | background-image: url('../images/some_image.png') 3 | } 4 | -------------------------------------------------------------------------------- /spec/support/relative_image_url.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | stylesheet_test 7 | 8 | 9 | 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /spec/support/space in url.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /spec/support/stylesheet_test.html: -------------------------------------------------------------------------------- 1 | 3 | 4 | 5 | 6 | 7 | stylesheet_test 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | -------------------------------------------------------------------------------- /spec/support/stylesheet_unescaped_image.html: -------------------------------------------------------------------------------- 1 | z 2 | 3 | 4 | 5 | 6 | stylesheet_test 7 | 8 | 9 | 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /spec/support/unescaped_image.css: -------------------------------------------------------------------------------- 1 | .test { 2 | background-image: url('https://raw.github.com/charlotte-ruby/image_scraper/master/some image.png') 3 | } 4 | --------------------------------------------------------------------------------