1 | /**
|
---|
2 | * @license
|
---|
3 | * Copyright Google LLC All Rights Reserved.
|
---|
4 | *
|
---|
5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
6 | * found in the LICENSE file at https://angular.io/license
|
---|
7 | */
|
---|
8 | import { HttpEventType } from '@angular/common/http';
|
---|
9 | import { Injectable } from '@angular/core';
|
---|
10 | import { Observable } from 'rxjs';
|
---|
11 | import { TestRequest } from './request';
|
---|
12 | /**
|
---|
13 | * A testing backend for `HttpClient` which both acts as an `HttpBackend`
|
---|
14 | * and as the `HttpTestingController`.
|
---|
15 | *
|
---|
16 | * `HttpClientTestingBackend` works by keeping a list of all open requests.
|
---|
17 | * As requests come in, they're added to the list. Users can assert that specific
|
---|
18 | * requests were made and then flush them. In the end, a verify() method asserts
|
---|
19 | * that no unexpected requests were made.
|
---|
20 | *
|
---|
21 | *
|
---|
22 | */
|
---|
23 | export class HttpClientTestingBackend {
|
---|
24 | constructor() {
|
---|
25 | /**
|
---|
26 | * List of pending requests which have not yet been expected.
|
---|
27 | */
|
---|
28 | this.open = [];
|
---|
29 | }
|
---|
30 | /**
|
---|
31 | * Handle an incoming request by queueing it in the list of open requests.
|
---|
32 | */
|
---|
33 | handle(req) {
|
---|
34 | return new Observable((observer) => {
|
---|
35 | const testReq = new TestRequest(req, observer);
|
---|
36 | this.open.push(testReq);
|
---|
37 | observer.next({ type: HttpEventType.Sent });
|
---|
38 | return () => {
|
---|
39 | testReq._cancelled = true;
|
---|
40 | };
|
---|
41 | });
|
---|
42 | }
|
---|
43 | /**
|
---|
44 | * Helper function to search for requests in the list of open requests.
|
---|
45 | */
|
---|
46 | _match(match) {
|
---|
47 | if (typeof match === 'string') {
|
---|
48 | return this.open.filter(testReq => testReq.request.urlWithParams === match);
|
---|
49 | }
|
---|
50 | else if (typeof match === 'function') {
|
---|
51 | return this.open.filter(testReq => match(testReq.request));
|
---|
52 | }
|
---|
53 | else {
|
---|
54 | return this.open.filter(testReq => (!match.method || testReq.request.method === match.method.toUpperCase()) &&
|
---|
55 | (!match.url || testReq.request.urlWithParams === match.url));
|
---|
56 | }
|
---|
57 | }
|
---|
58 | /**
|
---|
59 | * Search for requests in the list of open requests, and return all that match
|
---|
60 | * without asserting anything about the number of matches.
|
---|
61 | */
|
---|
62 | match(match) {
|
---|
63 | const results = this._match(match);
|
---|
64 | results.forEach(result => {
|
---|
65 | const index = this.open.indexOf(result);
|
---|
66 | if (index !== -1) {
|
---|
67 | this.open.splice(index, 1);
|
---|
68 | }
|
---|
69 | });
|
---|
70 | return results;
|
---|
71 | }
|
---|
72 | /**
|
---|
73 | * Expect that a single outstanding request matches the given matcher, and return
|
---|
74 | * it.
|
---|
75 | *
|
---|
76 | * Requests returned through this API will no longer be in the list of open requests,
|
---|
77 | * and thus will not match twice.
|
---|
78 | */
|
---|
79 | expectOne(match, description) {
|
---|
80 | description = description || this.descriptionFromMatcher(match);
|
---|
81 | const matches = this.match(match);
|
---|
82 | if (matches.length > 1) {
|
---|
83 | throw new Error(`Expected one matching request for criteria "${description}", found ${matches.length} requests.`);
|
---|
84 | }
|
---|
85 | if (matches.length === 0) {
|
---|
86 | let message = `Expected one matching request for criteria "${description}", found none.`;
|
---|
87 | if (this.open.length > 0) {
|
---|
88 | // Show the methods and URLs of open requests in the error, for convenience.
|
---|
89 | const requests = this.open
|
---|
90 | .map(testReq => {
|
---|
91 | const url = testReq.request.urlWithParams;
|
---|
92 | const method = testReq.request.method;
|
---|
93 | return `${method} ${url}`;
|
---|
94 | })
|
---|
95 | .join(', ');
|
---|
96 | message += ` Requests received are: ${requests}.`;
|
---|
97 | }
|
---|
98 | throw new Error(message);
|
---|
99 | }
|
---|
100 | return matches[0];
|
---|
101 | }
|
---|
102 | /**
|
---|
103 | * Expect that no outstanding requests match the given matcher, and throw an error
|
---|
104 | * if any do.
|
---|
105 | */
|
---|
106 | expectNone(match, description) {
|
---|
107 | description = description || this.descriptionFromMatcher(match);
|
---|
108 | const matches = this.match(match);
|
---|
109 | if (matches.length > 0) {
|
---|
110 | throw new Error(`Expected zero matching requests for criteria "${description}", found ${matches.length}.`);
|
---|
111 | }
|
---|
112 | }
|
---|
113 | /**
|
---|
114 | * Validate that there are no outstanding requests.
|
---|
115 | */
|
---|
116 | verify(opts = {}) {
|
---|
117 | let open = this.open;
|
---|
118 | // It's possible that some requests may be cancelled, and this is expected.
|
---|
119 | // The user can ask to ignore open requests which have been cancelled.
|
---|
120 | if (opts.ignoreCancelled) {
|
---|
121 | open = open.filter(testReq => !testReq.cancelled);
|
---|
122 | }
|
---|
123 | if (open.length > 0) {
|
---|
124 | // Show the methods and URLs of open requests in the error, for convenience.
|
---|
125 | const requests = open.map(testReq => {
|
---|
126 | const url = testReq.request.urlWithParams.split('?')[0];
|
---|
127 | const method = testReq.request.method;
|
---|
128 | return `${method} ${url}`;
|
---|
129 | })
|
---|
130 | .join(', ');
|
---|
131 | throw new Error(`Expected no open requests, found ${open.length}: ${requests}`);
|
---|
132 | }
|
---|
133 | }
|
---|
134 | descriptionFromMatcher(matcher) {
|
---|
135 | if (typeof matcher === 'string') {
|
---|
136 | return `Match URL: ${matcher}`;
|
---|
137 | }
|
---|
138 | else if (typeof matcher === 'object') {
|
---|
139 | const method = matcher.method || '(any)';
|
---|
140 | const url = matcher.url || '(any)';
|
---|
141 | return `Match method: ${method}, URL: ${url}`;
|
---|
142 | }
|
---|
143 | else {
|
---|
144 | return `Match by function: ${matcher.name}`;
|
---|
145 | }
|
---|
146 | }
|
---|
147 | }
|
---|
148 | HttpClientTestingBackend.decorators = [
|
---|
149 | { type: Injectable }
|
---|
150 | ];
|
---|
151 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"backend.js","sourceRoot":"","sources":["../../../../../../../../packages/common/http/testing/src/backend.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {HttpBackend, HttpEvent, HttpEventType, HttpRequest} from '@angular/common/http';\nimport {Injectable} from '@angular/core';\nimport {Observable, Observer} from 'rxjs';\n\nimport {HttpTestingController, RequestMatch} from './api';\nimport {TestRequest} from './request';\n\n\n/**\n * A testing backend for `HttpClient` which both acts as an `HttpBackend`\n * and as the `HttpTestingController`.\n *\n * `HttpClientTestingBackend` works by keeping a list of all open requests.\n * As requests come in, they're added to the list. Users can assert that specific\n * requests were made and then flush them. In the end, a verify() method asserts\n * that no unexpected requests were made.\n *\n *\n */\n@Injectable()\nexport class HttpClientTestingBackend implements HttpBackend, HttpTestingController {\n  /**\n   * List of pending requests which have not yet been expected.\n   */\n  private open: TestRequest[] = [];\n\n  /**\n   * Handle an incoming request by queueing it in the list of open requests.\n   */\n  handle(req: HttpRequest<any>): Observable<HttpEvent<any>> {\n    return new Observable((observer: Observer<any>) => {\n      const testReq = new TestRequest(req, observer);\n      this.open.push(testReq);\n      observer.next({type: HttpEventType.Sent} as HttpEvent<any>);\n      return () => {\n        testReq._cancelled = true;\n      };\n    });\n  }\n\n  /**\n   * Helper function to search for requests in the list of open requests.\n   */\n  private _match(match: string|RequestMatch|((req: HttpRequest<any>) => boolean)): TestRequest[] {\n    if (typeof match === 'string') {\n      return this.open.filter(testReq => testReq.request.urlWithParams === match);\n    } else if (typeof match === 'function') {\n      return this.open.filter(testReq => match(testReq.request));\n    } else {\n      return this.open.filter(\n          testReq => (!match.method || testReq.request.method === match.method.toUpperCase()) &&\n              (!match.url || testReq.request.urlWithParams === match.url));\n    }\n  }\n\n  /**\n   * Search for requests in the list of open requests, and return all that match\n   * without asserting anything about the number of matches.\n   */\n  match(match: string|RequestMatch|((req: HttpRequest<any>) => boolean)): TestRequest[] {\n    const results = this._match(match);\n    results.forEach(result => {\n      const index = this.open.indexOf(result);\n      if (index !== -1) {\n        this.open.splice(index, 1);\n      }\n    });\n    return results;\n  }\n\n  /**\n   * Expect that a single outstanding request matches the given matcher, and return\n   * it.\n   *\n   * Requests returned through this API will no longer be in the list of open requests,\n   * and thus will not match twice.\n   */\n  expectOne(match: string|RequestMatch|((req: HttpRequest<any>) => boolean), description?: string):\n      TestRequest {\n    description = description || this.descriptionFromMatcher(match);\n    const matches = this.match(match);\n    if (matches.length > 1) {\n      throw new Error(`Expected one matching request for criteria \"${description}\", found ${\n          matches.length} requests.`);\n    }\n    if (matches.length === 0) {\n      let message = `Expected one matching request for criteria \"${description}\", found none.`;\n      if (this.open.length > 0) {\n        // Show the methods and URLs of open requests in the error, for convenience.\n        const requests = this.open\n                             .map(testReq => {\n                               const url = testReq.request.urlWithParams;\n                               const method = testReq.request.method;\n                               return `${method} ${url}`;\n                             })\n                             .join(', ');\n        message += ` Requests received are: ${requests}.`;\n      }\n      throw new Error(message);\n    }\n    return matches[0];\n  }\n\n  /**\n   * Expect that no outstanding requests match the given matcher, and throw an error\n   * if any do.\n   */\n  expectNone(match: string|RequestMatch|((req: HttpRequest<any>) => boolean), description?: string):\n      void {\n    description = description || this.descriptionFromMatcher(match);\n    const matches = this.match(match);\n    if (matches.length > 0) {\n      throw new Error(`Expected zero matching requests for criteria \"${description}\", found ${\n          matches.length}.`);\n    }\n  }\n\n  /**\n   * Validate that there are no outstanding requests.\n   */\n  verify(opts: {ignoreCancelled?: boolean} = {}): void {\n    let open = this.open;\n    // It's possible that some requests may be cancelled, and this is expected.\n    // The user can ask to ignore open requests which have been cancelled.\n    if (opts.ignoreCancelled) {\n      open = open.filter(testReq => !testReq.cancelled);\n    }\n    if (open.length > 0) {\n      // Show the methods and URLs of open requests in the error, for convenience.\n      const requests = open.map(testReq => {\n                             const url = testReq.request.urlWithParams.split('?')[0];\n                             const method = testReq.request.method;\n                             return `${method} ${url}`;\n                           })\n                           .join(', ');\n      throw new Error(`Expected no open requests, found ${open.length}: ${requests}`);\n    }\n  }\n\n  private descriptionFromMatcher(matcher: string|RequestMatch|\n                                 ((req: HttpRequest<any>) => boolean)): string {\n    if (typeof matcher === 'string') {\n      return `Match URL: ${matcher}`;\n    } else if (typeof matcher === 'object') {\n      const method = matcher.method || '(any)';\n      const url = matcher.url || '(any)';\n      return `Match method: ${method}, URL: ${url}`;\n    } else {\n      return `Match by function: ${matcher.name}`;\n    }\n  }\n}\n"]} |
---|