mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
add cluster icon migration code (#673)
Signed-off-by: Sebastian Malton <smalton@mirantis.com>
This commit is contained in:
parent
b75cac4546
commit
a12aa8a04f
@ -175,6 +175,7 @@
|
|||||||
"crypto-js": "^4.0.0",
|
"crypto-js": "^4.0.0",
|
||||||
"electron-updater": "^4.3.1",
|
"electron-updater": "^4.3.1",
|
||||||
"electron-window-state": "^5.0.3",
|
"electron-window-state": "^5.0.3",
|
||||||
|
"file-type": "^14.7.1",
|
||||||
"filenamify": "^4.1.0",
|
"filenamify": "^4.1.0",
|
||||||
"fs-extra": "^9.0.1",
|
"fs-extra": "^9.0.1",
|
||||||
"handlebars": "^4.7.6",
|
"handlebars": "^4.7.6",
|
||||||
@ -184,6 +185,7 @@
|
|||||||
"jsonpath": "^1.0.2",
|
"jsonpath": "^1.0.2",
|
||||||
"lodash": "^4.17.15",
|
"lodash": "^4.17.15",
|
||||||
"mac-ca": "^1.0.4",
|
"mac-ca": "^1.0.4",
|
||||||
|
"make-synchronous": "^0.1.1",
|
||||||
"marked": "^1.1.0",
|
"marked": "^1.1.0",
|
||||||
"md5-file": "^5.0.0",
|
"md5-file": "^5.0.0",
|
||||||
"mobx": "^5.15.5",
|
"mobx": "^5.15.5",
|
||||||
|
|||||||
@ -1,6 +1,5 @@
|
|||||||
import type { WorkspaceId } from "./workspace-store";
|
import type { WorkspaceId } from "./workspace-store";
|
||||||
import path from "path";
|
import { ipcRenderer } from "electron";
|
||||||
import { app, ipcRenderer, remote } from "electron";
|
|
||||||
import { unlink } from "fs-extra";
|
import { unlink } from "fs-extra";
|
||||||
import { action, computed, observable, toJS } from "mobx";
|
import { action, computed, observable, toJS } from "mobx";
|
||||||
import { BaseStore } from "./base-store";
|
import { BaseStore } from "./base-store";
|
||||||
@ -50,11 +49,6 @@ export interface ClusterPreferences {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export class ClusterStore extends BaseStore<ClusterStoreModel> {
|
export class ClusterStore extends BaseStore<ClusterStoreModel> {
|
||||||
static get iconsDir() {
|
|
||||||
// TODO: remove remote cheat
|
|
||||||
return path.join((app || remote.app).getPath("userData"), "icons");
|
|
||||||
}
|
|
||||||
|
|
||||||
private constructor() {
|
private constructor() {
|
||||||
super({
|
super({
|
||||||
configName: "lens-cluster-store",
|
configName: "lens-cluster-store",
|
||||||
|
|||||||
@ -6,6 +6,10 @@ import { ClusterStore } from "./cluster-store";
|
|||||||
import { workspaceStore } from "./workspace-store";
|
import { workspaceStore } from "./workspace-store";
|
||||||
import { saveConfigToAppFiles } from "./kube-helpers";
|
import { saveConfigToAppFiles } from "./kube-helpers";
|
||||||
|
|
||||||
|
const testDataIcon = fs.readFileSync("test-data/cluster-store-migration-icon.png")
|
||||||
|
|
||||||
|
console.log("") // fix bug
|
||||||
|
|
||||||
let clusterStore: ClusterStore;
|
let clusterStore: ClusterStore;
|
||||||
|
|
||||||
describe("empty config", () => {
|
describe("empty config", () => {
|
||||||
@ -236,12 +240,13 @@ describe("pre 2.6.0 config with a cluster icon", () => {
|
|||||||
},
|
},
|
||||||
cluster1: {
|
cluster1: {
|
||||||
kubeConfig: "foo",
|
kubeConfig: "foo",
|
||||||
icon: "icon path",
|
icon: "icon_path",
|
||||||
preferences: {
|
preferences: {
|
||||||
terminalCWD: "/tmp"
|
terminalCWD: "/tmp"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
})
|
}),
|
||||||
|
"icon_path": testDataIcon,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
mockFs(mockOpts);
|
mockFs(mockOpts);
|
||||||
@ -257,7 +262,7 @@ describe("pre 2.6.0 config with a cluster icon", () => {
|
|||||||
const storedClusterData = clusterStore.clustersList[0];
|
const storedClusterData = clusterStore.clustersList[0];
|
||||||
expect(storedClusterData.hasOwnProperty('icon')).toBe(false);
|
expect(storedClusterData.hasOwnProperty('icon')).toBe(false);
|
||||||
expect(storedClusterData.preferences.hasOwnProperty('icon')).toBe(true);
|
expect(storedClusterData.preferences.hasOwnProperty('icon')).toBe(true);
|
||||||
expect(storedClusterData.preferences.icon).toBe("icon path");
|
expect(storedClusterData.preferences.icon.startsWith("data:image/jpeg;base64,")).toBe(true);
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
@ -274,7 +279,6 @@ describe("for a pre 2.7.0-beta.0 config without a workspace", () => {
|
|||||||
},
|
},
|
||||||
cluster1: {
|
cluster1: {
|
||||||
kubeConfig: "foo",
|
kubeConfig: "foo",
|
||||||
icon: "icon path",
|
|
||||||
preferences: {
|
preferences: {
|
||||||
terminalCWD: "/tmp"
|
terminalCWD: "/tmp"
|
||||||
}
|
}
|
||||||
@ -305,16 +309,20 @@ describe("pre 3.6.0-beta.1 config with an existing cluster", () => {
|
|||||||
'lens-cluster-store.json': JSON.stringify({
|
'lens-cluster-store.json': JSON.stringify({
|
||||||
__internal__: {
|
__internal__: {
|
||||||
migrations: {
|
migrations: {
|
||||||
version: "2.7.0"
|
version: "3.5.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
clusters: [
|
clusters: [
|
||||||
{
|
{
|
||||||
id: 'cluster1',
|
id: 'cluster1',
|
||||||
kubeConfig: 'kubeconfig content'
|
kubeConfig: 'kubeconfig content',
|
||||||
|
preferences: {
|
||||||
|
icon: "store://icon_path",
|
||||||
|
}
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
})
|
}),
|
||||||
|
"icon_path": testDataIcon,
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
mockFs(mockOpts);
|
mockFs(mockOpts);
|
||||||
@ -330,4 +338,9 @@ describe("pre 3.6.0-beta.1 config with an existing cluster", () => {
|
|||||||
const config = clusterStore.clustersList[0].kubeConfigPath;
|
const config = clusterStore.clustersList[0].kubeConfigPath;
|
||||||
expect(fs.readFileSync(config, "utf8")).toBe("kubeconfig content");
|
expect(fs.readFileSync(config, "utf8")).toBe("kubeconfig content");
|
||||||
})
|
})
|
||||||
|
|
||||||
|
it("migrates to modern format with icon not in file", async () => {
|
||||||
|
const { icon } = clusterStore.clustersList[0].preferences;
|
||||||
|
expect(icon.startsWith("data:image/jpeg;base64, ")).toBe(true);
|
||||||
|
})
|
||||||
})
|
})
|
||||||
@ -6,7 +6,7 @@ export default migration({
|
|||||||
run(store, log) {
|
run(store, log) {
|
||||||
for (const value of store) {
|
for (const value of store) {
|
||||||
const clusterKey = value[0];
|
const clusterKey = value[0];
|
||||||
if(clusterKey === "__internal__") continue
|
if (clusterKey === "__internal__") continue
|
||||||
const cluster = value[1];
|
const cluster = value[1];
|
||||||
cluster.workspace = "default"
|
cluster.workspace = "default"
|
||||||
store.set(clusterKey, cluster)
|
store.set(clusterKey, cluster)
|
||||||
|
|||||||
@ -1,35 +1,72 @@
|
|||||||
// Move embedded kubeconfig into separate file and add reference to it to cluster settings
|
// Move embedded kubeconfig into separate file and add reference to it to cluster settings
|
||||||
|
// convert file path cluster icons to their base64 encoded versions
|
||||||
|
|
||||||
import path from "path"
|
import path from "path"
|
||||||
import { app, remote } from "electron"
|
import { app, remote } from "electron"
|
||||||
import { migration } from "../migration-wrapper";
|
import { migration } from "../migration-wrapper";
|
||||||
import { ensureDirSync } from "fs-extra"
|
import fse from "fs-extra"
|
||||||
import { ClusterModel } from "../../common/cluster-store";
|
import { ClusterModel } from "../../common/cluster-store";
|
||||||
import { loadConfig, saveConfigToAppFiles } from "../../common/kube-helpers";
|
import { loadConfig, saveConfigToAppFiles } from "../../common/kube-helpers";
|
||||||
|
import makeSynchronous from "make-synchronous"
|
||||||
|
|
||||||
|
const AsyncFunction = Object.getPrototypeOf(async function () { return }).constructor;
|
||||||
|
const getFileTypeFnString = `return require("file-type").fromBuffer(fileData)`;
|
||||||
|
const getFileType = new AsyncFunction("fileData", getFileTypeFnString);
|
||||||
|
|
||||||
export default migration({
|
export default migration({
|
||||||
version: "3.6.0-beta.1",
|
version: "3.6.0-beta.1",
|
||||||
run(store, printLog) {
|
run(store, printLog) {
|
||||||
const migratedClusters: ClusterModel[] = []
|
const userDataPath = (app || remote.app).getPath("userData")
|
||||||
const storedClusters: ClusterModel[] = store.get("clusters");
|
const kubeConfigBase = path.join(userDataPath, "kubeconfigs")
|
||||||
const kubeConfigBase = path.join((app || remote.app).getPath("userData"), "kubeconfigs")
|
const storedClusters: ClusterModel[] = store.get("clusters") || [];
|
||||||
|
|
||||||
if (!storedClusters) return;
|
if (!storedClusters.length) return;
|
||||||
ensureDirSync(kubeConfigBase);
|
fse.ensureDirSync(kubeConfigBase);
|
||||||
|
|
||||||
printLog("Number of clusters to migrate: ", storedClusters.length)
|
printLog("Number of clusters to migrate: ", storedClusters.length)
|
||||||
for (const cluster of storedClusters) {
|
const migratedClusters = storedClusters
|
||||||
|
.map(cluster => {
|
||||||
|
/**
|
||||||
|
* migrate kubeconfig
|
||||||
|
*/
|
||||||
try {
|
try {
|
||||||
// take the embedded kubeconfig and dump it into a file
|
// take the embedded kubeconfig and dump it into a file
|
||||||
cluster.kubeConfigPath = saveConfigToAppFiles(cluster.id, cluster.kubeConfig)
|
cluster.kubeConfigPath = saveConfigToAppFiles(cluster.id, cluster.kubeConfig)
|
||||||
cluster.contextName = loadConfig(cluster.kubeConfigPath).getCurrentContext();
|
cluster.contextName = loadConfig(cluster.kubeConfigPath).getCurrentContext();
|
||||||
delete cluster.kubeConfig;
|
delete cluster.kubeConfig;
|
||||||
migratedClusters.push(cluster)
|
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
printLog(`Failed to migrate Kubeconfig for cluster "${cluster.id}"`, error)
|
printLog(`Failed to migrate Kubeconfig for cluster "${cluster.id}", removing cluster...`, error)
|
||||||
|
return undefined;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* migrate cluster icon
|
||||||
|
*/
|
||||||
|
try {
|
||||||
|
if (cluster.preferences?.icon) {
|
||||||
|
printLog(`migrating ${cluster.preferences.icon} for ${cluster.preferences.clusterName}`)
|
||||||
|
const iconPath = cluster.preferences.icon.replace("store://", "")
|
||||||
|
const fileData = fse.readFileSync(path.join(userDataPath, iconPath));
|
||||||
|
const { mime = "" } = makeSynchronous(getFileType)(fileData);
|
||||||
|
|
||||||
|
if (!mime) {
|
||||||
|
printLog(`mime type not detected for ${cluster.preferences.clusterName}'s icon: ${iconPath}`)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
cluster.preferences.icon = `data:${mime};base64, ${fileData.toString('base64')}`;
|
||||||
|
} else {
|
||||||
|
delete cluster.preferences?.icon;
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
printLog(`Failed to migrate cluster icon for cluster "${cluster.id}"`, error)
|
||||||
|
delete cluster.preferences.icon;
|
||||||
|
}
|
||||||
|
|
||||||
|
return cluster;
|
||||||
|
})
|
||||||
|
.filter(c => c);
|
||||||
|
|
||||||
// "overwrite" the cluster configs
|
// "overwrite" the cluster configs
|
||||||
if (migratedClusters.length > 0) {
|
if (migratedClusters.length > 0) {
|
||||||
store.set("clusters", migratedClusters)
|
store.set("clusters", migratedClusters)
|
||||||
|
|||||||
@ -28,7 +28,7 @@ export class ClusterIconSetting extends React.Component<Props> {
|
|||||||
try {
|
try {
|
||||||
if (file) {
|
if (file) {
|
||||||
const buf = Buffer.from(await file.arrayBuffer());
|
const buf = Buffer.from(await file.arrayBuffer());
|
||||||
cluster.preferences.icon = `data:image/jpeg;base64, ${buf.toString('base64')}`;
|
cluster.preferences.icon = `data:${file.type};base64, ${buf.toString('base64')}`;
|
||||||
} else {
|
} else {
|
||||||
// this has to be done as a seperate branch (and not always) because `cluster`
|
// this has to be done as a seperate branch (and not always) because `cluster`
|
||||||
// is observable and triggers an update loop.
|
// is observable and triggers an update loop.
|
||||||
|
|||||||
BIN
test-data/cluster-store-migration-icon.png
Normal file
BIN
test-data/cluster-store-migration-icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 29 KiB |
65
yarn.lock
65
yarn.lock
@ -1588,6 +1588,11 @@
|
|||||||
dependencies:
|
dependencies:
|
||||||
defer-to-connect "^1.0.1"
|
defer-to-connect "^1.0.1"
|
||||||
|
|
||||||
|
"@tokenizer/token@^0.1.0", "@tokenizer/token@^0.1.1":
|
||||||
|
version "0.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/@tokenizer/token/-/token-0.1.1.tgz#f0d92c12f87079ddfd1b29f614758b9696bc29e3"
|
||||||
|
integrity sha512-XO6INPbZCxdprl+9qa/AAbFFOMzzwqYxpjPgLICrMD6C2FCw6qfJOPcBk6JqqPLSaZ/Qx87qn4rpPmPMwaAK6w==
|
||||||
|
|
||||||
"@types/anymatch@*":
|
"@types/anymatch@*":
|
||||||
version "1.3.1"
|
version "1.3.1"
|
||||||
resolved "https://registry.yarnpkg.com/@types/anymatch/-/anymatch-1.3.1.tgz#336badc1beecb9dacc38bea2cf32adf627a8421a"
|
resolved "https://registry.yarnpkg.com/@types/anymatch/-/anymatch-1.3.1.tgz#336badc1beecb9dacc38bea2cf32adf627a8421a"
|
||||||
@ -5196,6 +5201,16 @@ file-loader@^6.0.0:
|
|||||||
loader-utils "^2.0.0"
|
loader-utils "^2.0.0"
|
||||||
schema-utils "^2.6.5"
|
schema-utils "^2.6.5"
|
||||||
|
|
||||||
|
file-type@^14.7.1:
|
||||||
|
version "14.7.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/file-type/-/file-type-14.7.1.tgz#f748732b3e70478bff530e1cf0ec2fe33608b1bb"
|
||||||
|
integrity sha512-sXAMgFk67fQLcetXustxfKX+PZgHIUFn96Xld9uH8aXPdX3xOp0/jg9OdouVTvQrf7mrn+wAa4jN/y9fUOOiRA==
|
||||||
|
dependencies:
|
||||||
|
readable-web-to-node-stream "^2.0.0"
|
||||||
|
strtok3 "^6.0.3"
|
||||||
|
token-types "^2.0.0"
|
||||||
|
typedarray-to-buffer "^3.1.5"
|
||||||
|
|
||||||
file-uri-to-path@1.0.0:
|
file-uri-to-path@1.0.0:
|
||||||
version "1.0.0"
|
version "1.0.0"
|
||||||
resolved "https://registry.yarnpkg.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz#553a7b8446ff6f684359c445f1e37a05dacc33dd"
|
resolved "https://registry.yarnpkg.com/file-uri-to-path/-/file-uri-to-path-1.0.0.tgz#553a7b8446ff6f684359c445f1e37a05dacc33dd"
|
||||||
@ -6074,7 +6089,7 @@ identity-obj-proxy@^3.0.0:
|
|||||||
dependencies:
|
dependencies:
|
||||||
harmony-reflect "^1.4.6"
|
harmony-reflect "^1.4.6"
|
||||||
|
|
||||||
ieee754@^1.1.4:
|
ieee754@^1.1.13, ieee754@^1.1.4:
|
||||||
version "1.1.13"
|
version "1.1.13"
|
||||||
resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.1.13.tgz#ec168558e95aa181fd87d37f55c32bbcb6708b84"
|
resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.1.13.tgz#ec168558e95aa181fd87d37f55c32bbcb6708b84"
|
||||||
integrity sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==
|
integrity sha512-4vf7I2LYV/HaWerSo3XmlMkp5eZ83i+/CDluXi/IGTs/O1sejBNhTtnxzmRZfvOUqj7lZjqHkeTvpgSFDlWZTg==
|
||||||
@ -7632,6 +7647,14 @@ make-plural@^6.2.1:
|
|||||||
resolved "https://registry.yarnpkg.com/make-plural/-/make-plural-6.2.1.tgz#2790af1d05fb2fc35a111ce759ffdb0aca1339a3"
|
resolved "https://registry.yarnpkg.com/make-plural/-/make-plural-6.2.1.tgz#2790af1d05fb2fc35a111ce759ffdb0aca1339a3"
|
||||||
integrity sha512-AmkruwJ9EjvyTv6AM8MBMK3TAeOJvhgTv5YQXzF0EP2qawhpvMjDpHvsdOIIT0Vn+BB0+IogmYZ1z+Ulm/m0Fg==
|
integrity sha512-AmkruwJ9EjvyTv6AM8MBMK3TAeOJvhgTv5YQXzF0EP2qawhpvMjDpHvsdOIIT0Vn+BB0+IogmYZ1z+Ulm/m0Fg==
|
||||||
|
|
||||||
|
make-synchronous@^0.1.1:
|
||||||
|
version "0.1.1"
|
||||||
|
resolved "https://registry.yarnpkg.com/make-synchronous/-/make-synchronous-0.1.1.tgz#0169f6ec769c3cf8948d66790da262740c1209e7"
|
||||||
|
integrity sha512-Y4SxxqhaoyMDokJQ0AZz0E+bLhRkOSR7Z/IQoTKPdS6HYi3aobal2kMHoHHoqBadPWjf07P4K1FQLXOx3wf9Yw==
|
||||||
|
dependencies:
|
||||||
|
subsume "^3.0.0"
|
||||||
|
type-fest "^0.16.0"
|
||||||
|
|
||||||
makeerror@1.0.x:
|
makeerror@1.0.x:
|
||||||
version "1.0.11"
|
version "1.0.11"
|
||||||
resolved "https://registry.yarnpkg.com/makeerror/-/makeerror-1.0.11.tgz#e01a5c9109f2af79660e4e8b9587790184f5a96c"
|
resolved "https://registry.yarnpkg.com/makeerror/-/makeerror-1.0.11.tgz#e01a5c9109f2af79660e4e8b9587790184f5a96c"
|
||||||
@ -8943,6 +8966,11 @@ pbkdf2@^3.0.3:
|
|||||||
safe-buffer "^5.0.1"
|
safe-buffer "^5.0.1"
|
||||||
sha.js "^2.4.8"
|
sha.js "^2.4.8"
|
||||||
|
|
||||||
|
peek-readable@^3.1.0:
|
||||||
|
version "3.1.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/peek-readable/-/peek-readable-3.1.0.tgz#250b08b7de09db8573d7fd8ea475215bbff14348"
|
||||||
|
integrity sha512-KGuODSTV6hcgdZvDrIDBUkN0utcAVj1LL7FfGbM0viKTtCHmtZcuEJ+lGqsp0fTFkGqesdtemV2yUSMeyy3ddA==
|
||||||
|
|
||||||
pend@~1.2.0:
|
pend@~1.2.0:
|
||||||
version "1.2.0"
|
version "1.2.0"
|
||||||
resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50"
|
resolved "https://registry.yarnpkg.com/pend/-/pend-1.2.0.tgz#7a57eb550a6783f9115331fcf4663d5c8e007a50"
|
||||||
@ -9536,6 +9564,11 @@ readable-stream@^3.1.1, readable-stream@^3.6.0:
|
|||||||
string_decoder "^1.1.1"
|
string_decoder "^1.1.1"
|
||||||
util-deprecate "^1.0.1"
|
util-deprecate "^1.0.1"
|
||||||
|
|
||||||
|
readable-web-to-node-stream@^2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/readable-web-to-node-stream/-/readable-web-to-node-stream-2.0.0.tgz#751e632f466552ac0d5c440cc01470352f93c4b7"
|
||||||
|
integrity sha512-+oZJurc4hXpaaqsN68GoZGQAQIA3qr09Or4fqEsargABnbe5Aau8hFn6ISVleT3cpY/0n/8drn7huyyEvTbghA==
|
||||||
|
|
||||||
readdirp@^2.2.1:
|
readdirp@^2.2.1:
|
||||||
version "2.2.1"
|
version "2.2.1"
|
||||||
resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-2.2.1.tgz#0e87622a3325aa33e892285caf8b4e846529a525"
|
resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-2.2.1.tgz#0e87622a3325aa33e892285caf8b4e846529a525"
|
||||||
@ -10655,6 +10688,15 @@ strip-outer@^1.0.1:
|
|||||||
dependencies:
|
dependencies:
|
||||||
escape-string-regexp "^1.0.2"
|
escape-string-regexp "^1.0.2"
|
||||||
|
|
||||||
|
strtok3@^6.0.3:
|
||||||
|
version "6.0.4"
|
||||||
|
resolved "https://registry.yarnpkg.com/strtok3/-/strtok3-6.0.4.tgz#ede0d20fde5aa9fda56417c3558eaafccc724694"
|
||||||
|
integrity sha512-rqWMKwsbN9APU47bQTMEYTPcwdpKDtmf1jVhHzNW2cL1WqAxaM9iBb9t5P2fj+RV2YsErUWgQzHD5JwV0uCTEQ==
|
||||||
|
dependencies:
|
||||||
|
"@tokenizer/token" "^0.1.1"
|
||||||
|
"@types/debug" "^4.1.5"
|
||||||
|
peek-readable "^3.1.0"
|
||||||
|
|
||||||
style-loader@^1.2.1:
|
style-loader@^1.2.1:
|
||||||
version "1.2.1"
|
version "1.2.1"
|
||||||
resolved "https://registry.yarnpkg.com/style-loader/-/style-loader-1.2.1.tgz#c5cbbfbf1170d076cfdd86e0109c5bba114baa1a"
|
resolved "https://registry.yarnpkg.com/style-loader/-/style-loader-1.2.1.tgz#c5cbbfbf1170d076cfdd86e0109c5bba114baa1a"
|
||||||
@ -10663,6 +10705,14 @@ style-loader@^1.2.1:
|
|||||||
loader-utils "^2.0.0"
|
loader-utils "^2.0.0"
|
||||||
schema-utils "^2.6.6"
|
schema-utils "^2.6.6"
|
||||||
|
|
||||||
|
subsume@^3.0.0:
|
||||||
|
version "3.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/subsume/-/subsume-3.0.0.tgz#22c92730f441ad72ee9af4bdad42dc4ff830cfaf"
|
||||||
|
integrity sha512-6n/UfV8UWKwJNO8OAOiKntwEMihuBeeoJfzpL542C+OuvT4iWG9SwjrXkOmsxjb4SteHUsos9SvrdqZ9+ICwTQ==
|
||||||
|
dependencies:
|
||||||
|
escape-string-regexp "^2.0.0"
|
||||||
|
unique-string "^2.0.0"
|
||||||
|
|
||||||
sumchecker@^3.0.1:
|
sumchecker@^3.0.1:
|
||||||
version "3.0.1"
|
version "3.0.1"
|
||||||
resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz#6377e996795abb0b6d348e9b3e1dfb24345a8e42"
|
resolved "https://registry.yarnpkg.com/sumchecker/-/sumchecker-3.0.1.tgz#6377e996795abb0b6d348e9b3e1dfb24345a8e42"
|
||||||
@ -10984,6 +11034,14 @@ to-regex@^3.0.1, to-regex@^3.0.2:
|
|||||||
regex-not "^1.0.2"
|
regex-not "^1.0.2"
|
||||||
safe-regex "^1.1.0"
|
safe-regex "^1.1.0"
|
||||||
|
|
||||||
|
token-types@^2.0.0:
|
||||||
|
version "2.0.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/token-types/-/token-types-2.0.0.tgz#b23618af744818299c6fbf125e0fdad98bab7e85"
|
||||||
|
integrity sha512-WWvu8sGK8/ZmGusekZJJ5NM6rRVTTDO7/bahz4NGiSDb/XsmdYBn6a1N/bymUHuWYTWeuLUg98wUzvE4jPdCZw==
|
||||||
|
dependencies:
|
||||||
|
"@tokenizer/token" "^0.1.0"
|
||||||
|
ieee754 "^1.1.13"
|
||||||
|
|
||||||
touch@^3.1.0:
|
touch@^3.1.0:
|
||||||
version "3.1.0"
|
version "3.1.0"
|
||||||
resolved "https://registry.yarnpkg.com/touch/-/touch-3.1.0.tgz#fe365f5f75ec9ed4e56825e0bb76d24ab74af83b"
|
resolved "https://registry.yarnpkg.com/touch/-/touch-3.1.0.tgz#fe365f5f75ec9ed4e56825e0bb76d24ab74af83b"
|
||||||
@ -11157,6 +11215,11 @@ type-fest@^0.13.1:
|
|||||||
resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz#0172cb5bce80b0bd542ea348db50c7e21834d934"
|
resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.13.1.tgz#0172cb5bce80b0bd542ea348db50c7e21834d934"
|
||||||
integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg==
|
integrity sha512-34R7HTnG0XIJcBSn5XhDd7nNFPRcXYRZrBB2O2jdKqYODldSzBAqzsWoZYYvduky73toYS/ESqxPvkDf/F0XMg==
|
||||||
|
|
||||||
|
type-fest@^0.16.0:
|
||||||
|
version "0.16.0"
|
||||||
|
resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.16.0.tgz#3240b891a78b0deae910dbeb86553e552a148860"
|
||||||
|
integrity sha512-eaBzG6MxNzEn9kiwvtre90cXaNLkmadMWa1zQMs3XORCXNbsH/OewwbxC5ia9dCxIxnTAsSxXJaa/p5y8DlvJg==
|
||||||
|
|
||||||
type-fest@^0.6.0:
|
type-fest@^0.6.0:
|
||||||
version "0.6.0"
|
version "0.6.0"
|
||||||
resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.6.0.tgz#8d2a2370d3df886eb5c90ada1c5bf6188acf838b"
|
resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.6.0.tgz#8d2a2370d3df886eb5c90ada1c5bf6188acf838b"
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user