Some checks failed
		
		
	
	Types tests / Test (lts/*) (push) Has been cancelled
				
			Lint / Lint (lts/*) (push) Has been cancelled
				
			CodeQL / Analyze (javascript) (push) Has been cancelled
				
			CI / Test (20) (push) Has been cancelled
				
			CI / Test (22) (push) Has been cancelled
				
			CI / Test (24) (push) Has been cancelled
				
			
		
			
				
	
	
		
			201 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			201 lines
		
	
	
		
			6.3 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* Copyright 2017 Mozilla Foundation
 | 
						|
 *
 | 
						|
 * Licensed under the Apache License, Version 2.0 (the "License");
 | 
						|
 * you may not use this file except in compliance with the License.
 | 
						|
 * You may obtain a copy of the License at
 | 
						|
 *
 | 
						|
 *     http://www.apache.org/licenses/LICENSE-2.0
 | 
						|
 *
 | 
						|
 * Unless required by applicable law or agreed to in writing, software
 | 
						|
 * distributed under the License is distributed on an "AS IS" BASIS,
 | 
						|
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 | 
						|
 * See the License for the specific language governing permissions and
 | 
						|
 * limitations under the License.
 | 
						|
 */
 | 
						|
 | 
						|
import { AbortException, ResponseException } from "../../src/shared/util.js";
 | 
						|
import { PDFNetworkStream } from "../../src/display/network.js";
 | 
						|
import { testCrossOriginRedirects } from "./common_pdfstream_tests.js";
 | 
						|
import { TestPdfsServer } from "./test_utils.js";
 | 
						|
 | 
						|
describe("network", function () {
 | 
						|
  const pdf1 = new URL("../pdfs/tracemonkey.pdf", window.location).href;
 | 
						|
  const pdf1Length = 1016315;
 | 
						|
 | 
						|
  it("read without stream and range", async function () {
 | 
						|
    const stream = new PDFNetworkStream({
 | 
						|
      url: pdf1,
 | 
						|
      rangeChunkSize: 65536,
 | 
						|
      disableStream: true,
 | 
						|
      disableRange: true,
 | 
						|
    });
 | 
						|
 | 
						|
    const fullReader = stream.getFullReader();
 | 
						|
 | 
						|
    let isStreamingSupported, isRangeSupported;
 | 
						|
    await fullReader.headersReady.then(function () {
 | 
						|
      isStreamingSupported = fullReader.isStreamingSupported;
 | 
						|
      isRangeSupported = fullReader.isRangeSupported;
 | 
						|
    });
 | 
						|
 | 
						|
    let len = 0,
 | 
						|
      count = 0;
 | 
						|
    const read = function () {
 | 
						|
      return fullReader.read().then(function (result) {
 | 
						|
        if (result.done) {
 | 
						|
          return undefined;
 | 
						|
        }
 | 
						|
        count++;
 | 
						|
        len += result.value.byteLength;
 | 
						|
        return read();
 | 
						|
      });
 | 
						|
    };
 | 
						|
 | 
						|
    await read();
 | 
						|
 | 
						|
    expect(len).toEqual(pdf1Length);
 | 
						|
    expect(count).toEqual(1);
 | 
						|
    expect(isStreamingSupported).toEqual(false);
 | 
						|
    expect(isRangeSupported).toEqual(false);
 | 
						|
  });
 | 
						|
 | 
						|
  it("read custom ranges", async function () {
 | 
						|
    // We don't test on browsers that don't support range request, so
 | 
						|
    // requiring this test to pass.
 | 
						|
    const rangeSize = 32768;
 | 
						|
    const stream = new PDFNetworkStream({
 | 
						|
      url: pdf1,
 | 
						|
      length: pdf1Length,
 | 
						|
      rangeChunkSize: rangeSize,
 | 
						|
      disableStream: true,
 | 
						|
      disableRange: false,
 | 
						|
    });
 | 
						|
 | 
						|
    const fullReader = stream.getFullReader();
 | 
						|
 | 
						|
    let isStreamingSupported, isRangeSupported, fullReaderCancelled;
 | 
						|
    await fullReader.headersReady.then(function () {
 | 
						|
      isStreamingSupported = fullReader.isStreamingSupported;
 | 
						|
      isRangeSupported = fullReader.isRangeSupported;
 | 
						|
      // we shall be able to close the full reader without issues
 | 
						|
      fullReader.cancel(new AbortException("Don't need fullReader."));
 | 
						|
      fullReaderCancelled = true;
 | 
						|
    });
 | 
						|
 | 
						|
    // Skipping fullReader results, requesting something from the PDF end.
 | 
						|
    const tailSize = pdf1Length % rangeSize || rangeSize;
 | 
						|
 | 
						|
    const range1Reader = stream.getRangeReader(
 | 
						|
      pdf1Length - tailSize - rangeSize,
 | 
						|
      pdf1Length - tailSize
 | 
						|
    );
 | 
						|
    const range2Reader = stream.getRangeReader(
 | 
						|
      pdf1Length - tailSize,
 | 
						|
      pdf1Length
 | 
						|
    );
 | 
						|
 | 
						|
    const result1 = { value: 0 },
 | 
						|
      result2 = { value: 0 };
 | 
						|
    const read = function (reader, lenResult) {
 | 
						|
      return reader.read().then(function (result) {
 | 
						|
        if (result.done) {
 | 
						|
          return undefined;
 | 
						|
        }
 | 
						|
        lenResult.value += result.value.byteLength;
 | 
						|
        return read(reader, lenResult);
 | 
						|
      });
 | 
						|
    };
 | 
						|
 | 
						|
    await Promise.all([
 | 
						|
      read(range1Reader, result1),
 | 
						|
      read(range2Reader, result2),
 | 
						|
    ]);
 | 
						|
 | 
						|
    expect(result1.value).toEqual(rangeSize);
 | 
						|
    expect(result2.value).toEqual(tailSize);
 | 
						|
    expect(isStreamingSupported).toEqual(false);
 | 
						|
    expect(isRangeSupported).toEqual(true);
 | 
						|
    expect(fullReaderCancelled).toEqual(true);
 | 
						|
  });
 | 
						|
 | 
						|
  it(`handle reading ranges with missing/invalid "Content-Range" header`, async function () {
 | 
						|
    if (globalThis.chrome) {
 | 
						|
      pending("Fails intermittently in Google Chrome.");
 | 
						|
    }
 | 
						|
 | 
						|
    async function readRanges(mode) {
 | 
						|
      const rangeSize = 32768;
 | 
						|
      const stream = new PDFNetworkStream({
 | 
						|
        url: `${pdf1}?test-network-break-ranges=${mode}`,
 | 
						|
        length: pdf1Length,
 | 
						|
        rangeChunkSize: rangeSize,
 | 
						|
        disableStream: true,
 | 
						|
        disableRange: false,
 | 
						|
      });
 | 
						|
 | 
						|
      const fullReader = stream.getFullReader();
 | 
						|
 | 
						|
      await fullReader.headersReady;
 | 
						|
      // Ensure that range requests are supported.
 | 
						|
      expect(fullReader.isRangeSupported).toEqual(true);
 | 
						|
      // We shall be able to close the full reader without issues.
 | 
						|
      fullReader.cancel(new AbortException("Don't need fullReader."));
 | 
						|
 | 
						|
      const rangeReader = stream.getRangeReader(
 | 
						|
        pdf1Length - rangeSize,
 | 
						|
        pdf1Length
 | 
						|
      );
 | 
						|
 | 
						|
      try {
 | 
						|
        await rangeReader.read();
 | 
						|
 | 
						|
        // Shouldn't get here.
 | 
						|
        expect(false).toEqual(true);
 | 
						|
      } catch (ex) {
 | 
						|
        expect(ex instanceof ResponseException).toEqual(true);
 | 
						|
        expect(ex.status).toEqual(0);
 | 
						|
        expect(ex.missing).toEqual(false);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    await Promise.all([readRanges("missing"), readRanges("invalid")]);
 | 
						|
  });
 | 
						|
 | 
						|
  describe("Redirects", function () {
 | 
						|
    beforeAll(async function () {
 | 
						|
      await TestPdfsServer.ensureStarted();
 | 
						|
    });
 | 
						|
 | 
						|
    afterAll(async function () {
 | 
						|
      await TestPdfsServer.ensureStopped();
 | 
						|
    });
 | 
						|
 | 
						|
    it("redirects allowed if all responses are same-origin", async function () {
 | 
						|
      await testCrossOriginRedirects({
 | 
						|
        PDFStreamClass: PDFNetworkStream,
 | 
						|
        redirectIfRange: false,
 | 
						|
        async testRangeReader(rangeReader) {
 | 
						|
          await expectAsync(rangeReader.read()).toBeResolved();
 | 
						|
        },
 | 
						|
      });
 | 
						|
    });
 | 
						|
 | 
						|
    it("redirects blocked if any response is cross-origin", async function () {
 | 
						|
      await testCrossOriginRedirects({
 | 
						|
        PDFStreamClass: PDFNetworkStream,
 | 
						|
        redirectIfRange: true,
 | 
						|
        async testRangeReader(rangeReader) {
 | 
						|
          // When read (sync), error should be reported.
 | 
						|
          await expectAsync(rangeReader.read()).toBeRejectedWithError(
 | 
						|
            /^Expected range response-origin "http:.*" to match "http:.*"\.$/
 | 
						|
          );
 | 
						|
          // When read again (async), error should be consistent.
 | 
						|
          await expectAsync(rangeReader.read()).toBeRejectedWithError(
 | 
						|
            /^Expected range response-origin "http:.*" to match "http:.*"\.$/
 | 
						|
          );
 | 
						|
        },
 | 
						|
      });
 | 
						|
    });
 | 
						|
  });
 | 
						|
});
 |