Case Challenge for Proper, part of the interview process
Participant information
- Name: Manuel Panichelli
- Email: [email protected]
- LinkedIn: /in/manuel-panichelli/
Each case assignment has its own tag (case-1, case-2, etc.) and brief section in the README
The program will download the number of memes configured at amount from
http://icanhas.cheezburger.com/, advancing pages when necessary. It will save
them to a images/ local directory (relative path to where you're executing the
program).
Arguments:
--amount: Amount of memes to download (Default: 10)--threads(1-5): Number of threads (actually goroutines) to use to download images (Default: 1)
Example:
go run main.go --amount 20 --threads 3Assignment: Write a program that downloads the images from http://icanhas.cheezburger.com/ and stores them locally. Only download graphics that are memes on the page and not sponsored content. Download the first ten memes on the homepage and store them locally. Name them
1.jpg, ...,10.jpg.
I started out by inspecting the page, and noticed that memes are all img
elements with the image urls on the src tags.
When I tried scraping the images, I'd get weird URLs in the src tag like
. I realized
that's because it lazy loads images, and the src tag starts out with a dummy
value which is replaced with data-src later. So I considered both.
I found the scraping framework colly which seems easy
enough to use, so I'll use that from now on.
Program usage:
go run main.goIt will download the first 10 images from the site to a local images/
directory (relative path), that should exist.
Assignment: Extend the program to accept a parameter named
amountthat determines how many memes need to be downloaded. Hint: if it's more than on the first page, I expect the program to go to the next page.
Inspecting the page, I realized that the page links where just an href to
/page/{pageNumber}. I can take advantage of this by visiting
https://icanhas.cheezburger.com/page/{pageNumber} instead of doing something
more complicated that involves scraping to go to the next page.
To support flags, I'll just use the flag pkg
because it's simple. If it was more complicated I'd consider using something
like cobra
Program usage:
go run main.go --amount 20It will download the first amount images from the site to a local images/
directory (relative path), that will be created if it doesn't exist.
Images are being downloaded twice because they appear on the "Hot today" section and as a post on the homepage. For example, with this post
I noticed the sources are very similar, the only thing that changes is a part of the URL.
- Hot today source:
https://i.chzbgr.com/thumb400/3749638/h1C7C74B7/at-night-thumbnail-includes-one-black-and-white-picture-of-a-cat-sitting-next-to-a-sleeping-man - Post source:
https://i.chzbgr.com/thumb800/3749638/h1C7C74B7/at-night-thumbnail-includes-one-black-and-white-picture-of-a-cat-sitting-next-to-a-sleeping-man
The only thing that changes is the first part of the resource, thumb800 and
thumb400. You can change it to some numbers, like 1200, and receive a scaled
image. Using full you get the full scale image.
To only download one version of every image, I'll change the URLs so that they
are all full. Then remove the duplicates.
Assignment: Enable the above program to run on multiple threads. With an additional Parameter sent as
threadsthat is a number between 1 and 5, parallelize the process while continuing to support theamountflag.
Parallelizing scrapping the pages for images wouldn't be of much use, as we
don't know in advance how many images each has and thus how many pages we need
to scrap to reach the desired amount.
I'll parallelize downloading images. To do so, there will be a channel that has the desired image along with the filename, and there'll be a group of worker goroutines which will download and store them.
Note: This won't necessarily be parallel, but it will be concurrent (and that's all we can do).
Program usage:
go run main.go --amount 20 --threads 3It will download the first amount images from the site to a local images/
directory (relative path), that will be created if it doesn't exist. It uses the
number of threads configured with the threads argument to download the images
concurrently.
Assignment: Describe or showcase a testing strategy for the program. This could be unit, integration, or end to end tests.
To test the program, we need to replace the parts that interact with things
outside of it that we can't control reliably. This is just the GETs to the
main page to get the image url's, the GET to download the images and access to
the file system.
We can replace them with interfaces and create a test that fixes a certain HTML with some images, and check that it downloads them and whatever bytes were obtained are saved to the filesystem with the correct extensions.
To simplify testing of the main functionality, I abstracted away the HTML scrapping with an interface
type Scrapper interface {
CollectImageURLsFrom(page string) ([]string, error)
}so then I had to test that functionality separately.

