40

Can anyone tell me how to write test case for a link to download pdf file using jasmine framework ? Thanks in advance.

ted
  • 13,596
  • 9
  • 65
  • 107
user3061796
  • 553
  • 2
  • 5
  • 6

5 Answers5

47

I can currently set download path location

Chrome

capabilities: {
    'browserName': 'chrome',
    'platform': 'ANY',
    'version': 'ANY',
    'chromeOptions': {
        // Get rid of --ignore-certificate yellow warning
        args: ['--no-sandbox', '--test-type=browser'],
        // Set download path and avoid prompting for download even though
        // this is already the default on Chrome but for completeness
        prefs: {
            'download': {
                'prompt_for_download': false,
                'default_directory': '/e2e/downloads/',
            }
        }
    }
}

For remote testing you would need a more complex infrastructure like setting up a Samba share or network shared directory destination.

Firefox


var FirefoxProfile = require('firefox-profile');
var q = require('q');

[...]
getMultiCapabilities: getFirefoxProfile,
framework: 'jasmine2',

[...]
function getFirefoxProfile() {
    "use strict";
    var deferred = q.defer();
    var firefoxProfile = new FirefoxProfile();
    firefoxProfile.setPreference("browser.download.folderList", 2);
    firefoxProfile.setPreference("browser.download.manager.showWhenStarting", false);
    firefoxProfile.setPreference("browser.download.dir", '/tmp');
    firefoxProfile.setPreference("browser.helperApps.neverAsk.saveToDisk", "application/vnd.openxmlformats-officedocument.wordprocessingml.document");

    firefoxProfile.encoded(function(encodedProfile) {
        var multiCapabilities = [{
            browserName: 'firefox',
            firefox_profile : encodedProfile
        }];
        deferred.resolve(multiCapabilities);
    });
    return deferred.promise;
}

Finally and maybe obvious, to trigger the download you click on the download link as you know, e.g.

$('a.some-download-link').click();
Jason
  • 920
  • 8
  • 19
Leo Gallucci
  • 16,355
  • 12
  • 77
  • 110
  • Hello Leo, i'm trying to set a default path location for Chrome, but apparently your solution didn't worked for me. – andrepm Jan 12 '15 at 19:51
  • @LeoGallucci With above code, Firefox doesn't spawn and no errors. But if I use browserName:'firefox' in export.config, it opens firefox successfully. What could be reason? Thanks! – rohitkadam19 Jul 07 '16 at 06:53
  • 1
    Yes, of course if you want to use Firefox you need to set browserName:'firefox' – Leo Gallucci Jul 07 '16 at 11:36
  • @LeoGallucci I would need to use 1 config file and pass browserName as parameter how can i achieve this, i see for 'FIREFOX' "getMultiCapabilities" is used and for 'CHROME' "capabilities" is used. help would be greatly appreciated. – Nick Jan 09 '17 at 06:56
  • Has anyone had success developing the same for Internet Explorer? – Jeremy Kahan Mar 05 '17 at 16:04
46

I needed to check the contents of the downloaded file (a CSV export in my case) against an expected result, and found the following to work:

var filename = '/tmp/export.csv';
var fs = require('fs');

if (fs.existsSync(filename)) {
    // Make sure the browser doesn't have to rename the download.
    fs.unlinkSync(filename);
}

$('a.download').click();

browser.driver.wait(function() {
    // Wait until the file has been downloaded.
    // We need to wait thus as otherwise protractor has a nasty habit of
    // trying to do any following tests while the file is still being
    // downloaded and hasn't been moved to its final location.
    return fs.existsSync(filename);
}, 30000).then(function() {
    // Do whatever checks you need here.  This is a simple comparison;
    // for a larger file you might want to do calculate the file's MD5
    // hash and see if it matches what you expect.
    expect(fs.readFileSync(filename, { encoding: 'utf8' })).toEqual(
        "A,B,C\r\n"
    );
});

I found Leo's configuration suggestion helpful for allowing the download to be saved somewhere accessible.

The 30000ms timeout is the default, so could be omitted, but I'm leaving it in as a reminder in case someone would like to change it.

Stephen Baillie
  • 509
  • 4
  • 9
1

it could be the test for checking href attribute like so:

var link = element(by.css("a.pdf"));
expect(link.getAttribute('href')).toEqual('someExactUrl');
Stepan Suvorov
  • 25,118
  • 26
  • 108
  • 176
1

The solutions above would not work for remote browser testing, e.g. via BrowserStack. An alternative solution, just for Chrome, could look like this:

if ((await browser.getCapabilities()).get('browserName') === 'chrome') {
  await browser.driver.get('chrome://downloads/');
  const items =
    await browser.executeScript('return downloads.Manager.get().items_') as any[];
  expect(items.length).toBe(1);
  expect(items[0].file_name).toBe('some.pdf');
}
Rene Hamburger
  • 2,003
  • 16
  • 17
-1

One thing I've done in the past is to use an HTTP HEAD command. Basically, it's the same as a 'GET', but it only retrieves the headers.

Unfortunately, the web server needs to support 'HEAD' explicitly. If it does, you can actually try the URL and then check for 'application/pdf' in the Content-Type, without having to actually download the file.

If the server isn't set up to support HEAD, you can probably just check the link text like was suggested above.

Tom McKearney
  • 315
  • 2
  • 11