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 | /**
|
---|
9 | * Provides encoding and decoding of URL parameter and query-string values.
|
---|
10 | *
|
---|
11 | * Serializes and parses URL parameter keys and values to encode and decode them.
|
---|
12 | * If you pass URL query parameters without encoding,
|
---|
13 | * the query parameters can be misinterpreted at the receiving end.
|
---|
14 | *
|
---|
15 | *
|
---|
16 | * @publicApi
|
---|
17 | */
|
---|
18 | export class HttpUrlEncodingCodec {
|
---|
19 | /**
|
---|
20 | * Encodes a key name for a URL parameter or query-string.
|
---|
21 | * @param key The key name.
|
---|
22 | * @returns The encoded key name.
|
---|
23 | */
|
---|
24 | encodeKey(key) {
|
---|
25 | return standardEncoding(key);
|
---|
26 | }
|
---|
27 | /**
|
---|
28 | * Encodes the value of a URL parameter or query-string.
|
---|
29 | * @param value The value.
|
---|
30 | * @returns The encoded value.
|
---|
31 | */
|
---|
32 | encodeValue(value) {
|
---|
33 | return standardEncoding(value);
|
---|
34 | }
|
---|
35 | /**
|
---|
36 | * Decodes an encoded URL parameter or query-string key.
|
---|
37 | * @param key The encoded key name.
|
---|
38 | * @returns The decoded key name.
|
---|
39 | */
|
---|
40 | decodeKey(key) {
|
---|
41 | return decodeURIComponent(key);
|
---|
42 | }
|
---|
43 | /**
|
---|
44 | * Decodes an encoded URL parameter or query-string value.
|
---|
45 | * @param value The encoded value.
|
---|
46 | * @returns The decoded value.
|
---|
47 | */
|
---|
48 | decodeValue(value) {
|
---|
49 | return decodeURIComponent(value);
|
---|
50 | }
|
---|
51 | }
|
---|
52 | function paramParser(rawParams, codec) {
|
---|
53 | const map = new Map();
|
---|
54 | if (rawParams.length > 0) {
|
---|
55 | // The `window.location.search` can be used while creating an instance of the `HttpParams` class
|
---|
56 | // (e.g. `new HttpParams({ fromString: window.location.search })`). The `window.location.search`
|
---|
57 | // may start with the `?` char, so we strip it if it's present.
|
---|
58 | const params = rawParams.replace(/^\?/, '').split('&');
|
---|
59 | params.forEach((param) => {
|
---|
60 | const eqIdx = param.indexOf('=');
|
---|
61 | const [key, val] = eqIdx == -1 ?
|
---|
62 | [codec.decodeKey(param), ''] :
|
---|
63 | [codec.decodeKey(param.slice(0, eqIdx)), codec.decodeValue(param.slice(eqIdx + 1))];
|
---|
64 | const list = map.get(key) || [];
|
---|
65 | list.push(val);
|
---|
66 | map.set(key, list);
|
---|
67 | });
|
---|
68 | }
|
---|
69 | return map;
|
---|
70 | }
|
---|
71 | /**
|
---|
72 | * Encode input string with standard encodeURIComponent and then un-encode specific characters.
|
---|
73 | */
|
---|
74 | const STANDARD_ENCODING_REGEX = /%(\d[a-f0-9])/gi;
|
---|
75 | const STANDARD_ENCODING_REPLACEMENTS = {
|
---|
76 | '40': '@',
|
---|
77 | '3A': ':',
|
---|
78 | '24': '$',
|
---|
79 | '2C': ',',
|
---|
80 | '3B': ';',
|
---|
81 | '2B': '+',
|
---|
82 | '3D': '=',
|
---|
83 | '3F': '?',
|
---|
84 | '2F': '/',
|
---|
85 | };
|
---|
86 | function standardEncoding(v) {
|
---|
87 | return encodeURIComponent(v).replace(STANDARD_ENCODING_REGEX, (s, t) => { var _a; return (_a = STANDARD_ENCODING_REPLACEMENTS[t]) !== null && _a !== void 0 ? _a : s; });
|
---|
88 | }
|
---|
89 | function valueToString(value) {
|
---|
90 | return `${value}`;
|
---|
91 | }
|
---|
92 | /**
|
---|
93 | * An HTTP request/response body that represents serialized parameters,
|
---|
94 | * per the MIME type `application/x-www-form-urlencoded`.
|
---|
95 | *
|
---|
96 | * This class is immutable; all mutation operations return a new instance.
|
---|
97 | *
|
---|
98 | * @publicApi
|
---|
99 | */
|
---|
100 | export class HttpParams {
|
---|
101 | constructor(options = {}) {
|
---|
102 | this.updates = null;
|
---|
103 | this.cloneFrom = null;
|
---|
104 | this.encoder = options.encoder || new HttpUrlEncodingCodec();
|
---|
105 | if (!!options.fromString) {
|
---|
106 | if (!!options.fromObject) {
|
---|
107 | throw new Error(`Cannot specify both fromString and fromObject.`);
|
---|
108 | }
|
---|
109 | this.map = paramParser(options.fromString, this.encoder);
|
---|
110 | }
|
---|
111 | else if (!!options.fromObject) {
|
---|
112 | this.map = new Map();
|
---|
113 | Object.keys(options.fromObject).forEach(key => {
|
---|
114 | const value = options.fromObject[key];
|
---|
115 | this.map.set(key, Array.isArray(value) ? value : [value]);
|
---|
116 | });
|
---|
117 | }
|
---|
118 | else {
|
---|
119 | this.map = null;
|
---|
120 | }
|
---|
121 | }
|
---|
122 | /**
|
---|
123 | * Reports whether the body includes one or more values for a given parameter.
|
---|
124 | * @param param The parameter name.
|
---|
125 | * @returns True if the parameter has one or more values,
|
---|
126 | * false if it has no value or is not present.
|
---|
127 | */
|
---|
128 | has(param) {
|
---|
129 | this.init();
|
---|
130 | return this.map.has(param);
|
---|
131 | }
|
---|
132 | /**
|
---|
133 | * Retrieves the first value for a parameter.
|
---|
134 | * @param param The parameter name.
|
---|
135 | * @returns The first value of the given parameter,
|
---|
136 | * or `null` if the parameter is not present.
|
---|
137 | */
|
---|
138 | get(param) {
|
---|
139 | this.init();
|
---|
140 | const res = this.map.get(param);
|
---|
141 | return !!res ? res[0] : null;
|
---|
142 | }
|
---|
143 | /**
|
---|
144 | * Retrieves all values for a parameter.
|
---|
145 | * @param param The parameter name.
|
---|
146 | * @returns All values in a string array,
|
---|
147 | * or `null` if the parameter not present.
|
---|
148 | */
|
---|
149 | getAll(param) {
|
---|
150 | this.init();
|
---|
151 | return this.map.get(param) || null;
|
---|
152 | }
|
---|
153 | /**
|
---|
154 | * Retrieves all the parameters for this body.
|
---|
155 | * @returns The parameter names in a string array.
|
---|
156 | */
|
---|
157 | keys() {
|
---|
158 | this.init();
|
---|
159 | return Array.from(this.map.keys());
|
---|
160 | }
|
---|
161 | /**
|
---|
162 | * Appends a new value to existing values for a parameter.
|
---|
163 | * @param param The parameter name.
|
---|
164 | * @param value The new value to add.
|
---|
165 | * @return A new body with the appended value.
|
---|
166 | */
|
---|
167 | append(param, value) {
|
---|
168 | return this.clone({ param, value, op: 'a' });
|
---|
169 | }
|
---|
170 | /**
|
---|
171 | * Constructs a new body with appended values for the given parameter name.
|
---|
172 | * @param params parameters and values
|
---|
173 | * @return A new body with the new value.
|
---|
174 | */
|
---|
175 | appendAll(params) {
|
---|
176 | const updates = [];
|
---|
177 | Object.keys(params).forEach(param => {
|
---|
178 | const value = params[param];
|
---|
179 | if (Array.isArray(value)) {
|
---|
180 | value.forEach(_value => {
|
---|
181 | updates.push({ param, value: _value, op: 'a' });
|
---|
182 | });
|
---|
183 | }
|
---|
184 | else {
|
---|
185 | updates.push({ param, value: value, op: 'a' });
|
---|
186 | }
|
---|
187 | });
|
---|
188 | return this.clone(updates);
|
---|
189 | }
|
---|
190 | /**
|
---|
191 | * Replaces the value for a parameter.
|
---|
192 | * @param param The parameter name.
|
---|
193 | * @param value The new value.
|
---|
194 | * @return A new body with the new value.
|
---|
195 | */
|
---|
196 | set(param, value) {
|
---|
197 | return this.clone({ param, value, op: 's' });
|
---|
198 | }
|
---|
199 | /**
|
---|
200 | * Removes a given value or all values from a parameter.
|
---|
201 | * @param param The parameter name.
|
---|
202 | * @param value The value to remove, if provided.
|
---|
203 | * @return A new body with the given value removed, or with all values
|
---|
204 | * removed if no value is specified.
|
---|
205 | */
|
---|
206 | delete(param, value) {
|
---|
207 | return this.clone({ param, value, op: 'd' });
|
---|
208 | }
|
---|
209 | /**
|
---|
210 | * Serializes the body to an encoded string, where key-value pairs (separated by `=`) are
|
---|
211 | * separated by `&`s.
|
---|
212 | */
|
---|
213 | toString() {
|
---|
214 | this.init();
|
---|
215 | return this.keys()
|
---|
216 | .map(key => {
|
---|
217 | const eKey = this.encoder.encodeKey(key);
|
---|
218 | // `a: ['1']` produces `'a=1'`
|
---|
219 | // `b: []` produces `''`
|
---|
220 | // `c: ['1', '2']` produces `'c=1&c=2'`
|
---|
221 | return this.map.get(key).map(value => eKey + '=' + this.encoder.encodeValue(value))
|
---|
222 | .join('&');
|
---|
223 | })
|
---|
224 | // filter out empty values because `b: []` produces `''`
|
---|
225 | // which results in `a=1&&c=1&c=2` instead of `a=1&c=1&c=2` if we don't
|
---|
226 | .filter(param => param !== '')
|
---|
227 | .join('&');
|
---|
228 | }
|
---|
229 | clone(update) {
|
---|
230 | const clone = new HttpParams({ encoder: this.encoder });
|
---|
231 | clone.cloneFrom = this.cloneFrom || this;
|
---|
232 | clone.updates = (this.updates || []).concat(update);
|
---|
233 | return clone;
|
---|
234 | }
|
---|
235 | init() {
|
---|
236 | if (this.map === null) {
|
---|
237 | this.map = new Map();
|
---|
238 | }
|
---|
239 | if (this.cloneFrom !== null) {
|
---|
240 | this.cloneFrom.init();
|
---|
241 | this.cloneFrom.keys().forEach(key => this.map.set(key, this.cloneFrom.map.get(key)));
|
---|
242 | this.updates.forEach(update => {
|
---|
243 | switch (update.op) {
|
---|
244 | case 'a':
|
---|
245 | case 's':
|
---|
246 | const base = (update.op === 'a' ? this.map.get(update.param) : undefined) || [];
|
---|
247 | base.push(valueToString(update.value));
|
---|
248 | this.map.set(update.param, base);
|
---|
249 | break;
|
---|
250 | case 'd':
|
---|
251 | if (update.value !== undefined) {
|
---|
252 | let base = this.map.get(update.param) || [];
|
---|
253 | const idx = base.indexOf(valueToString(update.value));
|
---|
254 | if (idx !== -1) {
|
---|
255 | base.splice(idx, 1);
|
---|
256 | }
|
---|
257 | if (base.length > 0) {
|
---|
258 | this.map.set(update.param, base);
|
---|
259 | }
|
---|
260 | else {
|
---|
261 | this.map.delete(update.param);
|
---|
262 | }
|
---|
263 | }
|
---|
264 | else {
|
---|
265 | this.map.delete(update.param);
|
---|
266 | break;
|
---|
267 | }
|
---|
268 | }
|
---|
269 | });
|
---|
270 | this.cloneFrom = this.updates = null;
|
---|
271 | }
|
---|
272 | }
|
---|
273 | }
|
---|
274 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"params.js","sourceRoot":"","sources":["../../../../../../../packages/common/http/src/params.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\n/**\n * A codec for encoding and decoding parameters in URLs.\n *\n * Used by `HttpParams`.\n *\n * @publicApi\n **/\nexport interface HttpParameterCodec {\n  encodeKey(key: string): string;\n  encodeValue(value: string): string;\n\n  decodeKey(key: string): string;\n  decodeValue(value: string): string;\n}\n\n/**\n * Provides encoding and decoding of URL parameter and query-string values.\n *\n * Serializes and parses URL parameter keys and values to encode and decode them.\n * If you pass URL query parameters without encoding,\n * the query parameters can be misinterpreted at the receiving end.\n *\n *\n * @publicApi\n */\nexport class HttpUrlEncodingCodec implements HttpParameterCodec {\n  /**\n   * Encodes a key name for a URL parameter or query-string.\n   * @param key The key name.\n   * @returns The encoded key name.\n   */\n  encodeKey(key: string): string {\n    return standardEncoding(key);\n  }\n\n  /**\n   * Encodes the value of a URL parameter or query-string.\n   * @param value The value.\n   * @returns The encoded value.\n   */\n  encodeValue(value: string): string {\n    return standardEncoding(value);\n  }\n\n  /**\n   * Decodes an encoded URL parameter or query-string key.\n   * @param key The encoded key name.\n   * @returns The decoded key name.\n   */\n  decodeKey(key: string): string {\n    return decodeURIComponent(key);\n  }\n\n  /**\n   * Decodes an encoded URL parameter or query-string value.\n   * @param value The encoded value.\n   * @returns The decoded value.\n   */\n  decodeValue(value: string) {\n    return decodeURIComponent(value);\n  }\n}\n\n\nfunction paramParser(rawParams: string, codec: HttpParameterCodec): Map<string, string[]> {\n  const map = new Map<string, string[]>();\n  if (rawParams.length > 0) {\n    // The `window.location.search` can be used while creating an instance of the `HttpParams` class\n    // (e.g. `new HttpParams({ fromString: window.location.search })`). The `window.location.search`\n    // may start with the `?` char, so we strip it if it's present.\n    const params: string[] = rawParams.replace(/^\\?/, '').split('&');\n    params.forEach((param: string) => {\n      const eqIdx = param.indexOf('=');\n      const [key, val]: string[] = eqIdx == -1 ?\n          [codec.decodeKey(param), ''] :\n          [codec.decodeKey(param.slice(0, eqIdx)), codec.decodeValue(param.slice(eqIdx + 1))];\n      const list = map.get(key) || [];\n      list.push(val);\n      map.set(key, list);\n    });\n  }\n  return map;\n}\n\n/**\n * Encode input string with standard encodeURIComponent and then un-encode specific characters.\n */\nconst STANDARD_ENCODING_REGEX = /%(\\d[a-f0-9])/gi;\nconst STANDARD_ENCODING_REPLACEMENTS: {[x: string]: string} = {\n  '40': '@',\n  '3A': ':',\n  '24': '$',\n  '2C': ',',\n  '3B': ';',\n  '2B': '+',\n  '3D': '=',\n  '3F': '?',\n  '2F': '/',\n};\n\nfunction standardEncoding(v: string): string {\n  return encodeURIComponent(v).replace(\n      STANDARD_ENCODING_REGEX, (s, t) => STANDARD_ENCODING_REPLACEMENTS[t] ?? s);\n}\n\nfunction valueToString(value: string|number|boolean): string {\n  return `${value}`;\n}\n\ninterface Update {\n  param: string;\n  value?: string|number|boolean;\n  op: 'a'|'d'|'s';\n}\n\n/**\n * Options used to construct an `HttpParams` instance.\n *\n * @publicApi\n */\nexport interface HttpParamsOptions {\n  /**\n   * String representation of the HTTP parameters in URL-query-string format.\n   * Mutually exclusive with `fromObject`.\n   */\n  fromString?: string;\n\n  /** Object map of the HTTP parameters. Mutually exclusive with `fromString`. */\n  fromObject?: {[param: string]: string|number|boolean|ReadonlyArray<string|number|boolean>};\n\n  /** Encoding codec used to parse and serialize the parameters. */\n  encoder?: HttpParameterCodec;\n}\n\n/**\n * An HTTP request/response body that represents serialized parameters,\n * per the MIME type `application/x-www-form-urlencoded`.\n *\n * This class is immutable; all mutation operations return a new instance.\n *\n * @publicApi\n */\nexport class HttpParams {\n  private map: Map<string, string[]>|null;\n  private encoder: HttpParameterCodec;\n  private updates: Update[]|null = null;\n  private cloneFrom: HttpParams|null = null;\n\n  constructor(options: HttpParamsOptions = {} as HttpParamsOptions) {\n    this.encoder = options.encoder || new HttpUrlEncodingCodec();\n    if (!!options.fromString) {\n      if (!!options.fromObject) {\n        throw new Error(`Cannot specify both fromString and fromObject.`);\n      }\n      this.map = paramParser(options.fromString, this.encoder);\n    } else if (!!options.fromObject) {\n      this.map = new Map<string, string[]>();\n      Object.keys(options.fromObject).forEach(key => {\n        const value = (options.fromObject as any)[key];\n        this.map!.set(key, Array.isArray(value) ? value : [value]);\n      });\n    } else {\n      this.map = null;\n    }\n  }\n\n  /**\n   * Reports whether the body includes one or more values for a given parameter.\n   * @param param The parameter name.\n   * @returns True if the parameter has one or more values,\n   * false if it has no value or is not present.\n   */\n  has(param: string): boolean {\n    this.init();\n    return this.map!.has(param);\n  }\n\n  /**\n   * Retrieves the first value for a parameter.\n   * @param param The parameter name.\n   * @returns The first value of the given parameter,\n   * or `null` if the parameter is not present.\n   */\n  get(param: string): string|null {\n    this.init();\n    const res = this.map!.get(param);\n    return !!res ? res[0] : null;\n  }\n\n  /**\n   * Retrieves all values for a  parameter.\n   * @param param The parameter name.\n   * @returns All values in a string array,\n   * or `null` if the parameter not present.\n   */\n  getAll(param: string): string[]|null {\n    this.init();\n    return this.map!.get(param) || null;\n  }\n\n  /**\n   * Retrieves all the parameters for this body.\n   * @returns The parameter names in a string array.\n   */\n  keys(): string[] {\n    this.init();\n    return Array.from(this.map!.keys());\n  }\n\n  /**\n   * Appends a new value to existing values for a parameter.\n   * @param param The parameter name.\n   * @param value The new value to add.\n   * @return A new body with the appended value.\n   */\n  append(param: string, value: string|number|boolean): HttpParams {\n    return this.clone({param, value, op: 'a'});\n  }\n\n  /**\n   * Constructs a new body with appended values for the given parameter name.\n   * @param params parameters and values\n   * @return A new body with the new value.\n   */\n  appendAll(params: {[param: string]: string|number|boolean|ReadonlyArray<string|number|boolean>}):\n      HttpParams {\n    const updates: Update[] = [];\n    Object.keys(params).forEach(param => {\n      const value = params[param];\n      if (Array.isArray(value)) {\n        value.forEach(_value => {\n          updates.push({param, value: _value, op: 'a'});\n        });\n      } else {\n        updates.push({param, value: value as (string | number | boolean), op: 'a'});\n      }\n    });\n    return this.clone(updates);\n  }\n\n  /**\n   * Replaces the value for a parameter.\n   * @param param The parameter name.\n   * @param value The new value.\n   * @return A new body with the new value.\n   */\n  set(param: string, value: string|number|boolean): HttpParams {\n    return this.clone({param, value, op: 's'});\n  }\n\n  /**\n   * Removes a given value or all values from a parameter.\n   * @param param The parameter name.\n   * @param value The value to remove, if provided.\n   * @return A new body with the given value removed, or with all values\n   * removed if no value is specified.\n   */\n  delete(param: string, value?: string|number|boolean): HttpParams {\n    return this.clone({param, value, op: 'd'});\n  }\n\n  /**\n   * Serializes the body to an encoded string, where key-value pairs (separated by `=`) are\n   * separated by `&`s.\n   */\n  toString(): string {\n    this.init();\n    return this.keys()\n        .map(key => {\n          const eKey = this.encoder.encodeKey(key);\n          // `a: ['1']` produces `'a=1'`\n          // `b: []` produces `''`\n          // `c: ['1', '2']` produces `'c=1&c=2'`\n          return this.map!.get(key)!.map(value => eKey + '=' + this.encoder.encodeValue(value))\n              .join('&');\n        })\n        // filter out empty values because `b: []` produces `''`\n        // which results in `a=1&&c=1&c=2` instead of `a=1&c=1&c=2` if we don't\n        .filter(param => param !== '')\n        .join('&');\n  }\n\n  private clone(update: Update|Update[]): HttpParams {\n    const clone = new HttpParams({encoder: this.encoder} as HttpParamsOptions);\n    clone.cloneFrom = this.cloneFrom || this;\n    clone.updates = (this.updates || []).concat(update);\n    return clone;\n  }\n\n  private init() {\n    if (this.map === null) {\n      this.map = new Map<string, string[]>();\n    }\n    if (this.cloneFrom !== null) {\n      this.cloneFrom.init();\n      this.cloneFrom.keys().forEach(key => this.map!.set(key, this.cloneFrom!.map!.get(key)!));\n      this.updates!.forEach(update => {\n        switch (update.op) {\n          case 'a':\n          case 's':\n            const base = (update.op === 'a' ? this.map!.get(update.param) : undefined) || [];\n            base.push(valueToString(update.value!));\n            this.map!.set(update.param, base);\n            break;\n          case 'd':\n            if (update.value !== undefined) {\n              let base = this.map!.get(update.param) || [];\n              const idx = base.indexOf(valueToString(update.value));\n              if (idx !== -1) {\n                base.splice(idx, 1);\n              }\n              if (base.length > 0) {\n                this.map!.set(update.param, base);\n              } else {\n                this.map!.delete(update.param);\n              }\n            } else {\n              this.map!.delete(update.param);\n              break;\n            }\n        }\n      });\n      this.cloneFrom = this.updates = null;\n    }\n  }\n}\n"]} |
---|