Update to 1.20.1 (#126)
This commit is contained in:
3
server.bat
Normal file
3
server.bat
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
npm install
|
||||||
|
npm run build
|
||||||
|
npm run runserver
|
||||||
@@ -2,18 +2,26 @@ const express = require('express');
|
|||||||
const app = express();
|
const app = express();
|
||||||
const path = require('path');
|
const path = require('path');
|
||||||
const htmlparser = require('node-html-parser');
|
const htmlparser = require('node-html-parser');
|
||||||
const { existsSync, mkdirSync } = require('fs');
|
const { existsSync, mkdirSync, readFileSync, writeFileSync } = require('fs');
|
||||||
const bodyParser = require('body-parser');
|
const bodyParser = require('body-parser');
|
||||||
const fs = require('fs/promises')
|
const fs = require('fs/promises')
|
||||||
|
|
||||||
app.use(express.static(path.join(process.cwd(), 'dist'), {index: false}));
|
app.use(express.static(path.join(process.cwd(), 'dist'), {index: false}));
|
||||||
app.use(bodyParser.json({ limit: 100000000 }));
|
app.use(bodyParser.json({ limit: 100000000 }));
|
||||||
|
|
||||||
|
|
||||||
|
let password = ''
|
||||||
|
|
||||||
const savePath = path.join(process.cwd(), "save")
|
const savePath = path.join(process.cwd(), "save")
|
||||||
if(!existsSync(savePath)){
|
if(!existsSync(savePath)){
|
||||||
mkdirSync(savePath)
|
mkdirSync(savePath)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const passwordPath = path.join(process.cwd(), 'save', '__password')
|
||||||
|
if(existsSync(passwordPath)){
|
||||||
|
password = readFileSync(passwordPath, 'utf-8')
|
||||||
|
}
|
||||||
|
|
||||||
app.get('/', async (req, res, next) => {
|
app.get('/', async (req, res, next) => {
|
||||||
console.log("connected")
|
console.log("connected")
|
||||||
try {
|
try {
|
||||||
@@ -62,9 +70,38 @@ app.post('/proxy', async (req, res, next) => {
|
|||||||
res.send(originalBody);
|
res.send(originalBody);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
app.get('/api/password', async(req, res)=> {
|
||||||
|
if(password === ''){
|
||||||
|
res.send({status: 'unset'})
|
||||||
|
}
|
||||||
|
else if(req.headers['risu-auth'] === password){
|
||||||
|
res.send({status:'correct'})
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
res.send({status:'incorrect'})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
|
||||||
|
app.post('/api/set_password', async (req, res) => {
|
||||||
|
if(password === ''){
|
||||||
|
password = req.body.password
|
||||||
|
writeFileSync(passwordPath, password, 'utf-8')
|
||||||
|
}
|
||||||
|
res.status(400).send("already set")
|
||||||
|
})
|
||||||
|
|
||||||
app.get('/api/read', async (req, res, next) => {
|
app.get('/api/read', async (req, res, next) => {
|
||||||
|
if(req.headers['risu-auth'].trim() !== password.trim()){
|
||||||
|
console.log('incorrect')
|
||||||
|
res.status(400).send({
|
||||||
|
error:'Password Incorrect'
|
||||||
|
});
|
||||||
|
return
|
||||||
|
}
|
||||||
const filePath = req.headers['file-path'];
|
const filePath = req.headers['file-path'];
|
||||||
if (!filePath) {
|
if (!filePath) {
|
||||||
|
console.log('no path')
|
||||||
res.status(400).send({
|
res.status(400).send({
|
||||||
error:'File path required'
|
error:'File path required'
|
||||||
});
|
});
|
||||||
@@ -91,6 +128,13 @@ app.get('/api/read', async (req, res, next) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
app.get('/api/remove', async (req, res, next) => {
|
app.get('/api/remove', async (req, res, next) => {
|
||||||
|
if(req.headers['risu-auth'].trim() !== password.trim()){
|
||||||
|
console.log('incorrect')
|
||||||
|
res.status(400).send({
|
||||||
|
error:'Password Incorrect'
|
||||||
|
});
|
||||||
|
return
|
||||||
|
}
|
||||||
const filePath = req.headers['file-path'];
|
const filePath = req.headers['file-path'];
|
||||||
if (!filePath) {
|
if (!filePath) {
|
||||||
res.status(400).send({
|
res.status(400).send({
|
||||||
@@ -110,6 +154,13 @@ app.get('/api/remove', async (req, res, next) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
app.get('/api/list', async (req, res, next) => {
|
app.get('/api/list', async (req, res, next) => {
|
||||||
|
if(req.headers['risu-auth'].trim() !== password.trim()){
|
||||||
|
console.log('incorrect')
|
||||||
|
res.status(400).send({
|
||||||
|
error:'Password Incorrect'
|
||||||
|
});
|
||||||
|
return
|
||||||
|
}
|
||||||
try {
|
try {
|
||||||
const data = (await fs.readdir(path.join(savePath))).map((v) => {
|
const data = (await fs.readdir(path.join(savePath))).map((v) => {
|
||||||
return Buffer.from(v, 'hex').toString('utf-8')
|
return Buffer.from(v, 'hex').toString('utf-8')
|
||||||
@@ -124,6 +175,13 @@ app.get('/api/list', async (req, res, next) => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
app.post('/api/write', async (req, res, next) => {
|
app.post('/api/write', async (req, res, next) => {
|
||||||
|
if(req.headers['risu-auth'].trim() !== password.trim()){
|
||||||
|
console.log('incorrect')
|
||||||
|
res.status(400).send({
|
||||||
|
error:'Password Incorrect'
|
||||||
|
});
|
||||||
|
return
|
||||||
|
}
|
||||||
const filePath = req.headers['file-path'];
|
const filePath = req.headers['file-path'];
|
||||||
const fileContent = Buffer.from(req.body.content, 'base64');
|
const fileContent = Buffer.from(req.body.content, 'base64');
|
||||||
if (!filePath || !fileContent) {
|
if (!filePath || !fileContent) {
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
},
|
},
|
||||||
"package": {
|
"package": {
|
||||||
"productName": "RisuAI",
|
"productName": "RisuAI",
|
||||||
"version": "1.20.0"
|
"version": "1.20.1"
|
||||||
},
|
},
|
||||||
"tauri": {
|
"tauri": {
|
||||||
"allowlist": {
|
"allowlist": {
|
||||||
|
|||||||
@@ -280,5 +280,7 @@ export const languageEnglish = {
|
|||||||
globalRegexScript: "Global Regex",
|
globalRegexScript: "Global Regex",
|
||||||
accessibility: "Accessibility",
|
accessibility: "Accessibility",
|
||||||
sendWithEnter: "Send with Enter Key",
|
sendWithEnter: "Send with Enter Key",
|
||||||
clickToEdit: "Click Text to Edit"
|
clickToEdit: "Click Text to Edit",
|
||||||
|
setNodePassword: "Set your password to security",
|
||||||
|
inputNodePassword: "Input your password. if you can't remember, remove save/__password.txt in your server files and restart the server."
|
||||||
}
|
}
|
||||||
@@ -14,10 +14,15 @@
|
|||||||
btn.focus()
|
btn.focus()
|
||||||
}
|
}
|
||||||
if($alertStore.type !== 'input'){
|
if($alertStore.type !== 'input'){
|
||||||
|
console.log('reset input')
|
||||||
input = ''
|
input = ''
|
||||||
}
|
}
|
||||||
|
|
||||||
})()
|
})()
|
||||||
|
|
||||||
|
alertStore.subscribe(() => {
|
||||||
|
console.log('alup')
|
||||||
|
})
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if $alertStore.type !== 'none' && $alertStore.type !== 'toast'}
|
{#if $alertStore.type !== 'none' && $alertStore.type !== 'toast'}
|
||||||
@@ -73,11 +78,12 @@
|
|||||||
})
|
})
|
||||||
}}>OK</button>
|
}}>OK</button>
|
||||||
{:else if $alertStore.type === 'input'}
|
{:else if $alertStore.type === 'input'}
|
||||||
<input class="text-neutral-200 mt-2 p-2 bg-transparent input-text focus:bg-selected" bind:value={input}>
|
<input class="text-neutral-200 mt-2 p-2 bg-transparent input-text focus:bg-selected" value="" id="alert-input">
|
||||||
<button bind:this={btn} class="mt-4 border-borderc bg-transparent outline-none border-solid border-1 p-2 text-lg text-neutral-200 hover:bg-green-500 transition-colors focus:border-3" on:click={() => {
|
<button bind:this={btn} class="mt-4 border-borderc bg-transparent outline-none border-solid border-1 p-2 text-lg text-neutral-200 hover:bg-green-500 transition-colors focus:border-3" on:click={() => {
|
||||||
alertStore.set({
|
alertStore.set({
|
||||||
type: 'none',
|
type: 'none',
|
||||||
msg: input
|
//@ts-ignore
|
||||||
|
msg: document.querySelector('#alert-input')?.value
|
||||||
})
|
})
|
||||||
}}>OK</button>
|
}}>OK</button>
|
||||||
{:else if $alertStore.type === 'selectChar'}
|
{:else if $alertStore.type === 'selectChar'}
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
import { language } from "src/lang";
|
import { language } from "src/lang";
|
||||||
import { alertConfirm } from "src/ts/alert";
|
import { alertConfirm } from "src/ts/alert";
|
||||||
import { checkDriver } from "src/ts/drive/drive";
|
import { checkDriver } from "src/ts/drive/drive";
|
||||||
import { isTauri } from "src/ts/storage/globalApi";
|
import { isNodeServer, isTauri } from "src/ts/storage/globalApi";
|
||||||
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -11,7 +11,7 @@
|
|||||||
on:click={async () => {
|
on:click={async () => {
|
||||||
if(await alertConfirm(language.backupConfirm)){
|
if(await alertConfirm(language.backupConfirm)){
|
||||||
localStorage.setItem('backup', 'save')
|
localStorage.setItem('backup', 'save')
|
||||||
if(isTauri){
|
if(isTauri || isNodeServer){
|
||||||
checkDriver('savetauri')
|
checkDriver('savetauri')
|
||||||
}
|
}
|
||||||
else{
|
else{
|
||||||
@@ -27,7 +27,7 @@
|
|||||||
on:click={async () => {
|
on:click={async () => {
|
||||||
if((await alertConfirm(language.backupLoadConfirm)) && (await alertConfirm(language.backupLoadConfirm2))){
|
if((await alertConfirm(language.backupLoadConfirm)) && (await alertConfirm(language.backupLoadConfirm2))){
|
||||||
localStorage.setItem('backup', 'load')
|
localStorage.setItem('backup', 'load')
|
||||||
if(isTauri){
|
if(isTauri || isNodeServer){
|
||||||
checkDriver('loadtauri')
|
checkDriver('loadtauri')
|
||||||
}
|
}
|
||||||
else{
|
else{
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { get } from "svelte/store";
|
import { get } from "svelte/store";
|
||||||
import { alertError, alertInput, alertNormal, alertSelect, alertStore } from "../alert";
|
import { alertError, alertInput, alertNormal, alertSelect, alertStore } from "../alert";
|
||||||
import { DataBase, setDatabase, type Database } from "../storage/database";
|
import { DataBase, setDatabase, type Database } from "../storage/database";
|
||||||
import { forageStorage, getUnpargeables, isTauri } from "../storage/globalApi";
|
import { forageStorage, getUnpargeables, isNodeServer, isTauri, openURL } from "../storage/globalApi";
|
||||||
import pako from "pako";
|
import pako from "pako";
|
||||||
import { BaseDirectory, exists, readBinaryFile, readDir, writeBinaryFile } from "@tauri-apps/api/fs";
|
import { BaseDirectory, exists, readBinaryFile, readDir, writeBinaryFile } from "@tauri-apps/api/fs";
|
||||||
import { language } from "../../lang";
|
import { language } from "../../lang";
|
||||||
@@ -10,7 +10,7 @@ import { open } from '@tauri-apps/api/shell';
|
|||||||
|
|
||||||
export async function checkDriver(type:'save'|'load'|'loadtauri'|'savetauri'|'reftoken'){
|
export async function checkDriver(type:'save'|'load'|'loadtauri'|'savetauri'|'reftoken'){
|
||||||
const CLIENT_ID = '580075990041-l26k2d3c0nemmqiu3d3aag01npfrkn76.apps.googleusercontent.com';
|
const CLIENT_ID = '580075990041-l26k2d3c0nemmqiu3d3aag01npfrkn76.apps.googleusercontent.com';
|
||||||
const REDIRECT_URI = isTauri ? "https://risuai.xyz/" : `https://${location.host}/`
|
const REDIRECT_URI = (isTauri || isNodeServer) ? "https://risuai.xyz/" : `https://${location.host}/`
|
||||||
const SCOPE = 'https://www.googleapis.com/auth/drive.file https://www.googleapis.com/auth/drive.appdata';
|
const SCOPE = 'https://www.googleapis.com/auth/drive.file https://www.googleapis.com/auth/drive.appdata';
|
||||||
const encodedRedirectUri = encodeURIComponent(REDIRECT_URI);
|
const encodedRedirectUri = encodeURIComponent(REDIRECT_URI);
|
||||||
const authorizationUrl = `https://accounts.google.com/o/oauth2/auth?client_id=${CLIENT_ID}&redirect_uri=${encodedRedirectUri}&scope=${SCOPE}&response_type=code&state=${type}`;
|
const authorizationUrl = `https://accounts.google.com/o/oauth2/auth?client_id=${CLIENT_ID}&redirect_uri=${encodedRedirectUri}&scope=${SCOPE}&response_type=code&state=${type}`;
|
||||||
@@ -18,7 +18,7 @@ export async function checkDriver(type:'save'|'load'|'loadtauri'|'savetauri'|'re
|
|||||||
|
|
||||||
if(type === 'reftoken'){
|
if(type === 'reftoken'){
|
||||||
const authorizationUrl = `https://accounts.google.com/o/oauth2/auth?client_id=${CLIENT_ID}&redirect_uri=${encodedRedirectUri}&scope=${SCOPE}&response_type=code&state=${"accesstauri"}&access_type=offline&prompt=consent`;
|
const authorizationUrl = `https://accounts.google.com/o/oauth2/auth?client_id=${CLIENT_ID}&redirect_uri=${encodedRedirectUri}&scope=${SCOPE}&response_type=code&state=${"accesstauri"}&access_type=offline&prompt=consent`;
|
||||||
open(authorizationUrl)
|
openURL(authorizationUrl)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -28,7 +28,12 @@ export async function checkDriver(type:'save'|'load'|'loadtauri'|'savetauri'|'re
|
|||||||
else{
|
else{
|
||||||
|
|
||||||
try {
|
try {
|
||||||
open(authorizationUrl)
|
if(isTauri){
|
||||||
|
openURL(authorizationUrl)
|
||||||
|
}
|
||||||
|
else{
|
||||||
|
window.open(authorizationUrl)
|
||||||
|
}
|
||||||
let code = await alertInput(language.pasteAuthCode)
|
let code = await alertInput(language.pasteAuthCode)
|
||||||
if(code.includes(' ')){
|
if(code.includes(' ')){
|
||||||
code = code.substring(code.lastIndexOf(' ')).trim()
|
code = code.substring(code.lastIndexOf(' ')).trim()
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ import { cloneDeep } from 'lodash';
|
|||||||
|
|
||||||
export const DataBase = writable({} as any as Database)
|
export const DataBase = writable({} as any as Database)
|
||||||
export const loadedStore = writable(false)
|
export const loadedStore = writable(false)
|
||||||
export let appVer = '1.20.0'
|
export let appVer = '1.20.1'
|
||||||
|
|
||||||
|
|
||||||
export function setDatabase(data:Database){
|
export function setDatabase(data:Database){
|
||||||
|
|||||||
@@ -1,6 +1,12 @@
|
|||||||
|
import { language } from "src/lang"
|
||||||
|
import { alertInput } from "../alert"
|
||||||
|
|
||||||
|
let auth:string = null
|
||||||
|
let authChecked = false
|
||||||
|
|
||||||
export class NodeStorage{
|
export class NodeStorage{
|
||||||
async setItem(key:string, value:Uint8Array) {
|
async setItem(key:string, value:Uint8Array) {
|
||||||
|
await this.checkAuth()
|
||||||
const da = await fetch('/api/write', {
|
const da = await fetch('/api/write', {
|
||||||
method: "POST",
|
method: "POST",
|
||||||
body: JSON.stringify({
|
body: JSON.stringify({
|
||||||
@@ -8,7 +14,8 @@ export class NodeStorage{
|
|||||||
}),
|
}),
|
||||||
headers: {
|
headers: {
|
||||||
'content-type': 'application/json',
|
'content-type': 'application/json',
|
||||||
'file-path': Buffer.from(key, 'utf-8').toString('hex')
|
'file-path': Buffer.from(key, 'utf-8').toString('hex'),
|
||||||
|
'risu-auth': auth
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
if(da.status < 200 || da.status >= 300){
|
if(da.status < 200 || da.status >= 300){
|
||||||
@@ -21,10 +28,12 @@ export class NodeStorage{
|
|||||||
|
|
||||||
}
|
}
|
||||||
async getItem(key:string):Promise<Buffer> {
|
async getItem(key:string):Promise<Buffer> {
|
||||||
|
await this.checkAuth()
|
||||||
const da = await fetch('/api/read', {
|
const da = await fetch('/api/read', {
|
||||||
method: "GET",
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
'file-path': Buffer.from(key, 'utf-8').toString('hex')
|
'file-path': Buffer.from(key, 'utf-8').toString('hex'),
|
||||||
|
'risu-auth': auth
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
const data = await da.json()
|
const data = await da.json()
|
||||||
@@ -40,8 +49,12 @@ export class NodeStorage{
|
|||||||
return Buffer.from(data.content, 'base64')
|
return Buffer.from(data.content, 'base64')
|
||||||
}
|
}
|
||||||
async keys():Promise<string[]>{
|
async keys():Promise<string[]>{
|
||||||
|
await this.checkAuth()
|
||||||
const da = await fetch('/api/list', {
|
const da = await fetch('/api/list', {
|
||||||
method: "GET",
|
method: "GET",
|
||||||
|
headers:{
|
||||||
|
'risu-auth': auth
|
||||||
|
}
|
||||||
})
|
})
|
||||||
const data = await da.json()
|
const data = await da.json()
|
||||||
if(da.status < 200 || da.status >= 300){
|
if(da.status < 200 || da.status >= 300){
|
||||||
@@ -53,10 +66,12 @@ export class NodeStorage{
|
|||||||
return data.content
|
return data.content
|
||||||
}
|
}
|
||||||
async removeItem(key:string){
|
async removeItem(key:string){
|
||||||
|
await this.checkAuth()
|
||||||
const da = await fetch('/api/list', {
|
const da = await fetch('/api/list', {
|
||||||
method: "GET",
|
method: "GET",
|
||||||
headers: {
|
headers: {
|
||||||
'file-path': Buffer.from(key, 'utf-8').toString('hex')
|
'file-path': Buffer.from(key, 'utf-8').toString('hex'),
|
||||||
|
'risu-auth': auth
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
if(da.status < 200 || da.status >= 300){
|
if(da.status < 200 || da.status >= 300){
|
||||||
@@ -68,6 +83,58 @@ export class NodeStorage{
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private async checkAuth(){
|
||||||
|
if(!auth){
|
||||||
|
auth = localStorage.getItem('risuauth')
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!authChecked){
|
||||||
|
const data = await (await fetch('/api/password',{
|
||||||
|
headers: {
|
||||||
|
'risu-auth': auth ?? ''
|
||||||
|
}
|
||||||
|
})).json()
|
||||||
|
|
||||||
|
if(data.status === 'unset'){
|
||||||
|
const input = await digestPassword(await alertInput(language.setNodePassword))
|
||||||
|
await fetch('/api/set_password',{
|
||||||
|
method: "POST",
|
||||||
|
body:JSON.stringify({
|
||||||
|
password: input
|
||||||
|
}),
|
||||||
|
headers: {
|
||||||
|
'content-type': 'application/json'
|
||||||
|
}
|
||||||
|
})
|
||||||
|
auth = input
|
||||||
|
localStorage.setItem('risuauth', auth)
|
||||||
|
}
|
||||||
|
else if(data.status === 'incorrect'){
|
||||||
|
while(true){
|
||||||
|
const input = await digestPassword(await alertInput(language.inputNodePassword))
|
||||||
|
const data = await (await fetch('/api/password',{
|
||||||
|
headers: {
|
||||||
|
'risu-auth': input ?? ''
|
||||||
|
}
|
||||||
|
})).json()
|
||||||
|
if(data.status !== 'unset'){
|
||||||
|
auth = input
|
||||||
|
localStorage.setItem('risuauth', auth)
|
||||||
|
await this.checkAuth()
|
||||||
|
break
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
listItem = this.keys
|
listItem = this.keys
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async function digestPassword(message:string) {
|
||||||
|
const encoder = new TextEncoder();
|
||||||
|
const data = encoder.encode(message);
|
||||||
|
const hash = Buffer.from(await crypto.subtle.digest("SHA-256", data)).toString('hex');
|
||||||
|
return hash;
|
||||||
|
}
|
||||||
@@ -1 +1 @@
|
|||||||
{"version":"1.20.0"}
|
{"version":"1.20.1"}
|
||||||
Reference in New Issue
Block a user