| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378 |
- 'use strict';
- import utils from '../utils.js';
- import parseHeaders from '../helpers/parseHeaders.js';
- const $internals = Symbol('internals');
- const isValidHeaderValue = (value) => !/[\r\n]/.test(value);
- function assertValidHeaderValue(value, header) {
- if (value === false || value == null) {
- return;
- }
- if (utils.isArray(value)) {
- value.forEach((v) => assertValidHeaderValue(v, header));
- return;
- }
- if (!isValidHeaderValue(String(value))) {
- throw new Error(`Invalid character in header content ["${header}"]`);
- }
- }
- function normalizeHeader(header) {
- return header && String(header).trim().toLowerCase();
- }
- function stripTrailingCRLF(str) {
- let end = str.length;
- while (end > 0) {
- const charCode = str.charCodeAt(end - 1);
- if (charCode !== 10 && charCode !== 13) {
- break;
- }
- end -= 1;
- }
- return end === str.length ? str : str.slice(0, end);
- }
- function normalizeValue(value) {
- if (value === false || value == null) {
- return value;
- }
- return utils.isArray(value) ? value.map(normalizeValue) : stripTrailingCRLF(String(value));
- }
- function parseTokens(str) {
- const tokens = Object.create(null);
- const tokensRE = /([^\s,;=]+)\s*(?:=\s*([^,;]+))?/g;
- let match;
- while ((match = tokensRE.exec(str))) {
- tokens[match[1]] = match[2];
- }
- return tokens;
- }
- const isValidHeaderName = (str) => /^[-_a-zA-Z0-9^`|~,!#$%&'*+.]+$/.test(str.trim());
- function matchHeaderValue(context, value, header, filter, isHeaderNameFilter) {
- if (utils.isFunction(filter)) {
- return filter.call(this, value, header);
- }
- if (isHeaderNameFilter) {
- value = header;
- }
- if (!utils.isString(value)) return;
- if (utils.isString(filter)) {
- return value.indexOf(filter) !== -1;
- }
- if (utils.isRegExp(filter)) {
- return filter.test(value);
- }
- }
- function formatHeader(header) {
- return header
- .trim()
- .toLowerCase()
- .replace(/([a-z\d])(\w*)/g, (w, char, str) => {
- return char.toUpperCase() + str;
- });
- }
- function buildAccessors(obj, header) {
- const accessorName = utils.toCamelCase(' ' + header);
- ['get', 'set', 'has'].forEach((methodName) => {
- Object.defineProperty(obj, methodName + accessorName, {
- value: function (arg1, arg2, arg3) {
- return this[methodName].call(this, header, arg1, arg2, arg3);
- },
- configurable: true,
- });
- });
- }
- class AxiosHeaders {
- constructor(headers) {
- headers && this.set(headers);
- }
- set(header, valueOrRewrite, rewrite) {
- const self = this;
- function setHeader(_value, _header, _rewrite) {
- const lHeader = normalizeHeader(_header);
- if (!lHeader) {
- throw new Error('header name must be a non-empty string');
- }
- const key = utils.findKey(self, lHeader);
- if (
- !key ||
- self[key] === undefined ||
- _rewrite === true ||
- (_rewrite === undefined && self[key] !== false)
- ) {
- assertValidHeaderValue(_value, _header);
- self[key || _header] = normalizeValue(_value);
- }
- }
- const setHeaders = (headers, _rewrite) =>
- utils.forEach(headers, (_value, _header) => setHeader(_value, _header, _rewrite));
- if (utils.isPlainObject(header) || header instanceof this.constructor) {
- setHeaders(header, valueOrRewrite);
- } else if (utils.isString(header) && (header = header.trim()) && !isValidHeaderName(header)) {
- setHeaders(parseHeaders(header), valueOrRewrite);
- } else if (utils.isObject(header) && utils.isIterable(header)) {
- let obj = {},
- dest,
- key;
- for (const entry of header) {
- if (!utils.isArray(entry)) {
- throw TypeError('Object iterator must return a key-value pair');
- }
- obj[(key = entry[0])] = (dest = obj[key])
- ? utils.isArray(dest)
- ? [...dest, entry[1]]
- : [dest, entry[1]]
- : entry[1];
- }
- setHeaders(obj, valueOrRewrite);
- } else {
- header != null && setHeader(valueOrRewrite, header, rewrite);
- }
- return this;
- }
- get(header, parser) {
- header = normalizeHeader(header);
- if (header) {
- const key = utils.findKey(this, header);
- if (key) {
- const value = this[key];
- if (!parser) {
- return value;
- }
- if (parser === true) {
- return parseTokens(value);
- }
- if (utils.isFunction(parser)) {
- return parser.call(this, value, key);
- }
- if (utils.isRegExp(parser)) {
- return parser.exec(value);
- }
- throw new TypeError('parser must be boolean|regexp|function');
- }
- }
- }
- has(header, matcher) {
- header = normalizeHeader(header);
- if (header) {
- const key = utils.findKey(this, header);
- return !!(
- key &&
- this[key] !== undefined &&
- (!matcher || matchHeaderValue(this, this[key], key, matcher))
- );
- }
- return false;
- }
- delete(header, matcher) {
- const self = this;
- let deleted = false;
- function deleteHeader(_header) {
- _header = normalizeHeader(_header);
- if (_header) {
- const key = utils.findKey(self, _header);
- if (key && (!matcher || matchHeaderValue(self, self[key], key, matcher))) {
- delete self[key];
- deleted = true;
- }
- }
- }
- if (utils.isArray(header)) {
- header.forEach(deleteHeader);
- } else {
- deleteHeader(header);
- }
- return deleted;
- }
- clear(matcher) {
- const keys = Object.keys(this);
- let i = keys.length;
- let deleted = false;
- while (i--) {
- const key = keys[i];
- if (!matcher || matchHeaderValue(this, this[key], key, matcher, true)) {
- delete this[key];
- deleted = true;
- }
- }
- return deleted;
- }
- normalize(format) {
- const self = this;
- const headers = {};
- utils.forEach(this, (value, header) => {
- const key = utils.findKey(headers, header);
- if (key) {
- self[key] = normalizeValue(value);
- delete self[header];
- return;
- }
- const normalized = format ? formatHeader(header) : String(header).trim();
- if (normalized !== header) {
- delete self[header];
- }
- self[normalized] = normalizeValue(value);
- headers[normalized] = true;
- });
- return this;
- }
- concat(...targets) {
- return this.constructor.concat(this, ...targets);
- }
- toJSON(asStrings) {
- const obj = Object.create(null);
- utils.forEach(this, (value, header) => {
- value != null &&
- value !== false &&
- (obj[header] = asStrings && utils.isArray(value) ? value.join(', ') : value);
- });
- return obj;
- }
- [Symbol.iterator]() {
- return Object.entries(this.toJSON())[Symbol.iterator]();
- }
- toString() {
- return Object.entries(this.toJSON())
- .map(([header, value]) => header + ': ' + value)
- .join('\n');
- }
- getSetCookie() {
- return this.get('set-cookie') || [];
- }
- get [Symbol.toStringTag]() {
- return 'AxiosHeaders';
- }
- static from(thing) {
- return thing instanceof this ? thing : new this(thing);
- }
- static concat(first, ...targets) {
- const computed = new this(first);
- targets.forEach((target) => computed.set(target));
- return computed;
- }
- static accessor(header) {
- const internals =
- (this[$internals] =
- this[$internals] =
- {
- accessors: {},
- });
- const accessors = internals.accessors;
- const prototype = this.prototype;
- function defineAccessor(_header) {
- const lHeader = normalizeHeader(_header);
- if (!accessors[lHeader]) {
- buildAccessors(prototype, _header);
- accessors[lHeader] = true;
- }
- }
- utils.isArray(header) ? header.forEach(defineAccessor) : defineAccessor(header);
- return this;
- }
- }
- AxiosHeaders.accessor([
- 'Content-Type',
- 'Content-Length',
- 'Accept',
- 'Accept-Encoding',
- 'User-Agent',
- 'Authorization',
- ]);
- // reserved names hotfix
- utils.reduceDescriptors(AxiosHeaders.prototype, ({ value }, key) => {
- let mapped = key[0].toUpperCase() + key.slice(1); // map `set` => `Set`
- return {
- get: () => value,
- set(headerValue) {
- this[mapped] = headerValue;
- },
- };
- });
- utils.freezeMethods(AxiosHeaders);
- export default AxiosHeaders;
|