Merge pull request #9951 from alexstrat/fix-chrome-storage

Fix chrome storage access scope
This commit is contained in:
Cheng Zhao 2017-07-24 14:01:15 +09:00 committed by GitHub
commit 25f168cecb
2 changed files with 103 additions and 60 deletions

View file

@ -173,7 +173,7 @@ exports.injectTo = function (extensionId, isBackgroundPage, context) {
onMessage: chrome.runtime.onMessage
}
chrome.storage = require('./extensions/storage')
chrome.storage = require('./extensions/storage').setup(extensionId)
chrome.pageAction = {
show () {},

View file

@ -1,28 +1,74 @@
const getStorage = (storageType) => {
const data = window.localStorage.getItem(`__chrome.storage.${storageType}__`)
if (data != null) {
return JSON.parse(data)
} else {
return {}
const fs = require('fs')
const path = require('path')
const { remote } = require('electron')
const { app } = remote
const getChromeStoragePath = (storageType, extensionId) => {
return path.join(
app.getPath('userData'), `/Chrome Storage/${extensionId}-${storageType}.json`)
}
const mkdirp = (dir, callback) => {
fs.mkdir(dir, (error) => {
if (error && error.code === 'ENOENT') {
mkdirp(path.dirname(dir), (error) => {
if (!error) {
mkdirp(dir, callback)
}
})
} else if (error && error.code === 'EEXIST') {
callback(null)
} else {
callback(error)
}
}
const setStorage = (storageType, storage) => {
const json = JSON.stringify(storage)
window.localStorage.setItem(`__chrome.storage.${storageType}__`, json)
}
const scheduleCallback = (items, callback) => {
setTimeout(function () {
callback(items)
})
}
const getStorageManager = (storageType) => {
const readChromeStorageFile = (storageType, extensionId, cb) => {
const filePath = getChromeStoragePath(storageType, extensionId)
fs.readFile(filePath, 'utf8', (err, data) => {
if (err && err.code === 'ENOENT') {
return cb(null, null)
}
cb(err, data)
})
}
const writeChromeStorageFile = (storageType, extensionId, data, cb) => {
const filePath = getChromeStoragePath(storageType, extensionId)
mkdirp(path.dirname(filePath), err => {
if (err) { /* we just ignore the errors of mkdir or mkdirp */ }
fs.writeFile(filePath, data, cb)
})
}
const getStorage = (storageType, extensionId, cb) => {
readChromeStorageFile(storageType, extensionId, (err, data) => {
if (err) throw err
if (!cb) throw new TypeError('No callback provided')
if (data !== null) {
cb(JSON.parse(data))
} else {
cb({})
}
})
}
const setStorage = (storageType, extensionId, storage, cb) => {
const json = JSON.stringify(storage)
writeChromeStorageFile(storageType, extensionId, json, err => {
if (err) throw err
if (cb) cb()
})
}
const getStorageManager = (storageType, extensionId) => {
return {
get (keys, callback) {
const storage = getStorage(storageType)
if (keys == null) return scheduleCallback(storage, callback)
getStorage(storageType, extensionId, storage => {
if (keys == null) return callback(storage)
let defaults = {}
switch (typeof keys) {
@ -36,7 +82,7 @@ const getStorageManager = (storageType) => {
}
break
}
if (keys.length === 0) return scheduleCallback({}, callback)
if (keys.length === 0) return callback({})
let items = {}
keys.forEach(function (key) {
@ -44,24 +90,22 @@ const getStorageManager = (storageType) => {
if (value == null) value = defaults[key]
items[key] = value
})
scheduleCallback(items, callback)
callback(items)
})
},
set (items, callback) {
const storage = getStorage(storageType)
getStorage(storageType, extensionId, storage => {
Object.keys(items).forEach(function (name) {
storage[name] = items[name]
})
setStorage(storageType, storage)
setTimeout(callback)
setStorage(storageType, extensionId, storage, callback)
})
},
remove (keys, callback) {
const storage = getStorage(storageType)
getStorage(storageType, extensionId, storage => {
if (!Array.isArray(keys)) {
keys = [keys]
}
@ -69,20 +113,19 @@ const getStorageManager = (storageType) => {
delete storage[key]
})
setStorage(storageType, storage)
setTimeout(callback)
setStorage(storageType, extensionId, storage, callback)
})
},
clear (callback) {
setStorage(storageType, {})
setTimeout(callback)
setStorage(storageType, extensionId, {}, callback)
}
}
}
module.exports = {
sync: getStorageManager('sync'),
local: getStorageManager('local')
setup: extensionId => ({
sync: getStorageManager('sync', extensionId),
local: getStorageManager('local', extensionId)
})
}