diff --git a/src/lib/extensions.js b/src/lib/extensions.js index 827847f5..1370214e 100644 --- a/src/lib/extensions.js +++ b/src/lib/extensions.js @@ -570,4 +570,13 @@ export default [ isGitHub: true, notes: "Gallery banner by Dillon", }, + { + name: "Ultimate DB", + description: "Store data online with two methods", + code: "Somroti/UltimateDB.js", + banner: "Somroti/UltimateDB.png", + creator: "Somroti and Logise", + cratorAlias: "Somroti", + notes: "The banner was made by Logise" + }, ]; diff --git a/static/extensions/Somroti/UltimateDB.js b/static/extensions/Somroti/UltimateDB.js new file mode 100644 index 00000000..24f91e4f --- /dev/null +++ b/static/extensions/Somroti/UltimateDB.js @@ -0,0 +1,318 @@ +(function(Scratch) { + 'use strict'; + if (!Scratch.extensions.unsandboxed) throw new Error("UltimateDB must be used without sandbox."); + // This extension is made in collaboration with Logise + const menuIconURI = "" + + class SomrotiUltimateDB { + constructor() { + this.somrotiAPI = "https://bdd.somroti-yt.workers.dev"; + this.firebaseAPI = "https://guessthepin-2fe64-default-rtdb.europe-west1.firebasedatabase.app"; + } + + getInfo() { + return { + id: "SomrotiUltimateDB", + name: "Ultimate DB", + color1: "#7b3ff2", + menuIconURI: menuIconURI, + blocks: [ + { blockType: Scratch.BlockType.LABEL, text: "SomrotiDB" }, + { + opcode: "getSomrotiKey", + blockType: Scratch.BlockType.REPORTER, + text: "get [KEY] from SomrotiDB", + arguments: { KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } } + }, + { + opcode: "setSomrotiKey", + blockType: Scratch.BlockType.COMMAND, + text: "set [KEY] to [VALUE] in SomrotiDB", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" } + } + }, + { + opcode: "deleteSomrotiKey", + blockType: Scratch.BlockType.COMMAND, + text: "delete [KEY] in SomrotiDB", + arguments: { KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } } + }, + { + opcode: "setProtectedSomrotiKey", + blockType: Scratch.BlockType.COMMAND, + text: "set protected key [KEY] to [VALUE] with password [PASSWORD] in SomrotiDB", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + }, + { + opcode: "getProtectedSomrotiKey", + blockType: Scratch.BlockType.REPORTER, + text: "get protected key [KEY] with password [PASSWORD] in SomrotiDB", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + }, + { + opcode: "checkSomrotiPassword", + blockType: Scratch.BlockType.BOOLEAN, + text: "password [PASSWORD] is correct for [KEY] in SomrotiDB?", + arguments: { + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" }, + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } + } + }, + + { blockType: Scratch.BlockType.LABEL, text: "FirebaseDB" }, + { + opcode: "setFirebaseURL", + blockType: Scratch.BlockType.COMMAND, + text: "set Firebase URL to [URL]", + arguments: { + URL: { type: Scratch.ArgumentType.STRING, defaultValue: this.firebaseAPI } + } + }, + { + opcode: "setFirebaseKey", + blockType: Scratch.BlockType.COMMAND, + text: "set key [KEY] to value [VALUE] in Firebase", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" } + } + }, + { + opcode: "getFirebaseKey", + blockType: Scratch.BlockType.REPORTER, + text: "get key [KEY] from Firebase", + arguments: { KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } } + }, + { + opcode: "setFirebaseKeyWithPassword", + blockType: Scratch.BlockType.COMMAND, + text: "set key [KEY] to [VALUE] with password [PASSWORD] in Firebase", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + }, + { + opcode: "getFirebaseKeyWithPassword", + blockType: Scratch.BlockType.REPORTER, + text: "get key [KEY] with password [PASSWORD] from Firebase", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + }, + { + opcode: "checkFirebasePassword", + blockType: Scratch.BlockType.BOOLEAN, + text: "check if [PASSWORD] is valid for [KEY] in Firebase", + arguments: { + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" }, + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" } + } + }, + { + opcode: "setFirebaseViewableKey", + blockType: Scratch.BlockType.COMMAND, + text: "set viewable key [KEY] to [VALUE] with password [PASSWORD] in Firebase", + arguments: { + KEY: { type: Scratch.ArgumentType.STRING, defaultValue: "key" }, + VALUE: { type: Scratch.ArgumentType.STRING, defaultValue: "value" }, + PASSWORD: { type: Scratch.ArgumentType.STRING, defaultValue: "password" } + } + } + ] + }; + } + + // --- Somroti --- + async getSomrotiKey({ KEY }) { + try { + const r = await fetch(`${this.somrotiAPI}/get/${encodeURIComponent(KEY)}`); + return r.ok ? await r.text() : ""; + } catch (e) { + console.error(e); + return ""; + } + } + + setSomrotiKey({ KEY, VALUE }) { + return fetch(`${this.somrotiAPI}/set/${encodeURIComponent(KEY)}`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ value: VALUE }) + }); + } + + deleteSomrotiKey({ KEY }) { + return fetch(`${this.somrotiAPI}/delete/${encodeURIComponent(KEY)}`, { + method: "DELETE" + }); + } + + setProtectedSomrotiKey({ KEY, VALUE, PASSWORD }) { + return fetch(`${this.somrotiAPI}/set_protected/${encodeURIComponent(KEY)}`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ value: VALUE, password: PASSWORD }) + }); + } + + async getProtectedSomrotiKey({ KEY, PASSWORD }) { + try { + const r = await fetch(`${this.somrotiAPI}/get_protected/${encodeURIComponent(KEY)}`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ password: PASSWORD }) + }); + return r.ok ? await r.text() : ""; + } catch (e) { + console.error(e); + return ""; + } + } + + async checkSomrotiPassword({ PASSWORD, KEY }) { + try { + const r = await fetch(`${this.somrotiAPI}/check_password/${encodeURIComponent(KEY)}`, { + method: "POST", + headers: { "Content-Type": "application/json" }, + body: JSON.stringify({ password: PASSWORD }) + }); + const json = await r.json(); + return !!json.match; + } catch (e) { + console.error(e); + return false; + } + } + + // --- Firebase --- + setFirebaseURL({ URL }) { + this.firebaseAPI = URL; + } + + async setFirebaseKey({ KEY, VALUE }) { + await fetch(`${this.firebaseAPI}/pin/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(VALUE) + }); + } + + async getFirebaseKey({ KEY }) { + const res = await fetch(`${this.firebaseAPI}/pin/${encodeURIComponent(KEY)}.json`); + const data = await res.json(); + return data ?? ""; + } + + async deriveKey(password, salt) { + const enc = new TextEncoder(); + const keyMaterial = await crypto.subtle.importKey( + "raw", enc.encode(password), "PBKDF2", false, ["deriveKey"] + ); + return crypto.subtle.deriveKey( + { name: "PBKDF2", salt, iterations: 100000, hash: "SHA-256" }, + keyMaterial, { name: "AES-GCM", length: 256 }, false, ["encrypt", "decrypt"] + ); + } + + async setFirebaseKeyWithPassword({ KEY, VALUE, PASSWORD }) { + const enc = new TextEncoder(); + const iv = crypto.getRandomValues(new Uint8Array(12)); + const salt = crypto.getRandomValues(new Uint8Array(16)); + const key = await this.deriveKey(PASSWORD, salt); + const encrypted = await crypto.subtle.encrypt({ name: "AES-GCM", iv }, key, enc.encode(VALUE)); + const payload = { + iv: Array.from(iv), + salt: Array.from(salt), + data: Array.from(new Uint8Array(encrypted)) + }; + await fetch(`${this.firebaseAPI}/cypher/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(payload) + }); + } + + async getFirebaseKeyWithPassword({ KEY, PASSWORD }) { + const res = await fetch(`${this.firebaseAPI}/cypher/${encodeURIComponent(KEY)}.json`); + const encryptedPackage = await res.json(); + if (!encryptedPackage?.data) return ""; + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + const decrypted = await crypto.subtle.decrypt({ name: "AES-GCM", iv }, key, data); + return new TextDecoder().decode(decrypted); + } catch (e) { + console.error(e); + return ""; + } + } + + async checkFirebasePassword({ KEY, PASSWORD }) { + const res = await fetch(`${this.firebaseAPI}/cypher/${encodeURIComponent(KEY)}.json`); + const encryptedPackage = await res.json(); + if (!encryptedPackage?.data) return false; + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + await crypto.subtle.decrypt({ name: "AES-GCM", iv }, key, data); + return true; + } catch (e) { + return false; + } + } + + async setFirebaseViewableKey({ KEY, VALUE, PASSWORD }) { + const res = await fetch(`${this.firebaseAPI}/cypher/${encodeURIComponent(KEY)}.json`); + const encryptedPackage = await res.json(); + + if (encryptedPackage?.iv && encryptedPackage?.salt && encryptedPackage?.data) { + try { + const iv = new Uint8Array(encryptedPackage.iv); + const salt = new Uint8Array(encryptedPackage.salt); + const data = new Uint8Array(encryptedPackage.data); + const key = await this.deriveKey(PASSWORD, salt); + await crypto.subtle.decrypt({ name: "AES-GCM", iv }, key, data); + } catch { + return; + } + } + + const enc = new TextEncoder(); + const newIv = crypto.getRandomValues(new Uint8Array(12)); + const newSalt = crypto.getRandomValues(new Uint8Array(16)); + const newKey = await this.deriveKey(PASSWORD, newSalt); + const encrypted = await crypto.subtle.encrypt({ name: "AES-GCM", iv: newIv }, newKey, enc.encode(VALUE)); + const packageData = { + iv: Array.from(newIv), + salt: Array.from(newSalt), + data: Array.from(new Uint8Array(encrypted)) + }; + + await fetch(`${this.firebaseAPI}/cypher/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(packageData) + }); + + await fetch(`${this.firebaseAPI}/pin/${encodeURIComponent(KEY)}.json`, { + method: "PUT", + body: JSON.stringify(VALUE) + }); + } + } + + Scratch.extensions.register(new SomrotiUltimateDB()); +})(Scratch); diff --git a/static/images/Somroti/UltimateDB.png b/static/images/Somroti/UltimateDB.png new file mode 100644 index 00000000..c60f9e23 Binary files /dev/null and b/static/images/Somroti/UltimateDB.png differ