mirror of
https://github.com/iconify/iconify.git
synced 2024-12-12 13:47:49 +00:00
chore(core): refactor loader to check icon names only in API calls
This commit is contained in:
parent
7f27d10da6
commit
51c5b29fcd
@ -1,5 +1,9 @@
|
||||
import type { IconifyIcon, IconifyJSON } from '@iconify/types';
|
||||
import { IconifyIconName, stringToIcon } from '@iconify/utils/lib/icon/name';
|
||||
import {
|
||||
IconifyIconName,
|
||||
matchIconName,
|
||||
stringToIcon,
|
||||
} from '@iconify/utils/lib/icon/name';
|
||||
import type { SortedIcons } from '../icon/sort';
|
||||
import { sortIcons } from '../icon/sort';
|
||||
import { storeCallback, updateCallbacks } from './callbacks';
|
||||
@ -61,53 +65,49 @@ function loadedNewIcons(storage: IconStorageWithAPI): void {
|
||||
}
|
||||
}
|
||||
|
||||
interface CheckIconNames {
|
||||
valid: string[];
|
||||
invalid: string[];
|
||||
}
|
||||
/**
|
||||
* Load icons
|
||||
* Check icon names for API
|
||||
*/
|
||||
function loadNewIcons(storage: IconStorageWithAPI, icons: string[]): void {
|
||||
// Add icons to queue
|
||||
if (!storage.iconsToLoad) {
|
||||
storage.iconsToLoad = icons;
|
||||
} else {
|
||||
storage.iconsToLoad = storage.iconsToLoad.concat(icons).sort();
|
||||
function checkIconNamesForAPI(icons: string[]): CheckIconNames {
|
||||
const valid: string[] = [];
|
||||
const invalid: string[] = [];
|
||||
|
||||
icons.forEach((name) => {
|
||||
(name.match(matchIconName) ? valid : invalid).push(name);
|
||||
});
|
||||
return {
|
||||
valid,
|
||||
invalid,
|
||||
};
|
||||
}
|
||||
|
||||
// Trigger update on next tick, mering multiple synchronous requests into one asynchronous request
|
||||
if (!storage.iconsQueueFlag) {
|
||||
storage.iconsQueueFlag = true;
|
||||
setTimeout(() => {
|
||||
storage.iconsQueueFlag = false;
|
||||
const { provider, prefix } = storage;
|
||||
|
||||
// Get icons and delete queue
|
||||
const icons = storage.iconsToLoad;
|
||||
delete storage.iconsToLoad;
|
||||
|
||||
// Get API module
|
||||
let api: ReturnType<typeof getAPIModule>;
|
||||
if (!icons || !(api = getAPIModule(provider))) {
|
||||
// No icons or no way to load icons!
|
||||
return;
|
||||
}
|
||||
|
||||
// Prepare parameters and run queries
|
||||
const params = api.prepare(provider, prefix, icons);
|
||||
params.forEach((item) => {
|
||||
sendAPIQuery(provider, item, (data) => {
|
||||
// Check for error
|
||||
if (typeof data !== 'object') {
|
||||
// Not found: mark as missing
|
||||
item.icons.forEach((name) => {
|
||||
/**
|
||||
* Parse loader response
|
||||
*/
|
||||
function parseLoaderResponse(
|
||||
storage: IconStorageWithAPI,
|
||||
icons: string[],
|
||||
data: unknown
|
||||
) {
|
||||
const fail = () => {
|
||||
icons.forEach((name) => {
|
||||
storage.missing.add(name);
|
||||
});
|
||||
};
|
||||
|
||||
// Check for error
|
||||
if (typeof data !== 'object' || !data) {
|
||||
fail();
|
||||
} else {
|
||||
// Add icons to storage
|
||||
try {
|
||||
const parsed = addIconSet(
|
||||
storage,
|
||||
data as IconifyJSON
|
||||
);
|
||||
const parsed = addIconSet(storage, data as IconifyJSON);
|
||||
if (!parsed.length) {
|
||||
fail();
|
||||
return;
|
||||
}
|
||||
|
||||
@ -128,6 +128,61 @@ function loadNewIcons(storage: IconStorageWithAPI, icons: string[]): void {
|
||||
|
||||
// Trigger update on next tick
|
||||
loadedNewIcons(storage);
|
||||
}
|
||||
|
||||
/**
|
||||
* Load icons
|
||||
*/
|
||||
function loadNewIcons(storage: IconStorageWithAPI, icons: string[]): void {
|
||||
// Add icons to queue
|
||||
if (!storage.iconsToLoad) {
|
||||
storage.iconsToLoad = icons;
|
||||
} else {
|
||||
storage.iconsToLoad = storage.iconsToLoad.concat(icons).sort();
|
||||
}
|
||||
|
||||
// Trigger update on next tick, mering multiple synchronous requests into one asynchronous request
|
||||
if (!storage.iconsQueueFlag) {
|
||||
storage.iconsQueueFlag = true;
|
||||
setTimeout(() => {
|
||||
storage.iconsQueueFlag = false;
|
||||
const { provider, prefix } = storage;
|
||||
|
||||
// Get icons and delete queue
|
||||
// Icons should not be undefined, but just in case assume it can be
|
||||
const icons = storage.iconsToLoad;
|
||||
delete storage.iconsToLoad;
|
||||
|
||||
// TODO: check for custom loader
|
||||
|
||||
// Using API loader
|
||||
// Validate icon names for API
|
||||
const { valid, invalid } = checkIconNamesForAPI(icons || []);
|
||||
|
||||
if (invalid.length) {
|
||||
// Invalid icons
|
||||
parseLoaderResponse(storage, invalid, null);
|
||||
}
|
||||
if (!valid.length) {
|
||||
// No valid icons to load
|
||||
return;
|
||||
}
|
||||
|
||||
// Get API module
|
||||
const api = prefix.match(matchIconName)
|
||||
? getAPIModule(provider)
|
||||
: null;
|
||||
if (!api) {
|
||||
// API module not found
|
||||
parseLoaderResponse(storage, valid, null);
|
||||
return;
|
||||
}
|
||||
|
||||
// Prepare parameters and run queries
|
||||
const params = api.prepare(provider, prefix, valid);
|
||||
params.forEach((item) => {
|
||||
sendAPIQuery(provider, item, (data) => {
|
||||
parseLoaderResponse(storage, item.icons, data);
|
||||
});
|
||||
});
|
||||
});
|
||||
@ -232,9 +287,9 @@ export const loadIcons: IconifyLoadIcons = (
|
||||
// Load icons on next tick to make sure result is not returned before callback is stored and
|
||||
// to consolidate multiple synchronous loadIcons() calls into one asynchronous API call
|
||||
sources.forEach((storage) => {
|
||||
const { provider, prefix } = storage;
|
||||
if (newIcons[provider][prefix].length) {
|
||||
loadNewIcons(storage, newIcons[provider][prefix]);
|
||||
const list = newIcons[storage.provider][storage.prefix];
|
||||
if (list.length) {
|
||||
loadNewIcons(storage, list);
|
||||
}
|
||||
});
|
||||
|
||||
|
@ -21,6 +21,7 @@ const detectFetch = (): FetchType | undefined => {
|
||||
if (typeof callback === 'function') {
|
||||
return callback;
|
||||
}
|
||||
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||
} catch (err) {
|
||||
//
|
||||
}
|
||||
|
@ -311,9 +311,9 @@ describe('Testing API loadIcons', () => {
|
||||
expect(loadedIcon).toBe(false);
|
||||
|
||||
// Test isPending
|
||||
expect(isPending({ provider, prefix, name: 'BadIconName' })).toBe(
|
||||
false
|
||||
);
|
||||
// After change to naming convention, icon name is valid and should be pending
|
||||
// Filtering invalid names is done in loader, not in API module
|
||||
expect(isPending({ provider, prefix, name: 'BadIconName' })).toBe(true);
|
||||
});
|
||||
|
||||
it('Loading one icon twice with Promise', () => {
|
||||
|
@ -29,9 +29,11 @@ describe('Testing mock API module', () => {
|
||||
|
||||
// Tests
|
||||
it('404 response', () => {
|
||||
return new Promise((fulfill) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
const prefix = nextPrefix();
|
||||
let isSync = true;
|
||||
|
||||
try {
|
||||
mockAPIData({
|
||||
type: 'icons',
|
||||
provider,
|
||||
@ -40,8 +42,6 @@ describe('Testing mock API module', () => {
|
||||
response: 404,
|
||||
});
|
||||
|
||||
let isSync = true;
|
||||
|
||||
loadIcons(
|
||||
[
|
||||
{
|
||||
@ -51,6 +51,7 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
],
|
||||
(loaded, missing, pending) => {
|
||||
try {
|
||||
expect(isSync).toBe(false);
|
||||
expect(loaded).toEqual([]);
|
||||
expect(pending).toEqual([]);
|
||||
@ -61,18 +62,27 @@ describe('Testing mock API module', () => {
|
||||
name: 'test1',
|
||||
},
|
||||
]);
|
||||
fulfill(true);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
resolve(true);
|
||||
}
|
||||
);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
|
||||
isSync = false;
|
||||
});
|
||||
});
|
||||
|
||||
it('Load few icons', () => {
|
||||
return new Promise((fulfill) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
const prefix = nextPrefix();
|
||||
let isSync = true;
|
||||
|
||||
try {
|
||||
mockAPIData({
|
||||
type: 'icons',
|
||||
provider,
|
||||
@ -105,8 +115,21 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
let isSync = true;
|
||||
// Data with invalid name: should not be requested from API because name is invalid
|
||||
// Split from main data because otherwise it would load when other icons are requested
|
||||
mockAPIData({
|
||||
type: 'icons',
|
||||
provider,
|
||||
prefix,
|
||||
response: {
|
||||
prefix,
|
||||
icons: {
|
||||
BadName: {
|
||||
body: '<g />',
|
||||
},
|
||||
},
|
||||
},
|
||||
});
|
||||
|
||||
loadIcons(
|
||||
[
|
||||
@ -120,8 +143,19 @@ describe('Testing mock API module', () => {
|
||||
prefix,
|
||||
name: 'test20',
|
||||
},
|
||||
{
|
||||
provider,
|
||||
prefix,
|
||||
name: 'BadName',
|
||||
},
|
||||
],
|
||||
(loaded, missing, pending) => {
|
||||
try {
|
||||
if (pending.length) {
|
||||
// Not ready to test yet
|
||||
return;
|
||||
}
|
||||
|
||||
expect(isSync).toBe(false);
|
||||
// All icons should have been loaded because API waits one tick before sending response, during which both queries are processed
|
||||
expect(loaded).toEqual([
|
||||
@ -137,20 +171,35 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
]);
|
||||
expect(pending).toEqual([]);
|
||||
expect(missing).toEqual([]);
|
||||
fulfill(true);
|
||||
expect(missing).toEqual([
|
||||
{
|
||||
provider,
|
||||
prefix,
|
||||
name: 'BadName',
|
||||
},
|
||||
]);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
resolve(true);
|
||||
}
|
||||
);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
|
||||
isSync = false;
|
||||
});
|
||||
});
|
||||
|
||||
it('Load in batches and testing delay', () => {
|
||||
return new Promise((fulfill, reject) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
const prefix = nextPrefix();
|
||||
let next: IconifyMockAPIDelayDoneCallback | undefined;
|
||||
let callbackCounter = 0;
|
||||
|
||||
try {
|
||||
mockAPIData({
|
||||
type: 'icons',
|
||||
provider,
|
||||
@ -187,8 +236,6 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
});
|
||||
|
||||
let callbackCounter = 0;
|
||||
|
||||
loadIcons(
|
||||
[
|
||||
{
|
||||
@ -204,6 +251,7 @@ describe('Testing mock API module', () => {
|
||||
],
|
||||
(loaded, missing, pending) => {
|
||||
callbackCounter++;
|
||||
try {
|
||||
switch (callbackCounter) {
|
||||
case 1:
|
||||
// First load: only 'test10'
|
||||
@ -224,7 +272,6 @@ describe('Testing mock API module', () => {
|
||||
|
||||
// Send second response
|
||||
expect(typeof next).toBe('function');
|
||||
// eslint-disable-next-line @typescript-eslint/no-non-null-assertion
|
||||
next!();
|
||||
break;
|
||||
|
||||
@ -243,7 +290,7 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
]);
|
||||
expect(missing).toEqual([]);
|
||||
fulfill(true);
|
||||
resolve(true);
|
||||
break;
|
||||
|
||||
default:
|
||||
@ -251,17 +298,24 @@ describe('Testing mock API module', () => {
|
||||
'Callback was called more times than expected'
|
||||
);
|
||||
}
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
}
|
||||
);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
// This is useful for testing component where loadIcons() cannot be accessed
|
||||
it('Using timer in callback for second test', () => {
|
||||
return new Promise((fulfill) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
const prefix = nextPrefix();
|
||||
const name = 'test1';
|
||||
|
||||
try {
|
||||
// Mock data
|
||||
mockAPIData({
|
||||
type: 'icons',
|
||||
@ -276,6 +330,7 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
},
|
||||
delay: (next) => {
|
||||
try {
|
||||
// Icon should not be loaded yet
|
||||
const storage = getStorage(provider, prefix);
|
||||
expect(iconInStorage(storage, name)).toBe(false);
|
||||
@ -285,8 +340,11 @@ describe('Testing mock API module', () => {
|
||||
|
||||
// Icon should be loaded now
|
||||
expect(iconInStorage(storage, name)).toBe(true);
|
||||
|
||||
fulfill(true);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
resolve(true);
|
||||
},
|
||||
});
|
||||
|
||||
@ -298,11 +356,17 @@ describe('Testing mock API module', () => {
|
||||
name,
|
||||
},
|
||||
]);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
});
|
||||
});
|
||||
|
||||
it('Custom query', () => {
|
||||
return new Promise((fulfill) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
let isSync = true;
|
||||
|
||||
try {
|
||||
mockAPIData({
|
||||
type: 'custom',
|
||||
provider,
|
||||
@ -312,8 +376,6 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
});
|
||||
|
||||
let isSync = true;
|
||||
|
||||
sendAPIQuery(
|
||||
provider,
|
||||
{
|
||||
@ -322,22 +384,33 @@ describe('Testing mock API module', () => {
|
||||
uri: '/test',
|
||||
},
|
||||
(data, error) => {
|
||||
try {
|
||||
expect(error).toBeUndefined();
|
||||
expect(data).toEqual({
|
||||
foo: true,
|
||||
});
|
||||
expect(isSync).toBe(false);
|
||||
fulfill(true);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
resolve(true);
|
||||
}
|
||||
);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
|
||||
isSync = false;
|
||||
});
|
||||
});
|
||||
|
||||
it('Custom query with host', () => {
|
||||
return new Promise((fulfill) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
const host = 'http://' + nextPrefix();
|
||||
let isSync = true;
|
||||
|
||||
try {
|
||||
setAPIModule(host, mockAPIModule);
|
||||
mockAPIData({
|
||||
type: 'host',
|
||||
@ -348,8 +421,6 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
});
|
||||
|
||||
let isSync = true;
|
||||
|
||||
sendAPIQuery(
|
||||
{
|
||||
resources: [host],
|
||||
@ -359,23 +430,33 @@ describe('Testing mock API module', () => {
|
||||
uri: '/test',
|
||||
},
|
||||
(data, error) => {
|
||||
try {
|
||||
expect(error).toBeUndefined();
|
||||
expect(data).toEqual({
|
||||
foo: 2,
|
||||
});
|
||||
expect(isSync).toBe(false);
|
||||
fulfill(true);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
resolve(true);
|
||||
}
|
||||
);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
|
||||
isSync = false;
|
||||
});
|
||||
});
|
||||
|
||||
it('not_found response', () => {
|
||||
return new Promise((fulfill) => {
|
||||
return new Promise((resolve, reject) => {
|
||||
const prefix = nextPrefix();
|
||||
let isSync = true;
|
||||
|
||||
try {
|
||||
mockAPIData({
|
||||
type: 'icons',
|
||||
provider,
|
||||
@ -388,8 +469,6 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
});
|
||||
|
||||
let isSync = true;
|
||||
|
||||
loadIcons(
|
||||
[
|
||||
{
|
||||
@ -399,6 +478,7 @@ describe('Testing mock API module', () => {
|
||||
},
|
||||
],
|
||||
(loaded, missing, pending) => {
|
||||
try {
|
||||
expect(isSync).toBe(false);
|
||||
expect(loaded).toEqual([]);
|
||||
expect(pending).toEqual([]);
|
||||
@ -409,9 +489,16 @@ describe('Testing mock API module', () => {
|
||||
name: 'test1',
|
||||
},
|
||||
]);
|
||||
fulfill(true);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
return;
|
||||
}
|
||||
resolve(true);
|
||||
}
|
||||
);
|
||||
} catch (error) {
|
||||
reject(error);
|
||||
}
|
||||
|
||||
isSync = false;
|
||||
});
|
||||
|
@ -319,4 +319,47 @@ describe('Testing parsing icon set', () => {
|
||||
parsedNames.sort((a, b) => a.localeCompare(b));
|
||||
expect(parsedNames).toEqual(expectedNames);
|
||||
});
|
||||
|
||||
test('Bug test 1', () => {
|
||||
// Names list
|
||||
const names: string[] = ['test20', 'test21', 'BadName'];
|
||||
|
||||
// Resolved data
|
||||
const expected: Record<string, ExtendedIconifyIcon | null> = {
|
||||
test20: {
|
||||
body: '<g />',
|
||||
},
|
||||
test21: {
|
||||
body: '<g />',
|
||||
},
|
||||
BadName: {
|
||||
body: '<g />',
|
||||
},
|
||||
};
|
||||
|
||||
// Do stuff
|
||||
expect(
|
||||
parseIconSet(
|
||||
{
|
||||
prefix: 'api-mock-02',
|
||||
icons: {
|
||||
test20: { body: '<g />' },
|
||||
test21: { body: '<g />' },
|
||||
BadName: { body: '<g />' },
|
||||
},
|
||||
},
|
||||
(name, data) => {
|
||||
// Make sure name matches
|
||||
expect(names.length).toBeGreaterThanOrEqual(1);
|
||||
expect(name).toBe(names.shift());
|
||||
|
||||
// Check icon data
|
||||
expect(data).toEqual(expected[name]);
|
||||
}
|
||||
)
|
||||
).toEqual(['test20', 'test21', 'BadName']);
|
||||
|
||||
// All names should have been parsed
|
||||
expect(names).toEqual([]);
|
||||
});
|
||||
});
|
||||
|
Loading…
Reference in New Issue
Block a user