@tinkoff/user-agent 0.4.176 → 0.4.181

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/lib/index.js CHANGED
@@ -2,349 +2,13 @@
2
2
 
3
3
  Object.defineProperty(exports, '__esModule', { value: true });
4
4
 
5
- var propOr = require('@tinkoff/utils/object/propOr');
6
- var compose = require('@tinkoff/utils/function/compose');
7
- var toLower = require('@tinkoff/utils/string/toLower');
8
- var uaParserJs = require('ua-parser-js');
9
- var isString = require('@tinkoff/utils/is/string');
10
- var browserslist = require('browserslist');
11
- var browserslistTinkoffConfig = require('@tinkoff/browserslist-config');
12
- var browserslistFileConfig = require('@tramvai/cli/lib/external/browserslist-normalized-file-config');
5
+ var userAgent = require('./userAgent.js');
6
+ var satisfies = require('./satisfies.js');
7
+ var clientHints = require('./client-hints.js');
13
8
 
14
- function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
15
9
 
16
- var propOr__default = /*#__PURE__*/_interopDefaultLegacy(propOr);
17
- var compose__default = /*#__PURE__*/_interopDefaultLegacy(compose);
18
- var toLower__default = /*#__PURE__*/_interopDefaultLegacy(toLower);
19
- var isString__default = /*#__PURE__*/_interopDefaultLegacy(isString);
20
- var browserslist__default = /*#__PURE__*/_interopDefaultLegacy(browserslist);
21
- var browserslistTinkoffConfig__default = /*#__PURE__*/_interopDefaultLegacy(browserslistTinkoffConfig);
22
- var browserslistFileConfig__default = /*#__PURE__*/_interopDefaultLegacy(browserslistFileConfig);
23
10
 
