wxt 0.12.4 → 0.13.0
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/dist/{chunk-GPKQ6U2A.js → chunk-GNIOSZT6.js} +2 -2
- package/dist/cli.js +2 -2
- package/dist/index.cjs +2 -2
- package/dist/index.d.cts +1 -1
- package/dist/index.d.ts +1 -1
- package/dist/index.js +1 -1
- package/dist/storage.cjs +310 -83
- package/dist/storage.d.cts +185 -12
- package/dist/storage.d.ts +185 -12
- package/dist/storage.js +309 -84
- package/dist/testing.cjs +1 -1
- package/dist/testing.js +1 -1
- package/package.json +2 -2
package/dist/storage.js
CHANGED
|
@@ -1,108 +1,333 @@
|
|
|
1
1
|
import "./chunk-VBXJIVYU.js";
|
|
2
2
|
|
|
3
|
-
// src/storage.ts
|
|
4
|
-
import {
|
|
5
|
-
createStorage,
|
|
6
|
-
defineDriver
|
|
7
|
-
} from "unstorage";
|
|
8
|
-
|
|
9
3
|
// src/browser.ts
|
|
10
4
|
import originalBrowser from "webextension-polyfill";
|
|
11
5
|
var browser = originalBrowser;
|
|
12
6
|
|
|
13
7
|
// src/storage.ts
|
|
14
|
-
|
|
15
|
-
var
|
|
16
|
-
|
|
17
|
-
|
|
8
|
+
import { dequal } from "dequal/lite";
|
|
9
|
+
var storage = createStorage();
|
|
10
|
+
function createStorage() {
|
|
11
|
+
const drivers = {
|
|
12
|
+
local: createDriver("local"),
|
|
13
|
+
session: createDriver("session"),
|
|
14
|
+
sync: createDriver("sync"),
|
|
15
|
+
managed: createDriver("managed")
|
|
16
|
+
};
|
|
17
|
+
const getDriver = (area) => {
|
|
18
|
+
const driver = drivers[area];
|
|
19
|
+
if (driver == null) {
|
|
20
|
+
const areaNames = Object.keys(drivers).join(", ");
|
|
21
|
+
throw Error(`Invalid area "${area}". Options: ${areaNames}`);
|
|
22
|
+
}
|
|
23
|
+
return driver;
|
|
24
|
+
};
|
|
25
|
+
const resolveKey = (key) => {
|
|
26
|
+
const [driverArea, driverKey] = key.split(":", 2);
|
|
27
|
+
if (driverKey == null)
|
|
18
28
|
throw Error(
|
|
19
|
-
|
|
29
|
+
`Storage key should be in the form of "area:key", but recieved "${key}"`
|
|
20
30
|
);
|
|
31
|
+
return {
|
|
32
|
+
driverArea,
|
|
33
|
+
driverKey,
|
|
34
|
+
driver: getDriver(driverArea)
|
|
35
|
+
};
|
|
21
36
|
};
|
|
22
|
-
const
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
37
|
+
const getMetaKey = (key) => key + "$";
|
|
38
|
+
const getValueOrDefault = (value, defaultValue) => value ?? defaultValue ?? null;
|
|
39
|
+
const getMetaValue = (properties) => typeof properties === "object" && !Array.isArray(properties) ? properties : {};
|
|
40
|
+
const getItem = async (driver, driverKey, opts) => {
|
|
41
|
+
const res = await driver.getItem(driverKey);
|
|
42
|
+
return getValueOrDefault(res, opts?.defaultValue);
|
|
43
|
+
};
|
|
44
|
+
const getMeta = async (driver, driverKey) => {
|
|
45
|
+
const metaKey = getMetaKey(driverKey);
|
|
46
|
+
const res = await driver.getItem(metaKey);
|
|
47
|
+
return getMetaValue(res);
|
|
48
|
+
};
|
|
49
|
+
const setItem = async (driver, driverKey, value) => {
|
|
50
|
+
await driver.setItem(driverKey, value ?? null);
|
|
51
|
+
};
|
|
52
|
+
const setMeta = async (driver, driverKey, properties) => {
|
|
53
|
+
const metaKey = getMetaKey(driverKey);
|
|
54
|
+
const existingFields = getMetaValue(await driver.getItem(metaKey));
|
|
55
|
+
const newFields = { ...existingFields };
|
|
56
|
+
Object.entries(properties).forEach(([key, value]) => {
|
|
57
|
+
if (value == null) {
|
|
58
|
+
delete newFields[key];
|
|
59
|
+
} else {
|
|
60
|
+
newFields[key] = value;
|
|
61
|
+
}
|
|
27
62
|
});
|
|
63
|
+
await driver.setItem(metaKey, newFields);
|
|
28
64
|
};
|
|
29
|
-
const
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
65
|
+
const removeItem = async (driver, driverKey, opts) => {
|
|
66
|
+
await driver.removeItem(driverKey);
|
|
67
|
+
if (opts?.removeMeta) {
|
|
68
|
+
const metaKey = getMetaKey(driverKey);
|
|
69
|
+
await driver.removeItem(metaKey);
|
|
70
|
+
}
|
|
71
|
+
};
|
|
72
|
+
const removeMeta = async (driver, driverKey, properties) => {
|
|
73
|
+
const metaKey = getMetaKey(driverKey);
|
|
74
|
+
if (properties == null) {
|
|
75
|
+
await driver.removeItem(metaKey);
|
|
76
|
+
} else {
|
|
77
|
+
const newFields = getMetaValue(await driver.getItem(metaKey));
|
|
78
|
+
[properties].flat().forEach((field) => delete newFields[field]);
|
|
79
|
+
await driver.setItem(metaKey, newFields);
|
|
80
|
+
}
|
|
81
|
+
};
|
|
82
|
+
const watch = (driver, driverKey, cb) => {
|
|
83
|
+
return driver.watch(driverKey, cb);
|
|
84
|
+
};
|
|
85
|
+
const storage2 = {
|
|
86
|
+
getItem: async (key, opts) => {
|
|
87
|
+
const { driver, driverKey } = resolveKey(key);
|
|
88
|
+
return await getItem(driver, driverKey, opts);
|
|
89
|
+
},
|
|
90
|
+
getItems: async (keys) => {
|
|
91
|
+
const areaToKeyMap = /* @__PURE__ */ new Map();
|
|
92
|
+
const keyToOptsMap = /* @__PURE__ */ new Map();
|
|
93
|
+
keys.forEach((key) => {
|
|
94
|
+
let keyStr;
|
|
95
|
+
let opts;
|
|
96
|
+
if (typeof key === "string") {
|
|
97
|
+
keyStr = key;
|
|
98
|
+
} else {
|
|
99
|
+
keyStr = key.key;
|
|
100
|
+
opts = key.options;
|
|
101
|
+
}
|
|
102
|
+
const { driverArea, driverKey } = resolveKey(keyStr);
|
|
103
|
+
const keys2 = areaToKeyMap.get(driverArea) ?? [];
|
|
104
|
+
areaToKeyMap.set(driverArea, keys2.concat(driverKey));
|
|
105
|
+
keyToOptsMap.set(keyStr, opts);
|
|
106
|
+
});
|
|
107
|
+
const results = await Promise.all(
|
|
108
|
+
Array.from(areaToKeyMap.entries()).map(async ([driverArea, keys2]) => {
|
|
109
|
+
const driverResults = await drivers[driverArea].getItems(keys2);
|
|
110
|
+
return driverResults.map((driverResult) => {
|
|
111
|
+
const key = `${driverArea}:${driverResult.key}`;
|
|
112
|
+
const value = getValueOrDefault(
|
|
113
|
+
driverResult.value,
|
|
114
|
+
keyToOptsMap.get(key)?.defaultValue
|
|
115
|
+
);
|
|
116
|
+
return { key, value };
|
|
117
|
+
});
|
|
118
|
+
})
|
|
119
|
+
);
|
|
120
|
+
return results.flat();
|
|
121
|
+
},
|
|
122
|
+
getMeta: async (key) => {
|
|
123
|
+
const { driver, driverKey } = resolveKey(key);
|
|
124
|
+
return await getMeta(driver, driverKey);
|
|
125
|
+
},
|
|
126
|
+
setItem: async (key, value) => {
|
|
127
|
+
const { driver, driverKey } = resolveKey(key);
|
|
128
|
+
await setItem(driver, driverKey, value);
|
|
129
|
+
},
|
|
130
|
+
setItems: async (values) => {
|
|
131
|
+
const areaToKeyValueMap = /* @__PURE__ */ new Map();
|
|
132
|
+
values.forEach(({ key, value }) => {
|
|
133
|
+
const { driverArea, driverKey } = resolveKey(key);
|
|
134
|
+
const values2 = areaToKeyValueMap.get(driverArea) ?? [];
|
|
135
|
+
areaToKeyValueMap.set(
|
|
136
|
+
driverArea,
|
|
137
|
+
values2.concat({ key: driverKey, value })
|
|
138
|
+
);
|
|
139
|
+
});
|
|
140
|
+
await Promise.all(
|
|
141
|
+
Array.from(areaToKeyValueMap.entries()).map(
|
|
142
|
+
async ([driverArea, values2]) => {
|
|
143
|
+
const driver = getDriver(driverArea);
|
|
144
|
+
await driver.setItems(values2);
|
|
145
|
+
}
|
|
146
|
+
)
|
|
46
147
|
);
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
},
|
|
52
|
-
async
|
|
53
|
-
|
|
54
|
-
await
|
|
55
|
-
},
|
|
56
|
-
async
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
await
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
148
|
+
},
|
|
149
|
+
setMeta: async (key, properties) => {
|
|
150
|
+
const { driver, driverKey } = resolveKey(key);
|
|
151
|
+
await setMeta(driver, driverKey, properties);
|
|
152
|
+
},
|
|
153
|
+
removeItem: async (key, opts) => {
|
|
154
|
+
const { driver, driverKey } = resolveKey(key);
|
|
155
|
+
await removeItem(driver, driverKey, opts);
|
|
156
|
+
},
|
|
157
|
+
removeItems: async (keys) => {
|
|
158
|
+
const areaToKeysMap = /* @__PURE__ */ new Map();
|
|
159
|
+
keys.forEach((key) => {
|
|
160
|
+
let keyStr;
|
|
161
|
+
let opts;
|
|
162
|
+
if (typeof key === "string") {
|
|
163
|
+
keyStr = key;
|
|
164
|
+
} else {
|
|
165
|
+
keyStr = key.key;
|
|
166
|
+
opts = key.options;
|
|
167
|
+
}
|
|
168
|
+
const { driverArea, driverKey } = resolveKey(keyStr);
|
|
169
|
+
const areaKeys = areaToKeysMap.get(driverArea) ?? [];
|
|
170
|
+
areaKeys.push(driverKey);
|
|
171
|
+
if (opts?.removeMeta) {
|
|
172
|
+
areaKeys.push(getMetaKey(driverKey));
|
|
173
|
+
}
|
|
174
|
+
areaToKeysMap.set(driverArea, areaKeys);
|
|
175
|
+
});
|
|
176
|
+
await Promise.all(
|
|
177
|
+
Array.from(areaToKeysMap.entries()).map(async ([driverArea, keys2]) => {
|
|
178
|
+
const driver = getDriver(driverArea);
|
|
179
|
+
await driver.removeItems(keys2);
|
|
180
|
+
})
|
|
181
|
+
);
|
|
182
|
+
},
|
|
183
|
+
removeMeta: async (key, properties) => {
|
|
184
|
+
const { driver, driverKey } = resolveKey(key);
|
|
185
|
+
await removeMeta(driver, driverKey, properties);
|
|
186
|
+
},
|
|
187
|
+
snapshot: async (base, opts) => {
|
|
188
|
+
const driver = getDriver(base);
|
|
189
|
+
const data = await driver.snapshot();
|
|
190
|
+
opts?.excludeKeys?.forEach((key) => {
|
|
191
|
+
delete data[key];
|
|
192
|
+
delete data[getMetaKey(key)];
|
|
193
|
+
});
|
|
194
|
+
return data;
|
|
195
|
+
},
|
|
196
|
+
restoreSnapshot: async (base, data) => {
|
|
197
|
+
const driver = getDriver(base);
|
|
198
|
+
await driver.restoreSnapshot(data);
|
|
199
|
+
},
|
|
200
|
+
watch: (key, cb) => {
|
|
201
|
+
const { driver, driverKey } = resolveKey(key);
|
|
202
|
+
return watch(driver, driverKey, cb);
|
|
203
|
+
},
|
|
204
|
+
unwatch() {
|
|
205
|
+
Object.values(drivers).forEach((driver) => {
|
|
206
|
+
driver.unwatch();
|
|
207
|
+
});
|
|
208
|
+
},
|
|
209
|
+
defineItem: (key, opts) => {
|
|
210
|
+
const { driver, driverKey } = resolveKey(key);
|
|
211
|
+
const { version: targetVersion = 1, migrations = {} } = opts ?? {};
|
|
212
|
+
if (targetVersion < 1) {
|
|
213
|
+
throw Error(
|
|
214
|
+
"Storage item version cannot be less than 1. Initial versions should be set to 1, not 0."
|
|
83
215
|
);
|
|
84
216
|
}
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
217
|
+
const runMigrations = async () => {
|
|
218
|
+
const [value, meta] = await Promise.all([
|
|
219
|
+
// TODO: Optimize with getItems
|
|
220
|
+
getItem(driver, driverKey, void 0),
|
|
221
|
+
getMeta(driver, driverKey)
|
|
222
|
+
]);
|
|
223
|
+
if (value == null)
|
|
224
|
+
return;
|
|
225
|
+
const currentVersion = meta.v ?? 1;
|
|
226
|
+
if (currentVersion > targetVersion) {
|
|
227
|
+
throw Error(
|
|
228
|
+
`[wxt/storage] Migration ignored for "${key}", version downgrade detected (${currentVersion} -> ${targetVersion})`
|
|
90
229
|
);
|
|
91
230
|
}
|
|
231
|
+
const migrationsToRun = Array.from(
|
|
232
|
+
{ length: targetVersion - currentVersion },
|
|
233
|
+
(_, i) => currentVersion + i + 1
|
|
234
|
+
);
|
|
235
|
+
let migratedValue = value;
|
|
236
|
+
for (const migrateToVersion of migrationsToRun) {
|
|
237
|
+
migratedValue = await migrations?.[migrateToVersion]?.(migratedValue) ?? migratedValue;
|
|
238
|
+
}
|
|
239
|
+
await Promise.all([
|
|
240
|
+
// TODO: Optimize with `setItem`
|
|
241
|
+
setItem(driver, driverKey, migratedValue),
|
|
242
|
+
setMeta(driver, driverKey, { v: targetVersion })
|
|
243
|
+
]);
|
|
244
|
+
};
|
|
245
|
+
let _migrationsCompleted = runMigrations();
|
|
246
|
+
return {
|
|
247
|
+
_migrationsCompleted,
|
|
248
|
+
getValue: () => getItem(driver, driverKey, opts),
|
|
249
|
+
getMeta: () => getMeta(driver, driverKey),
|
|
250
|
+
setValue: (value) => setItem(driver, driverKey, value),
|
|
251
|
+
setMeta: (properties) => setMeta(driver, driverKey, properties),
|
|
252
|
+
removeValue: (opts2) => removeItem(driver, driverKey, opts2),
|
|
253
|
+
removeMeta: (properties) => removeMeta(driver, driverKey, properties),
|
|
254
|
+
watch: (cb) => watch(driver, driverKey, cb)
|
|
92
255
|
};
|
|
93
256
|
}
|
|
94
257
|
};
|
|
95
|
-
});
|
|
96
|
-
function createWebExtensionStorage() {
|
|
97
|
-
const storage2 = createStorage();
|
|
98
|
-
storage2.mount("local", webExtensionDriver({ storageArea: "local" }));
|
|
99
|
-
storage2.mount("session", webExtensionDriver({ storageArea: "session" }));
|
|
100
|
-
storage2.mount("sync", webExtensionDriver({ storageArea: "sync" }));
|
|
101
|
-
storage2.mount("managed", webExtensionDriver({ storageArea: "managed" }));
|
|
102
258
|
return storage2;
|
|
103
259
|
}
|
|
104
|
-
|
|
260
|
+
function createDriver(storageArea) {
|
|
261
|
+
const getStorageArea = () => {
|
|
262
|
+
if (browser.storage == null)
|
|
263
|
+
throw Error(
|
|
264
|
+
"You must add the 'storage' permission to your manifest to use 'wxt/storage'"
|
|
265
|
+
);
|
|
266
|
+
return browser.storage[storageArea];
|
|
267
|
+
};
|
|
268
|
+
const watchListeners = /* @__PURE__ */ new Set();
|
|
269
|
+
return {
|
|
270
|
+
getItem: async (key) => {
|
|
271
|
+
const res = await getStorageArea().get(key);
|
|
272
|
+
return res[key];
|
|
273
|
+
},
|
|
274
|
+
getItems: async (keys) => {
|
|
275
|
+
const result = await getStorageArea().get(keys);
|
|
276
|
+
return keys.map((key) => ({ key, value: result[key] ?? null }));
|
|
277
|
+
},
|
|
278
|
+
setItem: async (key, value) => {
|
|
279
|
+
if (value == null) {
|
|
280
|
+
await getStorageArea().remove(key);
|
|
281
|
+
} else {
|
|
282
|
+
await getStorageArea().set({ [key]: value });
|
|
283
|
+
}
|
|
284
|
+
},
|
|
285
|
+
setItems: async (values) => {
|
|
286
|
+
const map = values.reduce(
|
|
287
|
+
(map2, { key, value }) => {
|
|
288
|
+
map2[key] = value;
|
|
289
|
+
return map2;
|
|
290
|
+
},
|
|
291
|
+
{}
|
|
292
|
+
);
|
|
293
|
+
await getStorageArea().set(map);
|
|
294
|
+
},
|
|
295
|
+
removeItem: async (key) => {
|
|
296
|
+
await getStorageArea().remove(key);
|
|
297
|
+
},
|
|
298
|
+
removeItems: async (keys) => {
|
|
299
|
+
await getStorageArea().remove(keys);
|
|
300
|
+
},
|
|
301
|
+
snapshot: async () => {
|
|
302
|
+
return await getStorageArea().get();
|
|
303
|
+
},
|
|
304
|
+
restoreSnapshot: async (data) => {
|
|
305
|
+
await getStorageArea().set(data);
|
|
306
|
+
},
|
|
307
|
+
watch(key, cb) {
|
|
308
|
+
const listener = (changes) => {
|
|
309
|
+
const change = changes[key];
|
|
310
|
+
if (change == null)
|
|
311
|
+
return;
|
|
312
|
+
if (dequal(change.newValue, change.oldValue))
|
|
313
|
+
return;
|
|
314
|
+
cb(change.newValue ?? null, change.oldValue ?? null);
|
|
315
|
+
};
|
|
316
|
+
getStorageArea().onChanged.addListener(listener);
|
|
317
|
+
watchListeners.add(listener);
|
|
318
|
+
return () => {
|
|
319
|
+
getStorageArea().onChanged.removeListener(listener);
|
|
320
|
+
watchListeners.delete(listener);
|
|
321
|
+
};
|
|
322
|
+
},
|
|
323
|
+
unwatch() {
|
|
324
|
+
watchListeners.forEach((listener) => {
|
|
325
|
+
getStorageArea().onChanged.removeListener(listener);
|
|
326
|
+
});
|
|
327
|
+
watchListeners.clear();
|
|
328
|
+
}
|
|
329
|
+
};
|
|
330
|
+
}
|
|
105
331
|
export {
|
|
106
|
-
storage
|
|
107
|
-
webExtensionDriver
|
|
332
|
+
storage
|
|
108
333
|
};
|
package/dist/testing.cjs
CHANGED
|
@@ -441,7 +441,7 @@ function noopBackground() {
|
|
|
441
441
|
},
|
|
442
442
|
load(id) {
|
|
443
443
|
if (id === resolvedVirtualModuleId) {
|
|
444
|
-
return `import { defineBackground } from 'wxt/
|
|
444
|
+
return `import { defineBackground } from 'wxt/sandbox';
|
|
445
445
|
export default defineBackground(() => void 0)`;
|
|
446
446
|
}
|
|
447
447
|
}
|
package/dist/testing.js
CHANGED
package/package.json
CHANGED
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "wxt",
|
|
3
3
|
"type": "module",
|
|
4
|
-
"version": "0.
|
|
4
|
+
"version": "0.13.0",
|
|
5
5
|
"description": "Next gen framework for developing web extensions",
|
|
6
6
|
"engines": {
|
|
7
7
|
"node": ">=18",
|
|
@@ -89,6 +89,7 @@
|
|
|
89
89
|
"chokidar": "^3.5.3",
|
|
90
90
|
"consola": "^3.2.3",
|
|
91
91
|
"defu": "^6.1.3",
|
|
92
|
+
"dequal": "^2.0.3",
|
|
92
93
|
"esbuild": "^0.19.5",
|
|
93
94
|
"fast-glob": "^3.3.1",
|
|
94
95
|
"filesize": "^10.0.8",
|
|
@@ -107,7 +108,6 @@
|
|
|
107
108
|
"prompts": "^2.4.2",
|
|
108
109
|
"rollup-plugin-visualizer": "^5.9.2",
|
|
109
110
|
"unimport": "^3.4.0",
|
|
110
|
-
"unstorage": "^1.9.0",
|
|
111
111
|
"vite": "^5.0.0",
|
|
112
112
|
"web-ext-run": "^0.1.0",
|
|
113
113
|
"webextension-polyfill": "^0.10.0",
|