AxiosHeaders.js 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298
  1. 'use strict';
  2. import utils from '../utils.js';
  3. import parseHeaders from '../helpers/parseHeaders.js';
  4. const $internals = Symbol('internals');
  5. function normalizeHeader(header) {
  6. return header && String(header).trim().toLowerCase();
  7. }
  8. function normalizeValue(value) {
  9. if (value === false || value == null) {
  10. return value;
  11. }
  12. return utils.isArray(value) ? value.map(normalizeValue) : String(value);
  13. }
  14. function parseTokens(str) {
  15. const tokens = Object.create(null);
  16. const tokensRE = /([^\s,;=]+)\s*(?:=\s*([^,;]+))?/g;
  17. let match;
  18. while ((match = tokensRE.exec(str))) {
  19. tokens[match[1]] = match[2];
  20. }
  21. return tokens;
  22. }
  23. const isValidHeaderName = (str) => /^[-_a-zA-Z0-9^`|~,!#$%&'*+.]+$/.test(str.trim());
  24. function matchHeaderValue(context, value, header, filter, isHeaderNameFilter) {
  25. if (utils.isFunction(filter)) {
  26. return filter.call(this, value, header);
  27. }
  28. if (isHeaderNameFilter) {
  29. value = header;
  30. }
  31. if (!utils.isString(value)) return;
  32. if (utils.isString(filter)) {
  33. return value.indexOf(filter) !== -1;
  34. }
  35. if (utils.isRegExp(filter)) {
  36. return filter.test(value);
  37. }
  38. }
  39. function formatHeader(header) {
  40. return header.trim()
  41. .toLowerCase().replace(/([a-z\d])(\w*)/g, (w, char, str) => {
  42. return char.toUpperCase() + str;
  43. });
  44. }
  45. function buildAccessors(obj, header) {
  46. const accessorName = utils.toCamelCase(' ' + header);
  47. ['get', 'set', 'has'].forEach(methodName => {
  48. Object.defineProperty(obj, methodName + accessorName, {
  49. value: function(arg1, arg2, arg3) {
  50. return this[methodName].call(this, header, arg1, arg2, arg3);
  51. },
  52. configurable: true
  53. });
  54. });
  55. }
  56. class AxiosHeaders {
  57. constructor(headers) {
  58. headers && this.set(headers);
  59. }
  60. set(header, valueOrRewrite, rewrite) {
  61. const self = this;
  62. function setHeader(_value, _header, _rewrite) {
  63. const lHeader = normalizeHeader(_header);
  64. if (!lHeader) {
  65. throw new Error('header name must be a non-empty string');
  66. }
  67. const key = utils.findKey(self, lHeader);
  68. if(!key || self[key] === undefined || _rewrite === true || (_rewrite === undefined && self[key] !== false)) {
  69. self[key || _header] = normalizeValue(_value);
  70. }
  71. }
  72. const setHeaders = (headers, _rewrite) =>
  73. utils.forEach(headers, (_value, _header) => setHeader(_value, _header, _rewrite));
  74. if (utils.isPlainObject(header) || header instanceof this.constructor) {
  75. setHeaders(header, valueOrRewrite)
  76. } else if(utils.isString(header) && (header = header.trim()) && !isValidHeaderName(header)) {
  77. setHeaders(parseHeaders(header), valueOrRewrite);
  78. } else {
  79. header != null && setHeader(valueOrRewrite, header, rewrite);
  80. }
  81. return this;
  82. }
  83. get(header, parser) {
  84. header = normalizeHeader(header);
  85. if (header) {
  86. const key = utils.findKey(this, header);
  87. if (key) {
  88. const value = this[key];
  89. if (!parser) {
  90. return value;
  91. }
  92. if (parser === true) {
  93. return parseTokens(value);
  94. }
  95. if (utils.isFunction(parser)) {
  96. return parser.call(this, value, key);
  97. }
  98. if (utils.isRegExp(parser)) {
  99. return parser.exec(value);
  100. }
  101. throw new TypeError('parser must be boolean|regexp|function');
  102. }
  103. }
  104. }
  105. has(header, matcher) {
  106. header = normalizeHeader(header);
  107. if (header) {
  108. const key = utils.findKey(this, header);
  109. return !!(key && this[key] !== undefined && (!matcher || matchHeaderValue(this, this[key], key, matcher)));
  110. }
  111. return false;
  112. }
  113. delete(header, matcher) {
  114. const self = this;
  115. let deleted = false;
  116. function deleteHeader(_header) {
  117. _header = normalizeHeader(_header);
  118. if (_header) {
  119. const key = utils.findKey(self, _header);
  120. if (key && (!matcher || matchHeaderValue(self, self[key], key, matcher))) {
  121. delete self[key];
  122. deleted = true;
  123. }
  124. }
  125. }
  126. if (utils.isArray(header)) {
  127. header.forEach(deleteHeader);
  128. } else {
  129. deleteHeader(header);
  130. }
  131. return deleted;
  132. }
  133. clear(matcher) {
  134. const keys = Object.keys(this);
  135. let i = keys.length;
  136. let deleted = false;
  137. while (i--) {
  138. const key = keys[i];
  139. if(!matcher || matchHeaderValue(this, this[key], key, matcher, true)) {
  140. delete this[key];
  141. deleted = true;
  142. }
  143. }
  144. return deleted;
  145. }
  146. normalize(format) {
  147. const self = this;
  148. const headers = {};
  149. utils.forEach(this, (value, header) => {
  150. const key = utils.findKey(headers, header);
  151. if (key) {
  152. self[key] = normalizeValue(value);
  153. delete self[header];
  154. return;
  155. }
  156. const normalized = format ? formatHeader(header) : String(header).trim();
  157. if (normalized !== header) {
  158. delete self[header];
  159. }
  160. self[normalized] = normalizeValue(value);
  161. headers[normalized] = true;
  162. });
  163. return this;
  164. }
  165. concat(...targets) {
  166. return this.constructor.concat(this, ...targets);
  167. }
  168. toJSON(asStrings) {
  169. const obj = Object.create(null);
  170. utils.forEach(this, (value, header) => {
  171. value != null && value !== false && (obj[header] = asStrings && utils.isArray(value) ? value.join(', ') : value);
  172. });
  173. return obj;
  174. }
  175. [Symbol.iterator]() {
  176. return Object.entries(this.toJSON())[Symbol.iterator]();
  177. }
  178. toString() {
  179. return Object.entries(this.toJSON()).map(([header, value]) => header + ': ' + value).join('\n');
  180. }
  181. get [Symbol.toStringTag]() {
  182. return 'AxiosHeaders';
  183. }
  184. static from(thing) {
  185. return thing instanceof this ? thing : new this(thing);
  186. }
  187. static concat(first, ...targets) {
  188. const computed = new this(first);
  189. targets.forEach((target) => computed.set(target));
  190. return computed;
  191. }
  192. static accessor(header) {
  193. const internals = this[$internals] = (this[$internals] = {
  194. accessors: {}
  195. });
  196. const accessors = internals.accessors;
  197. const prototype = this.prototype;
  198. function defineAccessor(_header) {
  199. const lHeader = normalizeHeader(_header);
  200. if (!accessors[lHeader]) {
  201. buildAccessors(prototype, _header);
  202. accessors[lHeader] = true;
  203. }
  204. }
  205. utils.isArray(header) ? header.forEach(defineAccessor) : defineAccessor(header);
  206. return this;
  207. }
  208. }
  209. AxiosHeaders.accessor(['Content-Type', 'Content-Length', 'Accept', 'Accept-Encoding', 'User-Agent', 'Authorization']);
  210. // reserved names hotfix
  211. utils.reduceDescriptors(AxiosHeaders.prototype, ({value}, key) => {
  212. let mapped = key[0].toUpperCase() + key.slice(1); // map `set` => `Set`
  213. return {
  214. get: () => value,
  215. set(headerValue) {
  216. this[mapped] = headerValue;
  217. }
  218. }
  219. });
  220. utils.freezeMethods(AxiosHeaders);
  221. export default AxiosHeaders;