FuriKani/src/background.js

166 lines
5.5 KiB
JavaScript
Raw Normal View History

2022-03-21 13:03:34 +00:00
chrome.storage.local.get("token", (data) => {
if(data.token !== undefined)
return
chrome.storage.local.set({
"enabled": true,
"enabledVocab": true,
"enabledKanji": true
})
})
2022-03-21 12:34:22 +00:00
// Query a WaniKani API endpoint with the given token
2022-03-20 00:04:30 +00:00
const query = (token, url) =>
new Promise(async (resolve, reject) => {
2022-03-21 12:34:22 +00:00
// Create header with the authorization token
2022-03-20 00:04:30 +00:00
const requestHeaders = new Headers()
requestHeaders.append("Authorization", "Bearer " + token)
2022-03-21 12:34:22 +00:00
// Construct request
2022-03-20 00:04:30 +00:00
const endpoint = new Request(
url,
{
method: "GET",
headers: requestHeaders
}
)
2022-03-21 12:34:22 +00:00
// Fetch the response. If it returns HTTP 200, resolve the promise,
// otherwise reject it
2022-03-20 00:04:30 +00:00
var result = await fetch(endpoint)
if(!result.ok)
{
var message = await result.json()
return reject(message.error + " (" + message.code + ")")
}
return resolve(result.json())
})
2022-03-21 12:34:22 +00:00
// Update the local cache with new data
2022-03-20 00:04:30 +00:00
const updateCache = async (token, oldLevel, newLevel) => {
2022-03-20 00:23:01 +00:00
// If oldLevel is undefined, then a sync has never been performed (first time user)
2022-03-20 00:04:30 +00:00
if(oldLevel === undefined)
oldLevel = 1
2022-03-20 00:23:01 +00:00
// Create a list of levels that need to be synced
// If the new level is bigger than the old level (user leveled up)
// --> Just fetch the new level(s)
// If the new level is smaller than the old level (user reset)
// --> Fetch everything up until the new level
2022-03-20 00:04:30 +00:00
var levelArray = []
2022-03-20 00:23:01 +00:00
var startLevel = newLevel > oldLevel ? oldLevel : 1
for(var i = startLevel; i < newLevel; i++)
2022-03-20 00:04:30 +00:00
levelArray.push(i)
2022-03-20 00:23:01 +00:00
// Turn the array of levels into a comma separated list
2022-03-20 00:04:30 +00:00
var levelURLString = levelArray.join(",")
2022-03-21 12:53:22 +00:00
// Data buffers
var vocabulary = new Set();
var kanji = new Set();
// If the old level is less than the new level add the old data to the new data
if(oldLevel < newLevel)
{
await chrome.storage.local.get(["vocabulary", "kanji"], (data) => {
if(data.vocabulary !== undefined && data.kanji !== undefined)
{
data.vocabulary.forEach(vocabulary.add, vocabulary)
data.kanji.forEach(kanji.add, kanji)
}
})
}
// API endpoint
2022-03-20 00:04:30 +00:00
var url = "https://api.wanikani.com/v2/subjects?types=vocabulary&levels=" + levelURLString
2022-03-20 00:23:01 +00:00
// WaniKani only sends 1000 elements in one response and then provides us with a link to the
// next "page" of the data. We need to loop until the next page is null
2022-03-20 00:04:30 +00:00
do
{
2022-03-20 00:23:01 +00:00
// Query API, extract all the important info and then update the URL to the next page
2022-03-20 00:04:30 +00:00
var response = await query(token, url).catch(reason => console.error("WaniKani API request failed: " + reason))
if(response === undefined)
break
2022-03-20 00:23:01 +00:00
2022-03-20 00:04:30 +00:00
for(let i in response.data)
2022-03-21 12:53:22 +00:00
vocabulary.add(response.data[i].data.characters)
2022-03-20 00:04:30 +00:00
url = response.pages.next_url
} while(url !== null)
2022-03-20 00:23:01 +00:00
// Extract Kanji as well
2022-03-20 00:04:30 +00:00
var url = "https://api.wanikani.com/v2/subjects?types=kanji&levels=" + levelURLString
2022-03-20 00:23:01 +00:00
2022-03-20 00:04:30 +00:00
do
{
var response = await query(token, url).catch(reason => console.error("WaniKani API request failed: " + reason))
if(response === undefined)
break
for(let i in response.data)
2022-03-21 12:53:22 +00:00
kanji.add(response.data[i].data.characters)
2022-03-20 00:04:30 +00:00
url = response.pages.next_url
} while(url !== null)
2022-03-20 00:23:01 +00:00
// Cache the data
2022-03-20 00:04:30 +00:00
chrome.storage.local.set({
2022-03-21 12:53:22 +00:00
"vocabulary": [...vocabulary],
"kanji": [...kanji],
2022-03-20 00:23:01 +00:00
"level": newLevel
2022-03-20 00:04:30 +00:00
})
}
2022-03-20 00:23:01 +00:00
// Synchronizes local data with wanikanis data
2022-03-20 00:04:30 +00:00
const sync = () =>
new Promise((resolve, reject) => {
chrome.storage.local.get(["level", "token"], async (data) => {
2022-03-20 00:23:01 +00:00
// See if wanikani token is well formed
2022-03-20 00:04:30 +00:00
const apiTokenPattern = /[0-9a-f]{8}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{4}-[0-9a-f]{12}/
if(!apiTokenPattern.test(data.token))
return reject("Please set a valid WaniKani API Token")
2022-03-20 00:23:01 +00:00
// Get user info
2022-03-20 14:55:31 +00:00
var user = await query(data.token, "https://api.wanikani.com/v2/user").catch(reason => reject(reason))
2022-03-20 00:04:30 +00:00
if(user === undefined)
return
2022-03-20 00:23:01 +00:00
// Get user level
2022-03-20 00:04:30 +00:00
level = user.data.level
2022-03-20 00:23:01 +00:00
// if users level is larger than max allowed level, abort
2022-03-20 00:04:30 +00:00
if(level > user.data.subscription.max_level_granted)
2022-03-20 00:23:01 +00:00
{
chrome.storage.local.set({"validUserLevel": false})
2022-03-20 00:04:30 +00:00
return reject("User account level exceeds account level limit")
2022-03-20 00:23:01 +00:00
}
2022-03-20 00:04:30 +00:00
2022-03-30 14:35:14 +00:00
// If the wanikani level differs from the local level, update the cache
2022-03-20 00:04:30 +00:00
if(level !== data.level)
2022-03-30 14:37:47 +00:00
await updateCache(data.token, data.level, level)
2022-03-20 00:04:30 +00:00
2022-03-20 00:23:01 +00:00
chrome.storage.local.set({"validUserLevel": true})
2022-03-20 00:04:30 +00:00
resolve("Successfully synchronized data!")
})
})
2022-03-20 00:23:01 +00:00
// Message listener for the Sync button
2022-03-20 00:04:30 +00:00
chrome.runtime.onMessage.addListener((data, sender, sendResponse) => {
if(data.type === "sync")
sync()
.then(value => {
sendResponse({success: true})
})
.catch(reason => {
sendResponse({success: false, error: reason})
})
return true;
})
2022-03-20 00:23:01 +00:00
// At browser start, sync with wanikani
2022-03-30 14:35:14 +00:00
chrome.runtime.onStartup.addListener(() => {
sync().then(value => console.log(value)).catch(reason => {})
})