24
- const processUserAgentAttributeName = (attributeName = '') => attributeName.toLowerCase();
25
- const splitUserAgentAttributeVersion = (attributeVersion = '-1.-1.-1') => attributeVersion.split('.').map(Number);
26
- // https://www.chromium.org/updates/same-site/incompatible-clients
27
- const isSameSiteNoneCompatible = (userAgent) => {
28
- // На случай неполных данных из ua-parser-js
29
- try {
30
- const browserName = processUserAgentAttributeName(userAgent.browser.name);
31
- const [browserMajor, browserMinor, browserBuild] = splitUserAgentAttributeVersion(userAgent.browser.version);
32
- const osName = processUserAgentAttributeName(userAgent.os.name);
33
- const [osMajor, osMinor] = splitUserAgentAttributeVersion(userAgent.os.version);
34
- const engineName = processUserAgentAttributeName(userAgent.engine.name);
35
- const [engineMajor] = splitUserAgentAttributeVersion(userAgent.engine.version);
36
- if (osName === 'ios') {
37
- // На iOS 12 все браузеры не совместимы с samesite=none
38
- // При этом полагается, что если на iOS!=12 - любые браузеры совместимы с samesite=none, включая UCBrowser<12.13.2 и 51<=Chrome<=66 (WebKit)
39
- return osMajor !== 12;
40
- }
41
- // Встроенный браузер Mac OS парсится как Webkit
42
- if (browserName === 'safari' || browserName === 'webkit') {
43
- return !(osName === 'mac os' && osMajor === 10 && osMinor === 14);
44
- }
45
- if (browserName === 'ucbrowser') {
46
- const [major, minor, build] = [12, 13, 2];
47
- if (browserMajor !== major) {
48
- return browserMajor > major;
49
- }
50
- if (browserMinor !== minor) {
51
- return browserMinor > minor;
52
- }
53
- return browserBuild >= build;
54
- }
55
- if (browserName === 'chrome' || browserName === 'chromium') {
56
- return browserMajor < 51 || browserMajor > 66;
57
- }
58
- if (engineName === 'blink') {
59
- return engineMajor < 51 || engineMajor > 66;
60
- }
61
- return true;
62
- }
63
- catch {
64
- return true;
65
- }
66
- };
67
-
68
- const getMobileOs = (osName) => {
69
- switch (osName) {
70
- case 'Windows Phone':
71
- return 'winphone';
72
- case 'Android':
73
- return 'android';
74
- case 'iOS':
75
- return 'ios';
76
- case 'BlackBerry':
77
- case 'RIM Tablet OS':
78
- return 'blackberry';
79
- }
80
- };
81
- const getBrowserEngine = (browserName, engineName) => {
82
- switch (true) {
83
- case browserName === 'firefox':
84
- return 'firefox';
85
- case browserName === 'safari':
86
- return 'safari';
87
- case engineName === 'webkit' || engineName === 'blink' || engineName === 'chromium':
88
- return 'chrome';
89
- }
90
- return 'other';
91
- };
92
-
93
- const toLowerName = compose__default["default"](toLower__default["default"], propOr__default["default"]('name', ''));
94
- const uaParserExtensions = [
95
- // добавляем отдельные регекспы для ботов гугла и т.п.
96
- // это позволит для них получить отдельное имя браузера и обработать специальным образом
97
- // https://github.com/faisalman/ua-parser-js/issues/227
98
- // google, bing, msn
99
- [/((?:\S+)bot(?:-[imagevdo]{5})?)\/([\w.]+)/i],
100
- [uaParserJs.UAParser.BROWSER.NAME, uaParserJs.UAParser.BROWSER.VERSION, ['type', 'bot']],
101
- // google adsbot под видом обычного браузера
102
- [/[\s;(](adsbot[-\w]*?[\s;)])/i],
103
- [uaParserJs.UAParser.BROWSER.NAME, [uaParserJs.UAParser.BROWSER.VERSION, 'unknown'], ['type', 'bot']],
104
- // добавляем регекспы для браузеров которые пытаются казаться другими браузерами
105
- // например ua-parser-js Firefox Focus для ios считает как просто Firefox, что ломает проверки на версии
106
- // Firefox for iOS
107
- [/fxios\/([\w\\.-]+)/i],
108
- [[uaParserJs.UAParser.BROWSER.NAME, 'Firefox Focus'], uaParserJs.UAParser.BROWSER.VERSION],
109
- ];
110
- const parseUserAgentHeader = (userAgent) => {
111
- const uaParser = new uaParserJs.UAParser('', { browser: uaParserExtensions });
112
- const { ua, ...result } = uaParser.setUA(userAgent).getResult();
113
- const { browser, os, engine } = result;
114
- const browserName = toLowerName(browser);
115
- const engineName = toLowerName(engine);
116
- const sameSiteNoneCompatible = isSameSiteNoneCompatible(result);
117
- const mobileOS = getMobileOs(os.name);
118
- if (browserName === 'opera mobi') {
119
- result.device.type = 'mobile';
120
- }
121
- const browserEngine = getBrowserEngine(browserName, engineName);
122
- return {
123
- ...result,
124
- mobileOS,
125
- sameSiteNoneCompatible,
126
- browser: {
127
- ...browser,
128
- browserEngine,
129
- name: browserName,
130
- },
131
- };
132
- };
133
-
134
- const BROWSERS_LIST_MAP = {
135
- chrome: {
136
- type: 'desktop',
137
- name: 'chrome',
138
- },
139
- safari: {
140
- type: 'desktop',
141
- name: 'safari',
142
- },
143
- firefox: {
144
- type: 'desktop',
145
- name: 'firefox',
146
- },
147
- opera: {
148
- type: 'desktop',
149
- name: 'opera',
150
- },
151
- ie: {
152
- type: 'desktop',
153
- name: 'ie',
154
- },
155
- edge: {
156
- type: 'any',
157
- name: 'edge',
158
- },
159
- and_chr: {
160
- type: 'mobile',
161
- name: 'chrome',
162
- },
163
- ios_saf: {
164
- type: 'mobile',
165
- name: 'mobile safari',
166
- },
167
- android: {
168
- type: 'mobile',
169
- name: 'android browser',
170
- },
171
- op_mob: {
172
- type: 'mobile',
173
- name: 'opera',
174
- },
175
- and_uc: {
176
- type: 'mobile',
177
- name: 'ucbrowser',
178
- },
179
- and_ff: {
180
- type: 'mobile',
181
- name: 'firefox',
182
- },
183
- };
184
- const CHROMIUM_BASED_BROWSERS = [
185
- 'android browser',
186
- 'yandex',
187
- 'vivaldi' /* , 'chrome webview', 'opera', 'samsung' */,
188
- ];
189
-
190
- const deviceTypes = {
191
- mobile: 'mobile',
192
- tablet: 'mobile',
193
- desktop: 'desktop',
194
- };
195
- const normalizedBrowserslist = (query) => {
196
- const resolved = browserslist__default["default"](query, {
197
- // ставим в true, чтобы browserslist возвращал полный список для мобильных браузеров
198
- // т.к. по умолчанию, из-за того что `Can I use` хранит только последнюю версию мобильных браузеров
199
- // то и browserslist возвращал только самую последнюю версию для мобилок, а так он будет маппить
200
- // соответствие к десктопной версии
201
- mobileToDesktop: true,
202
- })
203
- // так как ie убрали из конфига @tinkoff/browserslist-config, то для того чтобы отсечь ie
204
- // добавляем фиктивную версию, чтобы вся функция вернула false для любой версии ie
205
- // причем если обычная версия ie была задана в кастомном конфиге или в конфиге переданном как аргумент
206
- // эта фиктивная версия ни на что не повлияет и будет использована кастомная конфигурация
207
- .concat(['ie 999']);
208
- const result = {};
209
- for (let i = 0; i < resolved.length; i++) {
210
- const [name, version] = resolved[i].split(' ');
211
- if (BROWSERS_LIST_MAP[name]) {
212
- const { type: mapType, name: mapName } = BROWSERS_LIST_MAP[name];
213
- const mapVersion = parseFloat(version);
214
- if (result[mapName]) {
215
- if (!result[mapName][mapType] || result[mapName][mapType] > mapVersion) {
216
- result[mapName][mapType] = mapVersion;
217
- }
218
- }
219
- else {
220
- result[mapName] = {
221
- [mapType]: mapVersion,
222
- };
223
- }
224
- }
225
- }
226
- return result;
227
- };
228
- const satisfies = (userAgent, browserslistConfig, { env = 'defaults' } = {}) => {
229
- var _a, _b;
230
- const ua = isString__default["default"](userAgent) ? parseUserAgentHeader(userAgent) : userAgent;
231
- const { engine: { name: engineName = '', version: engineVersion }, device: { type = '' } = {}, } = ua;
232
- let { browser: { name: browserName = '', version: browserVersion = '' } = {} } = ua;
233
- // Chromium based браузеры (yandex, samsung, opera, vivaldi, edge) указывают версию движка как `Chrome/*`.
234
- // А версия движка (blink) матчится в версию chromium один к одному
235
- // https://github.com/faisalman/ua-parser-js/pull/390
236
- // https://developer.mozilla.org/en-US/docs/Web/HTTP/Browser_detection_using_the_user_agent#Rendering_engine
237
- if (engineName === 'chromium' ||
238
- (engineName.toLowerCase() === 'blink' && CHROMIUM_BASED_BROWSERS.indexOf(browserName) !== -1)) {
239
- browserName = 'chrome';
240
- browserVersion = engineVersion || '';
241
- }
242
- // parseFloat - заберет мажорную + минорную версии, остальное отбросит
243
- const checkVersion = parseFloat(browserVersion);
244
- const deviceType = deviceTypes[type] || type || deviceTypes.desktop; // по умолчанию считаем устройство десктопом
245
- if (!browserName) {
246
- return null;
247
- }
248
- const targets = (_a = browserslistConfig !== null && browserslistConfig !== void 0 ? browserslistConfig : browserslistFileConfig__default["default"][env]) !== null && _a !== void 0 ? _a : browserslistTinkoffConfig__default["default"][env];
249
- const browsers = normalizedBrowserslist(targets);
250
- let hasEntry = false;
251
- if (browserName in browsers) {
252
- const browserInfo = browsers[browserName];
253
- const browserInfoVersion = (_b = browserInfo.any) !== null && _b !== void 0 ? _b : browserInfo[deviceType];
254
- hasEntry = !!browserInfoVersion;
255
- if (checkVersion >= browserInfoVersion) {
256
- return true;
257
- }
258
- }
259
- return hasEntry ? false : null; // null означает что не нашли соответствия браузеру в списке browserslist
260
- };
261
-
262
- const KNOWN_VENDORS = new Set(['Opera', 'Google Chrome', 'Microsoft Edge', 'Firefox', 'Safari']);
263
- const KNOWN_ENGINES = new Set(['Chromium']);
264
- const parseQuotedString = (str) => {
265
- var _a;
266
- if (!str) {
267
- return str;
268
- }
269
- try {
270
- return (_a = JSON.parse(str)) === null || _a === void 0 ? void 0 : _a.trim();
271
- }
272
- catch (err) {
273
- return str;
274
- }
275
- };
276
- const parseBrowser = (brandsList) => {
277
- const browser = {
278
- name: undefined,
279
- version: undefined,
280
- major: undefined,
281
- browserEngine: '',
282
- };
283
- const engine = {
284
- name: undefined,
285
- version: undefined,
286
- };
287
- brandsList.split(',').forEach((entry) => {
288
- const [name, version] = entry.split(/;\s*v=/).map(parseQuotedString);
289
- if (name && KNOWN_VENDORS.has(name)) {
290
- browser.name = name.toLowerCase();
291
- browser.version = version;
292
- browser.major = version;
293
- }
294
- if (name && KNOWN_ENGINES.has(name)) {
295
- engine.name = name.toLowerCase();
296
- engine.version = version;
297
- }
298
- });
299
- if (!browser.name && engine.name) {
300
- browser.name = engine.name;
301
- browser.version = engine.version;
302
- }
303
- browser.browserEngine = getBrowserEngine(browser.name, engine.name);
304
- return { browser, engine };
305
- };
306
- /**
307
- *
308
- * @description
309
- *
310
- * Some of the data are available only when additional headers for client-hints were sent from server:
311
- * - full browser version (only major version is available by default)
312
- * - OS version
313
- * - CPU architecture
314
- * - device model
315
- *
316
- * To able to use data you should first provide header `Accept-CH` with the list of headers that client should send.
317
- *
318
- * @param headers
319
- * @returns
320
- */
321
- const parseClientHintsHeaders = (headers) => {
322
- const { browser, engine } = parseBrowser(headers['sec-ch-ua-full-version-list'] || headers['sec-ch-ua']);
323
- const osName = parseQuotedString(headers['sec-ch-ua-platform']);
324
- const mobileOS = getMobileOs(osName);
325
- return {
326
- browser,
327
- engine,
328
- os: {
329
- name: osName,
330
- version: parseQuotedString(headers['sec-ch-ua-platform-version']),
331
- },
332
- cpu: {
333
- architecture: parseQuotedString(headers['sec-ch-ua-arch']),
334
- },
335
- mobileOS,
336
- device: {
337
- model: parseQuotedString(headers['sec-ch-ua-model']),
338
- type: headers['sec-ch-ua-mobile'] === '?1' ? 'mobile' : 'desktop',
339
- vendor: undefined,
340
- },
341
- // basically all of the browsers with client-hints support
342
- // also compatible with SameSite=None
343
- sameSiteNoneCompatible: true,
344
- };
345
- };
346
-
347
- exports.parse = parseUserAgentHeader;
348
- exports.parseClientHints = parseClientHintsHeaders;
349
- exports.parseUserAgentHeader = parseUserAgentHeader;
350
- exports.satisfies = satisfies;
11
+ exports.parse = userAgent.parseUserAgentHeader;
12
+ exports.parseUserAgentHeader = userAgent.parseUserAgentHeader;
13
+ exports.satisfies = satisfies.satisfies;
14
+ exports.parseClientHints = clientHints.parseClientHintsHeaders;
@@ -0,0 +1,45 @@
1
+ const processUserAgentAttributeName = (attributeName = '') => attributeName.toLowerCase();
2
+ const splitUserAgentAttributeVersion = (attributeVersion = '-1.-1.-1') => attributeVersion.split('.').map(Number);
3
+ // https://www.chromium.org/updates/same-site/incompatible-clients
4
+ const isSameSiteNoneCompatible = (userAgent) => {
5
+ // На случай неполных данных из ua-parser-js
6
+ try {
7
+ const browserName = processUserAgentAttributeName(userAgent.browser.name);
8
+ const [browserMajor, browserMinor, browserBuild] = splitUserAgentAttributeVersion(userAgent.browser.version);
9
+ const osName = processUserAgentAttributeName(userAgent.os.name);
10
+ const [osMajor, osMinor] = splitUserAgentAttributeVersion(userAgent.os.version);
11
+ const engineName = processUserAgentAttributeName(userAgent.engine.name);
12
+ const [engineMajor] = splitUserAgentAttributeVersion(userAgent.engine.version);
13
+ if (osName === 'ios') {
14
+ // На iOS 12 все браузеры не совместимы с samesite=none
15
+ // При этом полагается, что если на iOS!=12 - любые браузеры совместимы с samesite=none, включая UCBrowser<12.13.2 и 51<=Chrome<=66 (WebKit)
16
+ return osMajor !== 12;
17
+ }
18
+ // Встроенный браузер Mac OS парсится как Webkit
19
+ if (browserName === 'safari' || browserName === 'webkit') {
20
+ return !(osName === 'mac os' && osMajor === 10 && osMinor === 14);
21
+ }
22
+ if (browserName === 'ucbrowser') {
23
+ const [major, minor, build] = [12, 13, 2];
24
+ if (browserMajor !== major) {
25
+ return browserMajor > major;
26
+ }
27
+ if (browserMinor !== minor) {
28
+ return browserMinor > minor;
29
+ }
30
+ return browserBuild >= build;
31
+ }
32
+ if (browserName === 'chrome' || browserName === 'chromium') {
33
+ return browserMajor < 51 || browserMajor > 66;
34
+ }
35
+ if (engineName === 'blink') {
36
+ return engineMajor < 51 || engineMajor > 66;
37
+ }
38
+ return true;
39
+ }
40
+ catch {
41
+ return true;
42
+ }
43
+ };
44
+
45
+ export { isSameSiteNoneCompatible };
@@ -0,0 +1,49 @@
1
+ 'use strict';
2
+
3
+ Object.defineProperty(exports, '__esModule', { value: true });
4
+
5
+ const processUserAgentAttributeName = (attributeName = '') => attributeName.toLowerCase();
6
+ const splitUserAgentAttributeVersion = (attributeVersion = '-1.-1.-1') => attributeVersion.split('.').map(Number);
7
+ // https://www.chromium.org/updates/same-site/incompatible-clients
8
+ const isSameSiteNoneCompatible = (userAgent) => {
9
+ // На случай неполных данных из ua-parser-js
10
+ try {
11
+ const browserName = processUserAgentAttributeName(userAgent.browser.name);
12
+ const [browserMajor, browserMinor, browserBuild] = splitUserAgentAttributeVersion(userAgent.browser.version);
13
+ const osName = processUserAgentAttributeName(userAgent.os.name);
14
+ const [osMajor, osMinor] = splitUserAgentAttributeVersion(userAgent.os.version);
15
+ const engineName = processUserAgentAttributeName(userAgent.engine.name);
16
+ const [engineMajor] = splitUserAgentAttributeVersion(userAgent.engine.version);
17
+ if (osName === 'ios') {
18
+ // На iOS 12 все браузеры не совместимы с samesite=none
19
+ // При этом полагается, что если на iOS!=12 - любые браузеры совместимы с samesite=none, включая UCBrowser<12.13.2 и 51<=Chrome<=66 (WebKit)
20
+ return osMajor !== 12;
21
+ }
22
+ // Встроенный браузер Mac OS парсится как Webkit
23
+ if (browserName === 'safari' || browserName === 'webkit') {
24
+ return !(osName === 'mac os' && osMajor === 10 && osMinor === 14);
25
+ }
26
+ if (browserName === 'ucbrowser') {
27
+ const [major, minor, build] = [12, 13, 2];
28
+ if (browserMajor !== major) {
29
+ return browserMajor > major;
30
+ }
31
+ if (browserMinor !== minor) {
32
+ return browserMinor > minor;
33
+ }
34
+ return browserBuild >= build;
35
+ }
36
+ if (browserName === 'chrome' || browserName === 'chromium') {
37
+ return browserMajor < 51 || browserMajor > 66;
38
+ }
39
+ if (engineName === 'blink') {
40
+ return engineMajor < 51 || engineMajor > 66;
41
+ }
42
+ return true;
43
+ }
44
+ catch {
45
+ return true;
46
+ }
47
+ };
48
+
49
+ exports.isSameSiteNoneCompatible = isSameSiteNoneCompatible;
@@ -0,0 +1,80 @@
1
+ import isString from '@tinkoff/utils/is/string';
2
+ import browserslist from 'browserslist';
3
+ import browserslistTinkoffConfig from '@tinkoff/browserslist-config';
4
+ import browserslistFileConfig from '@tramvai/cli/lib/external/browserslist-normalized-file-config';
5
+ import { parseUserAgentHeader } from './userAgent.es.js';
6
+ import { CHROMIUM_BASED_BROWSERS, BROWSERS_LIST_MAP } from './constants.es.js';
7
+
8
+ const deviceTypes = {
9
+ mobile: 'mobile',
10
+ tablet: 'mobile',
11
+ desktop: 'desktop',
12
+ };
13
+ const normalizedBrowserslist = (query) => {
14
+ const resolved = browserslist(query, {
15
+ // ставим в true, чтобы browserslist возвращал полный список для мобильных браузеров
16
+ // т.к. по умолчанию, из-за того что `Can I use` хранит только последнюю версию мобильных браузеров
17
+ // то и browserslist возвращал только самую последнюю версию для мобилок, а так он будет маппить
18
+ // соответствие к десктопной версии
19
+ mobileToDesktop: true,
20
+ })
21
+ // так как ie убрали из конфига @tinkoff/browserslist-config, то для того чтобы отсечь ie
22
+ // добавляем фиктивную версию, чтобы вся функция вернула false для любой версии ie
23
+ // причем если обычная версия ie была задана в кастомном конфиге или в конфиге переданном как аргумент
24
+ // эта фиктивная версия ни на что не повлияет и будет использована кастомная конфигурация
25
+ .concat(['ie 999']);
26
+ const result = {};
27
+ for (let i = 0; i < resolved.length; i++) {
28
+ const [name, version] = resolved[i].split(' ');
29
+ if (BROWSERS_LIST_MAP[name]) {
30
+ const { type: mapType, name: mapName } = BROWSERS_LIST_MAP[name];
31
+ const mapVersion = parseFloat(version);
32
+ if (result[mapName]) {
33
+ if (!result[mapName][mapType] || result[mapName][mapType] > mapVersion) {
34
+ result[mapName][mapType] = mapVersion;
35
+ }
36
+ }
37
+ else {
38
+ result[mapName] = {
39
+ [mapType]: mapVersion,
40
+ };
41
+ }
42
+ }
43
+ }
44
+ return result;
45
+ };
46
+ const satisfies = (userAgent, browserslistConfig, { env = 'defaults' } = {}) => {
47
+ var _a, _b;
48
+ const ua = isString(userAgent) ? parseUserAgentHeader(userAgent) : userAgent;
49
+ const { engine: { name: engineName = '', version: engineVersion }, device: { type = '' } = {}, } = ua;
50
+ let { browser: { name: browserName = '', version: browserVersion = '' } = {} } = ua;
51
+ // Chromium based браузеры (yandex, samsung, opera, vivaldi, edge) указывают версию движка как `Chrome/*`.
52
+ // А версия движка (blink) матчится в версию chromium один к одному
53
+ // https://github.com/faisalman/ua-parser-js/pull/390
54
+ // https://developer.mozilla.org/en-US/docs/Web/HTTP/Browser_detection_using_the_user_agent#Rendering_engine
55
+ if (engineName === 'chromium' ||
56
+ (engineName.toLowerCase() === 'blink' && CHROMIUM_BASED_BROWSERS.indexOf(browserName) !== -1)) {
57
+ browserName = 'chrome';
58
+ browserVersion = engineVersion || '';
59
+ }
60
+ // parseFloat - заберет мажорную + минорную версии, остальное отбросит
61
+ const checkVersion = parseFloat(browserVersion);
62
+ const deviceType = deviceTypes[type] || type || deviceTypes.desktop; // по умолчанию считаем устройство десктопом
63
+ if (!browserName) {
64
+ return null;
65
+ }
66
+ const targets = (_a = browserslistConfig !== null && browserslistConfig !== void 0 ? browserslistConfig : browserslistFileConfig[env]) !== null && _a !== void 0 ? _a : browserslistTinkoffConfig[env];
67
+ const browsers = normalizedBrowserslist(targets);
68
+ let hasEntry = false;
69
+ if (browserName in browsers) {
70
+ const browserInfo = browsers[browserName];
71
+ const browserInfoVersion = (_b = browserInfo.any) !== null && _b !== void 0 ? _b : browserInfo[deviceType];
72
+ hasEntry = !!browserInfoVersion;
73
+ if (checkVersion >= browserInfoVersion) {
74
+ return true;
75
+ }
76
+ }
77
+ return hasEntry ? false : null; // null означает что не нашли соответствия браузеру в списке browserslist
78
+ };
79
+
80
+ export { satisfies };
@@ -0,0 +1,91 @@
1
+ 'use strict';
2
+
3
+ Object.defineProperty(exports, '__esModule', { value: true });
4
+
5
+ var isString = require('@tinkoff/utils/is/string');
6
+ var browserslist = require('browserslist');
7
+ var browserslistTinkoffConfig = require('@tinkoff/browserslist-config');
8
+ var browserslistFileConfig = require('@tramvai/cli/lib/external/browserslist-normalized-file-config');
9
+ var userAgent = require('./userAgent.js');
10
+ var constants = require('./constants.js');
11
+
12
+ function _interopDefaultLegacy (e) { return e && typeof e === 'object' && 'default' in e ? e : { 'default': e }; }
13
+
14
+ var isString__default = /*#__PURE__*/_interopDefaultLegacy(isString);
15
+ var browserslist__default = /*#__PURE__*/_interopDefaultLegacy(browserslist);
16
+ var browserslistTinkoffConfig__default = /*#__PURE__*/_interopDefaultLegacy(browserslistTinkoffConfig);
17
+ var browserslistFileConfig__default = /*#__PURE__*/_interopDefaultLegacy(browserslistFileConfig);
18
+
19
+ const deviceTypes = {
20
+ mobile: 'mobile',
21
+ tablet: 'mobile',
22
+ desktop: 'desktop',
23
+ };
24
+ const normalizedBrowserslist = (query) => {
25
+ const resolved = browserslist__default["default"](query, {
26
+ // ставим в true, чтобы browserslist возвращал полный список для мобильных браузеров
27
+ // т.к. по умолчанию, из-за того что `Can I use` хранит только последнюю версию мобильных браузеров
28
+ // то и browserslist возвращал только самую последнюю версию для мобилок, а так он будет маппить
29
+ // соответствие к десктопной версии
30
+ mobileToDesktop: true,
31
+ })
32
+ // так как ie убрали из конфига @tinkoff/browserslist-config, то для того чтобы отсечь ie
33
+ // добавляем фиктивную версию, чтобы вся функция вернула false для любой версии ie
34
+ // причем если обычная версия ie была задана в кастомном конфиге или в конфиге переданном как аргумент
35
+ // эта фиктивная версия ни на что не повлияет и будет использована кастомная конфигурация
36
+ .concat(['ie 999']);
37
+ const result = {};
38
+ for (let i = 0; i < resolved.length; i++) {
39
+ const [name, version] = resolved[i].split(' ');
40
+ if (constants.BROWSERS_LIST_MAP[name]) {
41
+ const { type: mapType, name: mapName } = constants.BROWSERS_LIST_MAP[name];
42
+ const mapVersion = parseFloat(version);
43
+ if (result[mapName]) {
44
+ if (!result[mapName][mapType] || result[mapName][mapType] > mapVersion) {
45
+ result[mapName][mapType] = mapVersion;
46
+ }
47
+ }
48
+ else {
49
+ result[mapName] = {
50
+ [mapType]: mapVersion,
51
+ };
52
+ }
53
+ }
54
+ }
55
+ return result;
56
+ };
57
+ const satisfies = (userAgent$1, browserslistConfig, { env = 'defaults' } = {}) => {
58
+ var _a, _b;
59
+ const ua = isString__default["default"](userAgent$1) ? userAgent.parseUserAgentHeader(userAgent$1) : userAgent$1;
60
+ const { engine: { name: engineName = '', version: engineVersion }, device: { type = '' } = {}, } = ua;
61
+ let { browser: { name: browserName = '', version: browserVersion = '' } = {} } = ua;
62
+ // Chromium based браузеры (yandex, samsung, opera, vivaldi, edge) указывают версию движка как `Chrome/*`.
63
+ // А версия движка (blink) матчится в версию chromium один к одному
64
+ // https://github.com/faisalman/ua-parser-js/pull/390
65
+ // https://developer.mozilla.org/en-US/docs/Web/HTTP/Browser_detection_using_the_user_agent#Rendering_engine
66
+ if (engineName === 'chromium' ||
67
+ (engineName.toLowerCase() === 'blink' && constants.CHROMIUM_BASED_BROWSERS.indexOf(browserName) !== -1)) {
68
+ browserName = 'chrome';
69
+ browserVersion = engineVersion || '';
70
+ }
71
+ // parseFloat - заберет мажорную + минорную версии, остальное отбросит
72
+ const checkVersion = parseFloat(browserVersion);
73
+ const deviceType = deviceTypes[type] || type || deviceTypes.desktop; // по умолчанию считаем устройство десктопом
74
+ if (!browserName) {
75
+ return null;
76
+ }
77
+ const targets = (_a = browserslistConfig !== null && browserslistConfig !== void 0 ? browserslistConfig : browserslistFileConfig__default["default"][env]) !== null && _a !== void 0 ? _a : browserslistTinkoffConfig__default["default"][env];
78
+ const browsers = normalizedBrowserslist(targets);
79
+ let hasEntry = false;
80
+ if (browserName in browsers) {
81
+ const browserInfo = browsers[browserName];
82
+ const browserInfoVersion = (_b = browserInfo.any) !== null && _b !== void 0 ? _b : browserInfo[deviceType];
83
+ hasEntry = !!browserInfoVersion;
84
+ if (checkVersion >= browserInfoVersion) {
85
+ return true;
86
+ }
87
+ }
88
+ return hasEntry ? false : null; // null означает что не нашли соответствия браузеру в списке browserslist
89
+ };
90
+
91
+ exports.satisfies = satisfies;