├── src ├── boot │ ├── .gitkeep │ ├── plugins.js │ ├── i18n.js │ ├── components.js │ └── axios.js ├── store │ ├── module-example │ │ ├── getters.js │ │ ├── actions.js │ │ ├── mutations.js │ │ ├── state.js │ │ └── index.js │ ├── pathify.js │ ├── store-flag.d.ts │ ├── modules │ │ ├── server.js │ │ └── app.js │ └── index.js ├── i18n │ ├── index.js │ └── en-US │ │ └── index.js ├── router │ ├── routes.js │ └── index.js ├── components │ ├── index.js │ ├── Logs.vue │ ├── EssentialLink.vue │ ├── TitleBar.vue │ ├── Files.vue │ ├── HBOperations.vue │ ├── ServerBinariesDownload.vue │ └── Config.vue ├── pages │ ├── ErrorNotFound.vue │ └── IndexPage.vue ├── css │ ├── quasar.variables.scss │ └── app.scss ├── plugins │ ├── util.js │ └── hb.js ├── index.template.html ├── layouts │ ├── MainDrawer.vue │ └── MainLayout.vue ├── assets │ └── quasar-logo-vertical.svg └── App.vue ├── public ├── logo.png ├── store.db ├── hbcdn.png ├── icon0.png ├── favicon.ico ├── hbglobal.png ├── store.clean.db ├── icons │ ├── favicon-16x16.png │ ├── favicon-32x32.png │ ├── favicon-96x96.png │ └── favicon-128x128.png ├── settings.ini └── kofi.svg ├── src-electron ├── icons │ ├── icon.ico │ ├── icon.png │ └── icon.icns ├── electron-flag.d.ts ├── src │ ├── ipcMain-Server.js │ ├── ipcMain-FTP.js │ ├── ipcMain.js │ ├── ipcMain-ServerBinaryDownload.js │ ├── db.js │ ├── hb.js │ ├── ftp.js │ ├── server.js │ └── baseImage.js ├── electron-preload.js └── electron-main.js ├── .vscode ├── settings.json └── extensions.json ├── .editorconfig ├── .postcssrc.js ├── babel.config.js ├── .gitignore ├── jsconfig.json ├── package.json ├── README.md └── quasar.config.js /src/boot/.gitkeep: -------------------------------------------------------------------------------- 1 | -------------------------------------------------------------------------------- /src/store/module-example/getters.js: -------------------------------------------------------------------------------- 1 | export function someGetter (/* state */) { 2 | } 3 | -------------------------------------------------------------------------------- /public/logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/logo.png -------------------------------------------------------------------------------- /public/store.db: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/store.db -------------------------------------------------------------------------------- /src/store/module-example/actions.js: -------------------------------------------------------------------------------- 1 | export function someAction (/* context */) { 2 | } 3 | -------------------------------------------------------------------------------- /src/store/module-example/mutations.js: -------------------------------------------------------------------------------- 1 | export function someMutation (/* state */) { 2 | } 3 | -------------------------------------------------------------------------------- /public/hbcdn.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/hbcdn.png -------------------------------------------------------------------------------- /public/icon0.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/icon0.png -------------------------------------------------------------------------------- /public/favicon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/favicon.ico -------------------------------------------------------------------------------- /public/hbglobal.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/hbglobal.png -------------------------------------------------------------------------------- /src/i18n/index.js: -------------------------------------------------------------------------------- 1 | import enUS from './en-US' 2 | 3 | export default { 4 | 'en-US': enUS 5 | } 6 | -------------------------------------------------------------------------------- /public/store.clean.db: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/store.clean.db -------------------------------------------------------------------------------- /src/store/module-example/state.js: -------------------------------------------------------------------------------- 1 | export default function () { 2 | return { 3 | // 4 | } 5 | } 6 | -------------------------------------------------------------------------------- /src-electron/icons/icon.ico: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/src-electron/icons/icon.ico -------------------------------------------------------------------------------- /src-electron/icons/icon.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/src-electron/icons/icon.png -------------------------------------------------------------------------------- /.vscode/settings.json: -------------------------------------------------------------------------------- 1 | { 2 | "editor.bracketPairColorization.enabled": true, 3 | "editor.guides.bracketPairs": true 4 | } -------------------------------------------------------------------------------- /public/icons/favicon-16x16.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/icons/favicon-16x16.png -------------------------------------------------------------------------------- /public/icons/favicon-32x32.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/icons/favicon-32x32.png -------------------------------------------------------------------------------- /public/icons/favicon-96x96.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/icons/favicon-96x96.png -------------------------------------------------------------------------------- /src-electron/icons/icon.icns: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/src-electron/icons/icon.icns -------------------------------------------------------------------------------- /public/icons/favicon-128x128.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/Gkiokan/hb-store-cdn-server/HEAD/public/icons/favicon-128x128.png -------------------------------------------------------------------------------- /.editorconfig: -------------------------------------------------------------------------------- 1 | root = true 2 | 3 | [*] 4 | charset = utf-8 5 | indent_style = space 6 | indent_size = 2 7 | end_of_line = lf 8 | insert_final_newline = true 9 | trim_trailing_whitespace = true 10 | -------------------------------------------------------------------------------- /src/store/pathify.js: -------------------------------------------------------------------------------- 1 | // "vuex-pathify": "^1.5.1" 2 | import pathify from 'vuex-pathify' 3 | export default pathify 4 | 5 | // options 6 | pathify.options.mapping = 'simple' 7 | pathify.options.deep = 2 8 | -------------------------------------------------------------------------------- /src/i18n/en-US/index.js: -------------------------------------------------------------------------------- 1 | // This is just an example, 2 | // so you can safely delete all default props below 3 | 4 | export default { 5 | failed: 'Action failed', 6 | success: 'Action was successful' 7 | } 8 | -------------------------------------------------------------------------------- /src/boot/plugins.js: -------------------------------------------------------------------------------- 1 | import util from '~/plugins/util' 2 | import hb from '~/plugins/hb' 3 | 4 | export default ({ app }) => { 5 | app.config.globalProperties.$hb = hb 6 | app.config.globalProperties.$util = util 7 | } 8 | -------------------------------------------------------------------------------- /.postcssrc.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable */ 2 | // https://github.com/michael-ciniawsky/postcss-load-config 3 | 4 | module.exports = { 5 | plugins: [ 6 | // to edit target browsers: use "browserslist" field in package.json 7 | require('autoprefixer') 8 | ] 9 | } 10 | -------------------------------------------------------------------------------- /public/settings.ini: -------------------------------------------------------------------------------- 1 | [Settings] 2 | CDN=http://192.168.43.93:6448 3 | Secure_Boot=1 4 | temppath=/user/app/NPXS39041/downloads 5 | TTF_Font=/system_ex/app/NPXS20113/bdjstack/lib/fonts/SCE-PS3-RD-R-LATIN.TTF 6 | StoreOnUSB=0 7 | Show_install_prog=0 8 | HomeMenu_Redirection=0 9 | Daemon_on_start=1 10 | Legacy=0 11 | -------------------------------------------------------------------------------- /src/store/module-example/index.js: -------------------------------------------------------------------------------- 1 | import state from './state' 2 | import * as getters from './getters' 3 | import * as mutations from './mutations' 4 | import * as actions from './actions' 5 | 6 | export default { 7 | namespaced: true, 8 | getters, 9 | mutations, 10 | actions, 11 | state 12 | } 13 | -------------------------------------------------------------------------------- /src/boot/i18n.js: -------------------------------------------------------------------------------- 1 | import { boot } from 'quasar/wrappers' 2 | import { createI18n } from 'vue-i18n' 3 | import messages from 'src/i18n' 4 | 5 | export default boot(({ app }) => { 6 | const i18n = createI18n({ 7 | locale: 'en-US', 8 | messages 9 | }) 10 | 11 | // Set i18n instance on app 12 | app.use(i18n) 13 | }) 14 | -------------------------------------------------------------------------------- /babel.config.js: -------------------------------------------------------------------------------- 1 | /* eslint-disable */ 2 | 3 | module.exports = api => { 4 | return { 5 | presets: [ 6 | [ 7 | '@quasar/babel-preset-app', 8 | api.caller(caller => caller && caller.target === 'node') 9 | ? { targets: { node: 'current' } } 10 | : {} 11 | ] 12 | ] 13 | } 14 | } 15 | -------------------------------------------------------------------------------- /src/store/store-flag.d.ts: -------------------------------------------------------------------------------- 1 | /* eslint-disable */ 2 | // THIS FEATURE-FLAG FILE IS AUTOGENERATED, 3 | // REMOVAL OR CHANGES WILL CAUSE RELATED TYPES TO STOP WORKING 4 | import "quasar/dist/types/feature-flag"; 5 | 6 | declare module "quasar/dist/types/feature-flag" { 7 | interface QuasarFeatureFlags { 8 | store: true; 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /.vscode/extensions.json: -------------------------------------------------------------------------------- 1 | { 2 | "recommendations": [ 3 | "editorconfig.editorconfig", 4 | "johnsoncodehk.volar", 5 | "wayou.vscode-todo-highlight" 6 | ], 7 | "unwantedRecommendations": [ 8 | "octref.vetur", 9 | "hookyqr.beautify", 10 | "dbaeumer.jshint", 11 | "ms-vscode.vscode-typescript-tslint-plugin" 12 | ] 13 | } -------------------------------------------------------------------------------- /src-electron/electron-flag.d.ts: -------------------------------------------------------------------------------- 1 | /* eslint-disable */ 2 | // THIS FEATURE-FLAG FILE IS AUTOGENERATED, 3 | // REMOVAL OR CHANGES WILL CAUSE RELATED TYPES TO STOP WORKING 4 | import "quasar/dist/types/feature-flag"; 5 | 6 | declare module "quasar/dist/types/feature-flag" { 7 | interface QuasarFeatureFlags { 8 | electron: true; 9 | } 10 | } 11 | -------------------------------------------------------------------------------- /src/router/routes.js: -------------------------------------------------------------------------------- 1 | 2 | const routes = [ 3 | { 4 | path: '/', 5 | component: () => import('layouts/MainLayout.vue'), 6 | children: [ 7 | { path: '', component: () => import('pages/IndexPage.vue') } 8 | ] 9 | }, 10 | 11 | // Always leave this as last one, 12 | // but you can also remove it 13 | { 14 | path: '/:catchAll(.*)*', 15 | component: () => import('pages/ErrorNotFound.vue') 16 | } 17 | ] 18 | 19 | export default routes 20 | -------------------------------------------------------------------------------- /src-electron/src/ipcMain-Server.js: -------------------------------------------------------------------------------- 1 | import { ipcMain } from 'electron' 2 | import server from './server' 3 | 4 | ipcMain.handle('server-start', async(event, config) => { 5 | server.start(JSON.parse(config)) 6 | }) 7 | 8 | ipcMain.handle('server-restart', async(event, config) => { 9 | server.restart(JSON.parse(config)) 10 | }) 11 | 12 | ipcMain.handle('server-stop', async(event, config) => { 13 | server.stop() 14 | }) 15 | 16 | ipcMain.handle('server-scan', async(event, config) => { 17 | server.rescanFolder(JSON.parse(config)) 18 | }) 19 | -------------------------------------------------------------------------------- /src/components/index.js: -------------------------------------------------------------------------------- 1 | /* 2 | Autoload all current vue files as component and register them by their name. 3 | --- 4 | Author: Gkiokan Sali 5 | Date: 2019-05-09 6 | */ 7 | 8 | import Vue from 'vue' 9 | 10 | const requireContext = require.context('./', false, /.*\.vue$/) 11 | const layouts = requireContext.keys() 12 | .map(file => 13 | [file.replace(/(^.\/)|(\.vue$)/g, ''), requireContext(file)] 14 | ) 15 | .reduce((components, [name, component]) => { 16 | let Component = component.default || component 17 | Vue.component(Component.name, Component) 18 | }, {}) 19 | -------------------------------------------------------------------------------- /src/components/Logs.vue: -------------------------------------------------------------------------------- 1 | 16 | 17 | 28 | 29 | 31 | -------------------------------------------------------------------------------- /src-electron/src/ipcMain-FTP.js: -------------------------------------------------------------------------------- 1 | import { app, ipcMain } from 'electron' 2 | import ftp from './ftp' 3 | 4 | ipcMain.handle('get-logs', (event, config, log) => { 5 | ftp.getLogs(JSON.parse(config), log) 6 | }) 7 | 8 | ipcMain.handle('clean-logs', (event, config) => { 9 | ftp.cleanLogs(JSON.parse(config)) 10 | }) 11 | 12 | 13 | ipcMain.handle('get-settings', async (event, config) => { 14 | await ftp.getSettings(JSON.parse(config)) 15 | }) 16 | 17 | ipcMain.handle('update-settings', async (event, config) => { 18 | // await ftp.getSettings(JSON.parse(config)) 19 | await ftp.updateSettings(JSON.parse(config)) 20 | }) 21 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | .thumbs.db 3 | node_modules 4 | 5 | # Quasar core related directories 6 | .quasar 7 | /dist 8 | 9 | # Cordova related directories and files 10 | /src-cordova/node_modules 11 | /src-cordova/platforms 12 | /src-cordova/plugins 13 | /src-cordova/www 14 | 15 | # Capacitor related directories and files 16 | /src-capacitor/www 17 | /src-capacitor/node_modules 18 | 19 | # BEX related directories and files 20 | /src-bex/www 21 | /src-bex/js/core 22 | 23 | # Log files 24 | npm-debug.log* 25 | yarn-debug.log* 26 | yarn-error.log* 27 | 28 | # Editor directories and files 29 | .idea 30 | *.suo 31 | *.ntvs* 32 | *.njsproj 33 | *.sln 34 | -------------------------------------------------------------------------------- /src/store/modules/server.js: -------------------------------------------------------------------------------- 1 | import { make } from 'vuex-pathify' 2 | 3 | export const state = { 4 | state: null, 5 | ip: "", 6 | port: "6448", 7 | 8 | ps4ip: "", 9 | ps4port: "2121", 10 | 11 | basePath: null, 12 | binaryVersion: "0.00", 13 | files: [], 14 | assets: [], 15 | } 16 | 17 | 18 | // make all mutations 19 | export const mutations = { 20 | ...make.mutations(state), 21 | 22 | } 23 | 24 | // actions 25 | export const actions = { 26 | ...make.actions(state), 27 | } 28 | 29 | // getters 30 | export const getters = { 31 | // make all getters (optional) 32 | ...make.getters(state), 33 | } 34 | 35 | // console.log({ 36 | // mutations, actions, getters 37 | // }) 38 | -------------------------------------------------------------------------------- /src/boot/components.js: -------------------------------------------------------------------------------- 1 | // import something here 2 | // import { boot } from 'quasar/wrappers' 3 | // import '~/components' 4 | 5 | // "async" is optional; 6 | // more info on params: https://quasar.dev/quasar-cli/boot-files 7 | 8 | 9 | export default async ( { app, store, router, Vue } ) => { 10 | 11 | const requireContext = require.context('~/components', true, /.*\.vue$/) 12 | const layouts = requireContext.keys() 13 | .map(file => 14 | [file.replace(/(^.\/)|(\.vue$)/g, ''), requireContext(file)] 15 | ) 16 | .reduce((components, [name, component]) => { 17 | let Component = component.default || component 18 | app.component(Component.name, Component) 19 | }, {}) 20 | 21 | } 22 | -------------------------------------------------------------------------------- /src/pages/ErrorNotFound.vue: -------------------------------------------------------------------------------- 1 | 24 | 25 | 32 | -------------------------------------------------------------------------------- /jsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "baseUrl": ".", 4 | "paths": { 5 | "src/*": [ 6 | "src/*" 7 | ], 8 | "app/*": [ 9 | "*" 10 | ], 11 | "components/*": [ 12 | "src/components/*" 13 | ], 14 | "layouts/*": [ 15 | "src/layouts/*" 16 | ], 17 | "pages/*": [ 18 | "src/pages/*" 19 | ], 20 | "assets/*": [ 21 | "src/assets/*" 22 | ], 23 | "boot/*": [ 24 | "src/boot/*" 25 | ], 26 | "stores/*": [ 27 | "src/stores/*" 28 | ], 29 | "vue$": [ 30 | "node_modules/vue/dist/vue.runtime.esm-bundler.js" 31 | ] 32 | } 33 | }, 34 | "exclude": [ 35 | "dist", 36 | ".quasar", 37 | "node_modules" 38 | ] 39 | } -------------------------------------------------------------------------------- /src/pages/IndexPage.vue: -------------------------------------------------------------------------------- 1 | 20 | 21 | 33 | -------------------------------------------------------------------------------- /src/store/modules/app.js: -------------------------------------------------------------------------------- 1 | import { make } from 'vuex-pathify' 2 | 3 | export const state = { 4 | search: '', 5 | selected: null, 6 | serverFilesVersion: false, 7 | pkgs: [], 8 | logs: [], 9 | } 10 | 11 | 12 | // make all mutations 13 | export const mutations = { 14 | ...make.mutations(state), 15 | 16 | } 17 | 18 | // actions 19 | export const actions = { 20 | ...make.actions(state), 21 | 22 | // addFiles({ commit, dispatch, state}, payload){ 23 | // commit('addFiles', payload) 24 | // } 25 | } 26 | 27 | // getters 28 | export const getters = { 29 | // make all getters (optional) 30 | ...make.getters(state), 31 | 32 | // overwrite default `items` getter 33 | // allFiles: state => { 34 | // return state.images 35 | // }, 36 | } 37 | 38 | // console.log({ 39 | // mutations, actions, getters 40 | // }) 41 | -------------------------------------------------------------------------------- /src/css/quasar.variables.scss: -------------------------------------------------------------------------------- 1 | // Quasar SCSS (& Sass) Variables 2 | // -------------------------------------------------- 3 | // To customize the look and feel of this app, you can override 4 | // the Sass/SCSS variables found in Quasar's source Sass/SCSS files. 5 | 6 | // Check documentation for full list of Quasar variables 7 | 8 | // Your own variables (that are declared here) and Quasar's own 9 | // ones will be available out of the box in your .vue/.scss/.sass files 10 | 11 | // It's highly recommended to change the default colors 12 | // to match your app's branding. 13 | // Tip: Use the "Theme Builder" on Quasar's documentation website. 14 | 15 | $primary : #1976D2; 16 | $secondary : #26A69A; 17 | $accent : #9C27B0; 18 | 19 | $dark : #1D1D1D; 20 | 21 | $positive : #21BA45; 22 | $negative : #C10015; 23 | $info : #31CCEC; 24 | $warning : #F2C037; 25 | -------------------------------------------------------------------------------- /src/components/EssentialLink.vue: -------------------------------------------------------------------------------- 1 | 21 | 22 | 50 | -------------------------------------------------------------------------------- /src/plugins/util.js: -------------------------------------------------------------------------------- 1 | let util = { 2 | getNetWorkInterfaces() { 3 | console.log('running') 4 | let os = require('os'); 5 | let ifaces = []; 6 | Object.keys(os.networkInterfaces()).forEach(function (ifname) { 7 | var alias = 0; 8 | os.networkInterfaces()[ifname].forEach(function (iface) { 9 | if ('IPv4' !== iface.family || iface.internal !== false) { 10 | return; 11 | } 12 | 13 | if (alias >= 1) { 14 | ifaces.push({ 15 | title: `${ifname}-${alias}:${iface.address}`, 16 | ip: iface.address 17 | }); 18 | } else { 19 | ifaces.push({ 20 | title: `${ifname}: ${iface.address}`, 21 | ip: iface.address 22 | }); 23 | } 24 | ++alias; 25 | }); 26 | }); 27 | return ifaces; 28 | }, 29 | } 30 | 31 | export default util 32 | -------------------------------------------------------------------------------- /src/boot/axios.js: -------------------------------------------------------------------------------- 1 | import { boot } from 'quasar/wrappers' 2 | import axios from 'axios' 3 | 4 | // Be careful when using SSR for cross-request state pollution 5 | // due to creating a Singleton instance here; 6 | // If any client changes this (global) instance, it might be a 7 | // good idea to move this instance creation inside of the 8 | // "export default () => {}" function below (which runs individually 9 | // for each client) 10 | const api = axios.create({ }) 11 | 12 | export default boot(({ app }) => { 13 | // for use inside Vue files (Options API) through this.$axios and this.$api 14 | 15 | app.config.globalProperties.$axios = axios 16 | // ^ ^ ^ this will allow you to use this.$axios (for Vue Options API form) 17 | // so you won't necessarily have to import axios in each vue file 18 | 19 | app.config.globalProperties.$api = api 20 | // ^ ^ ^ this will allow you to use this.$api (for Vue Options API form) 21 | // so you can easily perform requests against your app's API 22 | }) 23 | 24 | export { api } 25 | -------------------------------------------------------------------------------- /src/index.template.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | <%= productName %> 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 17 | 18 | 19 | 20 |
21 | 22 | 23 | -------------------------------------------------------------------------------- /src/router/index.js: -------------------------------------------------------------------------------- 1 | import { route } from 'quasar/wrappers' 2 | import { createRouter, createMemoryHistory, createWebHistory, createWebHashHistory } from 'vue-router' 3 | import routes from './routes' 4 | 5 | /* 6 | * If not building with SSR mode, you can 7 | * directly export the Router instantiation; 8 | * 9 | * The function below can be async too; either use 10 | * async/await or return a Promise which resolves 11 | * with the Router instance. 12 | */ 13 | 14 | export default route(function (/* { store, ssrContext } */) { 15 | const createHistory = process.env.SERVER 16 | ? createMemoryHistory 17 | : (process.env.VUE_ROUTER_MODE === 'history' ? createWebHistory : createWebHashHistory) 18 | 19 | const Router = createRouter({ 20 | scrollBehavior: () => ({ left: 0, top: 0 }), 21 | routes, 22 | 23 | // Leave this as is and make changes in quasar.conf.js instead! 24 | // quasar.conf.js -> build -> vueRouterMode 25 | // quasar.conf.js -> build -> publicPath 26 | history: createHistory(process.env.MODE === 'ssr' ? void 0 : process.env.VUE_ROUTER_BASE) 27 | }) 28 | 29 | return Router 30 | }) 31 | -------------------------------------------------------------------------------- /src/css/app.scss: -------------------------------------------------------------------------------- 1 | $offsetForTitleBar: -3px; 2 | 3 | 4 | html, body { 5 | /* Foreground, Background */ 6 | scrollbar-color: #999 #333; 7 | 8 | ::-webkit-scrollbar { 9 | width: 10px; /* Mostly for vertical scrollbars */ 10 | height: 5px; /* Mostly for horizontal scrollbars */ 11 | } 12 | 13 | ::-webkit-scrollbar-thumb { /* Foreground */ 14 | background: #2a2a2a; 15 | } 16 | ::-webkit-scrollbar-track { /* Background */ 17 | background: #1a1a1a; 18 | } 19 | 20 | } 21 | 22 | .q-panel-parent.panel-wrapper { 23 | position: absolute; 24 | top: 0px; left: 0px; right: 0px; bottom: 0px; 25 | border: 0px solid red; 26 | } 27 | 28 | .titleBar { 29 | cursor: pointer; 30 | margin-left: $offsetForTitleBar; 31 | margin-right: $offsetForTitleBar; 32 | margin-top: $offsetForTitleBar; 33 | } 34 | 35 | .draggable { 36 | -webkit-user-select: none; 37 | -webkit-app-region: drag; 38 | } 39 | 40 | .cursor-pointer { 41 | cursor: pointer !important; 42 | } 43 | 44 | .q-field--outlined.q-field--readonly .q-field__control:before { 45 | border-style: solid !important; 46 | } 47 | -------------------------------------------------------------------------------- /src-electron/src/ipcMain.js: -------------------------------------------------------------------------------- 1 | import { BrowserWindow, app, ipcMain, dialog, shell } from 'electron' 2 | import { download } from 'electron-dl' 3 | 4 | ipcMain.handle('open-dir', async (event, path) => { 5 | return dialog.showOpenDialog({ properties: ['openDirectory'] }) 6 | }) 7 | 8 | ipcMain.handle('getNetWorkInterfaces', async (event) => { 9 | let os = require('os'); 10 | let ifaces = []; 11 | Object.keys(os.networkInterfaces()).forEach(function (ifname) { 12 | var alias = 0; 13 | os.networkInterfaces()[ifname].forEach(function (iface) { 14 | if ('IPv4' !== iface.family || iface.internal !== false) { 15 | return; 16 | } 17 | 18 | if (alias >= 1) { 19 | ifaces.push({ 20 | title: `${ifname}-${alias}:${iface.address}`, 21 | ip: iface.address 22 | }); 23 | } else { 24 | ifaces.push({ 25 | title: `${ifname}: ${iface.address}`, 26 | ip: iface.address 27 | }); 28 | } 29 | ++alias; 30 | }); 31 | }); 32 | return ifaces; 33 | }) 34 | 35 | ipcMain.handle('open-url', (e, url) => { 36 | shell.openExternal(url) 37 | }) 38 | 39 | ipcMain.handle('closeApplication', () => { 40 | app.quit() 41 | }) 42 | -------------------------------------------------------------------------------- /src/components/TitleBar.vue: -------------------------------------------------------------------------------- 1 | 14 | 15 | 45 | 46 | 48 | -------------------------------------------------------------------------------- /src/store/index.js: -------------------------------------------------------------------------------- 1 | import { store } from 'quasar/wrappers' 2 | import { createStore } from 'vuex' 3 | import pathify from './pathify' 4 | 5 | // import example from './module-example' 6 | 7 | /* 8 | * If not building with SSR mode, you can 9 | * directly export the Store instantiation; 10 | * 11 | * The function below can be async too; either use 12 | * async/await or return a Promise which resolves 13 | * with the Store instance. 14 | */ 15 | 16 | const requireContext = require.context('./modules', false, /.*\.js$/) 17 | 18 | const modules = requireContext.keys() 19 | .map(file => 20 | [file.replace(/(^.\/)|(\.js$)/g, ''), requireContext(file)] 21 | ) 22 | .reduce((modules, [name, module]) => { 23 | if (module.namespaced === undefined) { 24 | module.namespaced = true 25 | } 26 | 27 | return { ...modules, [name]: module } 28 | }, {}) 29 | 30 | 31 | const Store = createStore({ 32 | plugins: [ pathify.plugin ], 33 | modules, 34 | 35 | mutations: { 36 | initialiseStore(state) { 37 | if(localStorage.getItem('store')) { 38 | let initialState = JSON.parse(JSON.stringify(state)) 39 | let restoredStore = JSON.parse(localStorage.getItem('store')) 40 | let finalStore = { ...initialState, ...restoredStore } 41 | 42 | // console.log(initialState) 43 | // console.log(finalStore) 44 | 45 | this.replaceState( 46 | Object.assign(state, finalStore) 47 | ); 48 | } 49 | } 50 | } 51 | // enable strict mode (adds overhead!) 52 | // for dev mode and --debug builds only 53 | // strict: process.env.DEBUGGING 54 | }) 55 | 56 | export default Store 57 | -------------------------------------------------------------------------------- /src/components/Files.vue: -------------------------------------------------------------------------------- 1 | 33 | 34 | 57 | 58 | 65 | -------------------------------------------------------------------------------- /src/plugins/hb.js: -------------------------------------------------------------------------------- 1 | import axios from 'axios' 2 | import store from '~/store' 3 | import semver from 'semver' 4 | 5 | const hb = { 6 | data: { 7 | source: 'https://api.github.com/repos/LightningMods/PS4-Store/releases', 8 | files: [ 9 | 'homebrew.elf', 10 | 'homebrew.elf.sig', 11 | 'remote.md5', 12 | 'store.prx', 13 | 'store.prx.sig', 14 | ], 15 | }, 16 | 17 | async getRelease(){ 18 | let { data } = await axios.get(this.data.source) 19 | return data.length ? data[0] : false 20 | }, 21 | 22 | getName(release=null){ 23 | if(!release) return "NO_RELEASE_OBJECT" 24 | return release.name 25 | }, 26 | 27 | getVersion(release=null){ 28 | if(!release) return "NO_RELEASE_OBJECT" 29 | return release.tag_name 30 | }, 31 | 32 | getAssets(release=null){ 33 | if(!release) return "NO_RELEASE_OBJECT" 34 | let assets = release.assets 35 | let urls = [] 36 | 37 | assets.map( f => { 38 | urls.push({ 39 | name: f.name, 40 | progress: 0, 41 | url: f.browser_download_url 42 | }) 43 | }) 44 | 45 | return urls 46 | }, 47 | 48 | checkVersion(currentVersion="0.00", version="0.00"){ 49 | console.log("compare ", currentVersion, version) 50 | return this.compareVersion(currentVersion, version) 51 | }, 52 | 53 | compareVersion(v1, v2) { 54 | const v1Parts = v1.split('.') 55 | const v2Parts = v2.split('.') 56 | const length = Math.max(v1Parts.length, v2Parts.length) 57 | for (let i = 0; i < length; i++) { 58 | const value = (parseInt(v1Parts[i]) || 0) - (parseInt(v2Parts[i]) || 0) 59 | if (value < 0) return -1 60 | if (value > 0) return 1 61 | } 62 | return 0 63 | }, 64 | 65 | 66 | } 67 | 68 | export default hb 69 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "hb-store-cdn-server", 3 | "version": "1.8.0", 4 | "description": "HB-Store CDN Server", 5 | "productName": "HB-Store CDN Server", 6 | "homepage": "https://gkiokan.net", 7 | "author": "Gkiokan ", 8 | "private": true, 9 | "scripts": { 10 | "test": "echo \"No test specified\" && exit 0", 11 | "postinstall": "electron-builder install-app-deps", 12 | "dev": "quasar dev -m electron", 13 | "build:win": "quasar build -m electron --target win32", 14 | "build:linux": "quasar build -m electron --target linux", 15 | "build:mac": "quasar build -m electron --target darwin", 16 | "build:all": "quasar build -m electron --target all" 17 | }, 18 | "dependencies": { 19 | "@njzy/ps4-pkg-info": "^0.1.0", 20 | "@quasar/extras": "^1.0.0", 21 | "axios": "^0.21.1", 22 | "basic-ftp": "^4.6.6", 23 | "better-sqlite3": "^7.5.1", 24 | "config-ini-parser": "^1.5.9", 25 | "core-js": "^3.6.5", 26 | "electron-dl": "^3.3.1", 27 | "express": "^4.17.3", 28 | "extract-zip": "^2.0.1", 29 | "fast-glob": "^3.2.11", 30 | "http": "0.0.1-security", 31 | "md5-file": "^5.0.0", 32 | "node-polyfill-webpack-plugin": "^1.1.4", 33 | "normalize-path": "^3.0.0", 34 | "os": "^0.1.2", 35 | "path": "^0.12.7", 36 | "ps4-pkg-info": "^1.0.1", 37 | "quasar": "^2.6.0", 38 | "semver": "^7.3.7", 39 | "url": "^0.11.0", 40 | "vue": "^3.0.0", 41 | "vue-i18n": "^9.0.0", 42 | "vue-router": "^4.0.0", 43 | "vuex": "^4.0.1", 44 | "vuex-pathify": "^3.0.0-beta" 45 | }, 46 | "devDependencies": { 47 | "@quasar/app-webpack": "^3.0.0", 48 | "electron": "^18.0.1", 49 | "electron-builder": "^23.0.3" 50 | }, 51 | "browserslist": [ 52 | "last 10 Chrome versions", 53 | "last 10 Firefox versions", 54 | "last 4 Edge versions", 55 | "last 7 Safari versions", 56 | "last 8 Android versions", 57 | "last 8 ChromeAndroid versions", 58 | "last 8 FirefoxAndroid versions", 59 | "last 10 iOS versions", 60 | "last 5 Opera versions" 61 | ], 62 | "engines": { 63 | "node": ">= 12.22.1", 64 | "npm": ">= 6.13.4", 65 | "yarn": ">= 1.21.1" 66 | } 67 | } 68 | -------------------------------------------------------------------------------- /src-electron/electron-preload.js: -------------------------------------------------------------------------------- 1 | /** 2 | * This file is used specifically for security reasons. 3 | * Here you can access Nodejs stuff and inject functionality into 4 | * the renderer thread (accessible there through the "window" object) 5 | * 6 | * WARNING! 7 | * If you import anything from node_modules, then make sure that the package is specified 8 | * in package.json > dependencies and NOT in devDependencies 9 | * 10 | * Example (injects window.myAPI.doAThing() into renderer thread): 11 | * 12 | * import { contextBridge } from 'electron' 13 | * 14 | * contextBridge.exposeInMainWorld('myAPI', { 15 | * doAThing: () => {} 16 | * }) 17 | */ 18 | 19 | import { contextBridge } from 'electron' 20 | import { ipcRenderer } from 'electron' 21 | 22 | contextBridge.exposeInMainWorld('hb', { 23 | openBasePathDialog: () => ipcRenderer.invoke('open-dir'), 24 | getNetWorkInterfaces: () => ipcRenderer.invoke('getNetWorkInterfaces'), 25 | downloadServerBinaries: (f) => ipcRenderer.invoke('download-server-binaries', f), 26 | closeApplication: () => ipcRenderer.invoke('closeApplication'), 27 | }) 28 | 29 | contextBridge.exposeInMainWorld('ipc', { 30 | on: (channel, cb) => ipcRenderer.on(channel, cb), 31 | open: (url) => ipcRenderer.invoke('open-url', url), 32 | removeListener: (channel, cb) => ipcRenderer.removeAllListeners(channel), 33 | checkServerBinaries: (ch, cb) => ipcRenderer.invoke('trigger-check-server-binaries') 34 | }) 35 | 36 | contextBridge.exposeInMainWorld('server', { 37 | start: (server) => ipcRenderer.invoke('server-start', server), 38 | restart: (server) => ipcRenderer.invoke('server-restart', server), 39 | stop: () => ipcRenderer.invoke('server-stop'), 40 | scan: (server) => ipcRenderer.invoke('server-scan', server), 41 | }) 42 | 43 | contextBridge.exposeInMainWorld('ftp', { 44 | getLogs: (server, log) => ipcRenderer.invoke('get-logs', server, log), 45 | cleanLogs: (server) => ipcRenderer.invoke('clean-logs', server), 46 | getSettings: (server) => ipcRenderer.invoke('get-settings', server), 47 | updateSettings: (server) => ipcRenderer.invoke('update-settings', server), 48 | restoreSettings: (server) => ipcRenderer.invoke('restore-settings', server), 49 | }) 50 | -------------------------------------------------------------------------------- /src-electron/src/ipcMain-ServerBinaryDownload.js: -------------------------------------------------------------------------------- 1 | import { BrowserWindow, app, ipcMain, dialog } from 'electron' 2 | import { download } from 'electron-dl' 3 | import fs from 'fs' 4 | import path from 'path' 5 | import extract from 'extract-zip' 6 | 7 | console.assert = () => {}; 8 | 9 | ipcMain.handle('download-server-binaries', async(event, file) => { 10 | console.log("Server assets to download", file) 11 | 12 | // const win = BrowserWindow.getFocusedWindow(); 13 | let win = BrowserWindow.getFocusedWindow(); 14 | 15 | if(!win){ 16 | let all = BrowserWindow.getAllWindows() 17 | win = all[0] 18 | } 19 | 20 | const binPath = app.getPath('userData') + '/bin' 21 | 22 | if (!fs.existsSync(binPath)){ 23 | fs.mkdirSync(binPath); 24 | } 25 | 26 | console.log("Server Binary Folder", binPath) 27 | console.log("Download file", file) 28 | 29 | try { 30 | await download(win, file, { 31 | directory: binPath, 32 | overwrite: true, 33 | onProgress: o => win.webContents.send('download-complete', { file, item: o }), 34 | errorMessage: e => alert(e), 35 | }) 36 | 37 | console.log("File should be downloaded " + file) 38 | 39 | // Post downlaod process 40 | if( file.includes('.zip')){ 41 | console.log("Found a zip file to extract at " + file) 42 | 43 | let filename = path.basename(file) 44 | let filePath = binPath + '/' + filename 45 | 46 | console.log({ filename, filePath, binPath }) 47 | 48 | try { 49 | console.log("[....] Extracting " + filename) 50 | extract(filePath, { dir: binPath }) 51 | console.log("[done] Extracting " + filename) 52 | } 53 | catch(e){ 54 | console.log("Error Extracting file " + filename) 55 | } 56 | } 57 | } 58 | catch (e) { alert(e); console.error('(download)', e); } 59 | 60 | // console.log(await download(win, url)); 61 | }) 62 | 63 | 64 | ipcMain.handle('trigger-check-server-binaries', () => { 65 | let win = BrowserWindow.getFocusedWindow(); 66 | 67 | if(!win){ 68 | let all = BrowserWindow.getAllWindows() 69 | win = all[0] 70 | } 71 | 72 | win.webContents.send('check-server-binaries') 73 | }) 74 | -------------------------------------------------------------------------------- /src-electron/src/db.js: -------------------------------------------------------------------------------- 1 | import { app, BrowserWindow } from 'electron' 2 | import fs from 'fs' 3 | import path from 'path' 4 | import Database from 'better-sqlite3'; 5 | 6 | let db; 7 | 8 | export default { 9 | 10 | getWindow(){ 11 | let win = BrowserWindow.getFocusedWindow(); 12 | 13 | if(!win){ 14 | let all = BrowserWindow.getAllWindows() 15 | win = all[0] 16 | } 17 | 18 | return win 19 | }, 20 | 21 | error(err=null){ 22 | this.getWindow().webContents.send('error', err) 23 | this.log(err) 24 | }, 25 | 26 | log(msg=null){ 27 | this.getWindow().webContents.send('log', msg) 28 | console.log("Server:: " + msg) 29 | }, 30 | 31 | 32 | getCleanStorePath(){ 33 | return path.resolve(__dirname, process.env.QUASAR_PUBLIC_FOLDER) + '/store.clean.db' 34 | }, 35 | 36 | getStorePath(){ 37 | return app.getPath('userData') + '/bin/store.db' 38 | }, 39 | 40 | renewDB(){ 41 | let clean = this.getCleanStorePath() 42 | let store = this.getStorePath() 43 | let dir = path.dirname(store) 44 | 45 | if (!fs.existsSync(dir)) { 46 | fs.mkdirSync(dir); 47 | } 48 | 49 | try { 50 | fs.copyFileSync(clean, store) 51 | this.log("store.db has been renewed") 52 | } 53 | catch(e){ 54 | this.error(e) 55 | } 56 | }, 57 | 58 | instance(){ 59 | const db = new Database(this.getStorePath()) 60 | this.db = db 61 | 62 | return db 63 | }, 64 | 65 | addAllItems(items){ 66 | const db = this.instance() 67 | 68 | const insert = db.prepare("INSERT INTO homebrews (pid,id,name,desc,image,package,version,picpath,desc_1,desc_2,ReviewStars,Size,Author,apptype,pv,main_icon_path,main_menu_pic,releaseddate) VALUES (CAST(@pid AS INTEGER),@id,@name,@desc,@image,@package,@version,@picpath,@desc_1,@desc_2,@ReviewStars,@Size,@Author,@apptype,@pv,@main_icon_path,@main_menu_pic,@releaseddate)") 69 | 70 | const insertAll = db.transaction( items => { 71 | for (const item of items) 72 | insert.run(item) 73 | }) 74 | 75 | insertAll(items) 76 | }, 77 | 78 | 79 | } 80 | -------------------------------------------------------------------------------- /src-electron/electron-main.js: -------------------------------------------------------------------------------- 1 | import { app, BrowserWindow, Menu, nativeTheme } from 'electron' 2 | import path from 'path' 3 | import os from 'os' 4 | import electronDl from 'electron-dl' 5 | 6 | // Initialise 7 | import './src/ipcMain' 8 | import './src/ipcMain-ServerBinaryDownload' 9 | import './src/ipcMain-Server' 10 | import './src/ipcMain-FTP' 11 | 12 | // Initialise config 13 | electronDl({ 14 | directory: app.getPath('userData') + '/bin', 15 | overwrite: true, 16 | }) 17 | 18 | // needed in case process is undefined under Linux 19 | const platform = process.platform || os.platform() 20 | 21 | try { 22 | if (platform === 'win32' && nativeTheme.shouldUseDarkColors === true) { 23 | require('fs').unlinkSync(path.join(app.getPath('userData'), 'DevTools Extensions')) 24 | } 25 | } 26 | catch (_) { } 27 | 28 | let mainWindow 29 | 30 | function createWindow () { 31 | /** 32 | * Initial window options 33 | */ 34 | mainWindow = new BrowserWindow({ 35 | icon: path.resolve(__dirname, 'icons/icon.png'), // tray icon 36 | width: 440, 37 | height: 600, 38 | useContentSize: true, 39 | frame: false, 40 | icon: path.join(__dirname, '/icons/icon.png'), 41 | webPreferences: { 42 | nodeIntegration: true, 43 | contextIsolation: true, 44 | enableRemoteModule: true, 45 | // More info: /quasar-cli/developing-electron-apps/electron-preload-script 46 | preload: path.resolve(__dirname, process.env.QUASAR_ELECTRON_PRELOAD), 47 | allowRunningInsecureContent: false, 48 | sandbox: false, 49 | } 50 | }) 51 | 52 | mainWindow.setMenu(null) 53 | mainWindow.loadURL(process.env.APP_URL) 54 | 55 | if (process.env.DEBUGGING) { 56 | // if on DEV or Production with debug enabled 57 | mainWindow.webContents.openDevTools() 58 | } 59 | else { 60 | // we're on production; no access to devtools pls 61 | mainWindow.webContents.on('devtools-opened', () => { 62 | mainWindow.webContents.closeDevTools() 63 | }) 64 | } 65 | 66 | mainWindow.on('closed', () => { 67 | mainWindow = null 68 | }) 69 | 70 | // Menu.setApplicationMenu(null) 71 | } 72 | 73 | app.whenReady().then(createWindow) 74 | 75 | app.on('window-all-closed', () => { 76 | if (platform !== 'darwin') { 77 | app.quit() 78 | } 79 | }) 80 | 81 | app.on('activate', () => { 82 | if (mainWindow === null) { 83 | createWindow() 84 | } 85 | }) 86 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # HB-Store CDN Server 2 | [![ko-fi](https://img.shields.io/badge/Buy%20me%20a%20Shisha%20on-Ko--fi-red)](https://ko-fi.com/M4M082WK8) 3 | [![os](https://img.shields.io/badge/platform-windows%20%7C%20macos%20%7C%20linux-lightgrey)](#) 4 | [![commits_since_release](https://img.shields.io/github/commits-since/gkiokan/hb-store-cdn-server/v1.7.2)](#) 5 | [![version](https://img.shields.io/github/package-json/v/gkiokan/hb-store-cdn-server)](#) 6 | [![downloads](https://img.shields.io/github/downloads/gkiokan/hb-store-cdn-server/total)](#) 7 | [![last_commit](https://img.shields.io/github/last-commit/gkiokan/hb-store-cdn-server)](#) 8 | 9 | This is the new HB-Store CDN Tool for you. 10 | 11 | ![HB-Store CDN Server](https://user-images.githubusercontent.com/7249224/170701265-ccd48140-e497-4c15-8382-efb100d3fc1b.jpeg) 12 | 13 | 14 | ## Features 15 | - [x] Dead simple usage 16 | - [x] Custom Network Interface and Port 17 | - [x] Select base path to scan all pkgs (deep scan) 18 | - [x] Integrated node express server 19 | - [x] Maps file paths correctly, even with special character and empty spaces 20 | - [x] List of found files with preview of icon0 and information 21 | - [x] Add Button for "update my settings.ini" (update of HB-Store Settings by one click) 22 | - [x] Add Button for "Get HB-Store logs" (retrieve the logs by one click) 23 | - [x] Configuration Validation 24 | - [x] Autoscan BasePath after you change it 25 | - [x] Public Domain Host support 26 | - [x] Add PS5 Support 27 | 28 | ## ToDo 29 | - [ ] More Tweaks 30 | - [x] CLI version [HB-Store CDN Server CLI-Version available here](https://github.com/Gkiokan/hb-store-cdn-cli-server) 31 | 32 | ## How To 33 | 1.) Selct your IP and choose a Port 34 | 2.) Add your base path where your pkgs lives 35 | 3.) Check for Server Binary 36 | -- sofar this steps only required once, it will autosave the values -- 37 | 4.) Start the Server -> it will show your CDN Address below the title bar 38 | 39 | ## How to configure HB-Store 40 | 1.) Open HB-Store app on your 41 | 2.) Set the given CDN Address as your CDN 42 | 3.) Save Settings 43 | 4.) Close the HB-Store by pressing options button and open it again 44 | -- settings needs to be done once -- 45 | 5.) You should see the content from your Server now. 46 | 47 | 48 | ## Support 49 | If you want to Support me and my development, you can do this here. 50 | 51 | [![ko-fi](https://ko-fi.com/img/githubbutton_sm.svg)](https://ko-fi.com/M4M082WK8) 52 | -------------------------------------------------------------------------------- /src-electron/src/hb.js: -------------------------------------------------------------------------------- 1 | import fs from 'fs' 2 | import pkgInfo from 'ps4-pkg-info' 3 | import path from 'path' 4 | import baseImage from './baseImage' 5 | 6 | export default { 7 | files: [], 8 | 9 | createItem(data, file, pid=1){ 10 | // let patchedFilename = item.name.replace(/[^a-zA-Z0-9-_.]/g, '') 11 | let patchedFilename = (file.charAt(0) == "/") ? file.substr(1).replace(/[^a-zA-Z0-9-_./]/g, '') : file.replace(/[^a-zA-Z0-9-_./]/g, '') 12 | let stats = fs.lstatSync(file) 13 | let size = this.formatBytes(stats.size, 2) 14 | 15 | let item = { 16 | "pid": pid, 17 | "id": data.paramSfo.TITLE_ID, 18 | "name": data.paramSfo.TITLE, 19 | "desc": "", 20 | "image": "__image", 21 | "package": "__package", 22 | "version": data.paramSfo.APP_VER, 23 | "picpath": "/user/app/NPXS39041/storedata/" + data.paramSfo.TITLE_ID + ".png", 24 | "desc_1": "", 25 | "desc_2": "", 26 | "ReviewStars": "Custom Rating", 27 | "Size": size, 28 | "Author": "HB-Store CDN", 29 | "apptype": "HB Game", 30 | "pv": "5.05+", 31 | "main_icon_path": "__image", 32 | "main_menu_pic": "/user/app/NPXS39041/storedata/" + data.paramSfo.TITLE_ID + ".png", 33 | "releaseddate": "2019-04-30", 34 | path: file, 35 | filename: path.basename(file), 36 | patchedFilename, 37 | icon0: data.icon0 ?? baseImage, 38 | } 39 | 40 | return item 41 | }, 42 | 43 | removeBasePath(item, toRemove){ 44 | item.patchedFilename = item.patchedFilename.replace(toRemove, '') 45 | 46 | if(item.patchedFilename.charAt(0) == '/') 47 | item.patchedFilename = item.patchedFilename.substr(1) 48 | 49 | return item 50 | }, 51 | 52 | getImagePathURI(data){ 53 | return data.patchedFilename + '/icon0.png' 54 | }, 55 | 56 | addImages(data=null, base){ 57 | let id = data.id 58 | let patched 59 | let image = base + '/' + this.getImagePathURI(data) 60 | 61 | data.image = image 62 | data.main_icon_path = image 63 | 64 | return data 65 | }, 66 | 67 | formatBytes(bytes, decimals=2, k=1000) { 68 | if (bytes === 0) return '0 Bytes'; 69 | 70 | const dm = decimals < 0 ? 0 : decimals; 71 | const sizes = ['Bytes', 'KB', 'MB', 'GB', 'TB', 'PB', 'EB', 'ZB', 'YB']; 72 | 73 | const i = Math.floor(Math.log(bytes) / Math.log(k)); 74 | 75 | return parseFloat((bytes / Math.pow(k, i)).toFixed(dm)) + ' ' + sizes[i]; 76 | }, 77 | 78 | } 79 | -------------------------------------------------------------------------------- /src/layouts/MainDrawer.vue: -------------------------------------------------------------------------------- 1 | 53 | 54 | 59 | 60 | 62 | -------------------------------------------------------------------------------- /src/layouts/MainLayout.vue: -------------------------------------------------------------------------------- 1 | 43 | 44 | 72 | 73 | 106 | -------------------------------------------------------------------------------- /src/components/HBOperations.vue: -------------------------------------------------------------------------------- 1 | 64 | 65 | 114 | 115 | 117 | -------------------------------------------------------------------------------- /src/assets/quasar-logo-vertical.svg: -------------------------------------------------------------------------------- 1 | 2 | 4 | 6 | 8 | 10 | 12 | 14 | 15 | -------------------------------------------------------------------------------- /src/App.vue: -------------------------------------------------------------------------------- 1 | 4 | 5 | 199 | -------------------------------------------------------------------------------- /src/components/ServerBinariesDownload.vue: -------------------------------------------------------------------------------- 1 | 32 | 33 | 172 | 173 | 175 | -------------------------------------------------------------------------------- /src/components/Config.vue: -------------------------------------------------------------------------------- 1 | 57 | 58 | 175 | 176 | 178 | -------------------------------------------------------------------------------- /src-electron/src/ftp.js: -------------------------------------------------------------------------------- 1 | import { app, BrowserWindow, dialog } from 'electron' 2 | import { download } from 'electron-dl' 3 | import { Client } from 'basic-ftp' 4 | import { ConfigIniParser } from 'config-ini-parser' 5 | import path from 'path' 6 | import fs from 'fs' 7 | 8 | export default { 9 | ip: "", 10 | port: "", 11 | 12 | files: { 13 | loader: "/user/app/NPXS39041/logs/loader.log", 14 | log: "/user/app/NPXS39041/logs/store.log", 15 | itemzflow: "/user/app/NPXS39041/logs/itemzflow.log", 16 | settings: "/user/app/NPXS39041/settings.ini", 17 | }, 18 | 19 | getWindow(){ 20 | let win = BrowserWindow.getFocusedWindow(); 21 | 22 | if(!win){ 23 | let all = BrowserWindow.getAllWindows() 24 | win = all[0] 25 | } 26 | 27 | return win 28 | }, 29 | 30 | getLocalFile(file=null){ 31 | return app.getPath('userData') + '/data/' + path.basename(file) 32 | }, 33 | 34 | setConfig(config){ 35 | this.ip = config.ps4ip 36 | this.port = config.ps4port 37 | }, 38 | 39 | error(err=null){ 40 | // deprecated 41 | // const win = BrowserWindow.getFocusedWindow(); 42 | this.getWindow().webContents.send('error', err) 43 | this.log(err) 44 | }, 45 | 46 | notify(msg=null){ 47 | this.getWindow().webContents.send('notify', msg) 48 | this.log(msg) 49 | }, 50 | 51 | log(msg=null){ 52 | this.getWindow().webContents.send('log', msg) 53 | console.log("FTP:: " + msg) 54 | }, 55 | 56 | loading(msg=null){ 57 | this.getWindow().webContents.send('loading', msg) 58 | console.log("Loading:: " + JSON.stringify(msg)) 59 | }, 60 | 61 | async getClient(){ 62 | const client = new Client(1000) 63 | client.ftp.verbose = true 64 | 65 | try { 66 | this.log("Connecting to ps4 though ftp " + this.ip + ':' + this.port) 67 | await client.access({ 68 | host: this.ip, 69 | port: this.port, 70 | user: "anonymous", 71 | password: "anonymous", 72 | secureOptions: { 73 | rejectUnauthorized: false, 74 | } 75 | }) 76 | 77 | // client.trackProgress(info => { 78 | // console.log("File", info.name) 79 | // console.log("Type", info.type) 80 | // console.log("Transferred", info.bytes) 81 | // console.log("Transferred Overall", info.bytesOverall) 82 | // }) 83 | } 84 | catch(err) { 85 | this.error("Cannot connect to PS4") 86 | throw err 87 | return 88 | } 89 | 90 | return client 91 | }, 92 | 93 | async download(target, source){ 94 | let client = await this.getClient() 95 | let size = await client.size(source) 96 | let dir = path.dirname(target) 97 | 98 | if (!fs.existsSync(dir)) { 99 | fs.mkdirSync(dir); 100 | } 101 | 102 | if(size === 0){ 103 | console.log(source, size) 104 | 105 | // double check for backward compatibility 106 | source = source.replace('store.log', 'log.txt') 107 | size = await client.size(source) 108 | 109 | if(size == 0){ 110 | throw path.basename(source) + " not found or empty" 111 | client.close() 112 | return 113 | } 114 | } 115 | 116 | let get = await client.downloadTo(target, source) 117 | client.close() 118 | return get 119 | }, 120 | 121 | async upload(source, target){ 122 | let client = await this.getClient() 123 | let get = await client.uploadFrom(source, target) 124 | client.close() 125 | return get 126 | }, 127 | 128 | async getLogs(config, log){ 129 | this.setConfig(config) 130 | 131 | let theFile = null; 132 | let theFileName = null; 133 | if(log in this.files) 134 | theFile = this.files[log] 135 | 136 | theFileName = path.basename(theFile) 137 | 138 | this.log("Trying to get logs from HB-Store ") 139 | this.loading({ message: "Trying to get " + theFileName + " from HB-Store" }) 140 | 141 | try { 142 | await this.download(this.getLocalFile(theFile), theFile) 143 | } 144 | catch(e){ 145 | console.log("DOWNLOAD ERROR FOR LOG.TXT") 146 | this.loading({ hide: true }) 147 | this.error(e) 148 | return 149 | } 150 | 151 | this.log("got " + theFileName + ", let's save it to the user space") 152 | this.loading({ message: "Loading " + theFileName + ", where should we save it?" }) 153 | 154 | let win = this.getWindow() 155 | let targetLogFile = await dialog.showSaveDialog(win, { 156 | title: "Save HB-Store " + theFileName, 157 | defaultPath: "*/" + theFileName, 158 | buttonLabel: "Save HB-Store Log", 159 | filters: [ 160 | { name: "HB-Store " + theFileName, extensions: ['log'] } 161 | ] 162 | }) 163 | 164 | this.loading({ hide: true }) 165 | if(targetLogFile.canceled) return 166 | 167 | await fs.copyFileSync(this.getLocalFile(theFile), targetLogFile.filePath) 168 | this.notify("HB-Store " + theFileName + " downloaded") 169 | }, 170 | 171 | async cleanLogs(config){ 172 | this.setConfig(config) 173 | this.loading({ message: "Cleaning HB-Store Logs"}) 174 | 175 | try { 176 | await fs.writeFileSync(this.getLocalFile(this.files.log), "===== CLEARED LOGS ======\n") 177 | } 178 | catch(e){ 179 | this.loading({ hide: true }) 180 | return this.error(e) 181 | } 182 | 183 | try { 184 | await this.upload(this.getLocalFile(this.files.log), this.files.log) 185 | } 186 | catch(e){ 187 | console.log("UPLOAD ERROR CATCH") 188 | this.loading({ hide: true }) 189 | // return this.error(e) 190 | return 191 | } 192 | 193 | this.loading({ hide: true }) 194 | this.notify("HB-Store log has been cleared") 195 | }, 196 | 197 | async getSettings(config){ 198 | this.setConfig(config) 199 | this.loading({ message: "Loading settings.ini from PS4" }) 200 | 201 | try { 202 | await this.download(this.getLocalFile(this.files.settings), this.files.settings) 203 | } 204 | catch(e){ 205 | this.loading({ hide: true }) 206 | this.error(e) 207 | console.log("ERROR IN GET SETTINGS.INI") 208 | throw e 209 | } 210 | 211 | this.loading({ hide: true }) 212 | }, 213 | 214 | async updateSettings(config){ 215 | try { 216 | await this.getSettings(config) 217 | } 218 | catch(e){ return } 219 | 220 | console.log("CONTINUE IN UPDATE SETTING") 221 | 222 | this.setConfig(config) 223 | this.loading({ message: "Updating Settings.ini "}) 224 | 225 | let parser = new ConfigIniParser() 226 | let cdn = config.cdn 227 | 228 | // load ini 229 | try { 230 | let ini = await fs.readFileSync(this.getLocalFile(this.files.settings), 'utf8') 231 | parser.parse(ini) 232 | } 233 | catch(e){ 234 | this.loading({ hide: true }) 235 | return this.error("Error in reading settings.ini") 236 | } 237 | 238 | // update CDN 239 | parser.set('Settings', 'CDN', cdn) 240 | try { 241 | await fs.writeFileSync(this.getLocalFile(this.files.settings), parser.stringify()) 242 | } 243 | catch(e){ 244 | this.loading({ hide: true }) 245 | return this.error(e) 246 | } 247 | 248 | // upload to ftp 249 | this.loading({ message: "Uploading new settings.ini to PS4"}) 250 | try { 251 | await this.upload(this.getLocalFile(this.files.settings), this.files.settings) 252 | } 253 | catch(e){ 254 | this.loading({ hide: true }) 255 | return this.error(e) 256 | } 257 | 258 | this.loading({ hide: true }) 259 | this.notify("Update HB-Store CDN to " + cdn) 260 | }, 261 | 262 | 263 | } 264 | -------------------------------------------------------------------------------- /quasar.config.js: -------------------------------------------------------------------------------- 1 | /* eslint-env node */ 2 | 3 | /* 4 | * This file runs in a Node context (it's NOT transpiled by Babel), so use only 5 | * the ES6 features that are supported by your Node version. https://node.green/ 6 | */ 7 | 8 | // Configuration for your app 9 | // https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js 10 | const path = require('path'); 11 | const fs = require('fs'); 12 | const webpack = require('webpack') 13 | const { configure } = require('quasar/wrappers'); 14 | 15 | module.exports = configure(function (ctx) { 16 | return { 17 | target: 'electron-renderer', 18 | // https://v2.quasar.dev/quasar-cli-webpack/supporting-ts 19 | supportTS: false, 20 | 21 | // https://v2.quasar.dev/quasar-cli-webpack/prefetch-feature 22 | // preFetch: true, 23 | 24 | // app boot file (/src/boot) 25 | // --> boot files are part of "main.js" 26 | // https://v2.quasar.dev/quasar-cli-webpack/boot-files 27 | boot: [ 28 | 'i18n', 29 | 'axios', 30 | 'components', 31 | 'plugins', 32 | ], 33 | 34 | // https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js#Property%3A-css 35 | css: [ 36 | 'app.scss' 37 | ], 38 | 39 | // https://github.com/quasarframework/quasar/tree/dev/extras 40 | extras: [ 41 | // 'ionicons-v4', 42 | // 'mdi-v5', 43 | 'fontawesome-v5', 44 | // 'eva-icons', 45 | // 'themify', 46 | // 'line-awesome', 47 | // 'roboto-font-latin-ext', // this or either 'roboto-font', NEVER both! 48 | 49 | 'roboto-font', // optional, you are not bound to it 50 | 'material-icons', // optional, you are not bound to it 51 | ], 52 | 53 | // Full list of options: https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js#Property%3A-build 54 | build: { 55 | vueRouterMode: 'hash', // available values: 'hash', 'history' 56 | 57 | // transpile: false, 58 | // publicPath: '/', 59 | 60 | // Add dependencies for transpiling with Babel (Array of string/regex) 61 | // (from node_modules, which are by default not transpiled). 62 | // Applies only if "transpile" is set to true. 63 | // transpileDependencies: [], 64 | 65 | // rtl: true, // https://quasar.dev/options/rtl-support 66 | // preloadChunks: true, 67 | // showProgress: false, 68 | // gzip: true, 69 | // analyze: true, 70 | 71 | // Options below are automatically set depending on the env, set them if you want to override 72 | // extractCSS: false, 73 | 74 | // https://v2.quasar.dev/quasar-cli-webpack/handling-webpack 75 | // "chain" is a webpack-chain object https://github.com/neutrinojs/webpack-chain 76 | 77 | env: { 78 | version: require(__dirname + '/package.json').version, 79 | }, 80 | 81 | chainWebpack (chain){ 82 | const nodePolyfillWebpackPlugin = require('node-polyfill-webpack-plugin') 83 | chain.plugin('node-polyfill').use(nodePolyfillWebpackPlugin) 84 | }, 85 | 86 | extendWebpack (cfg) { 87 | cfg.resolve.alias = { 88 | ...cfg.resolve.alias, 89 | '~': path.resolve(__dirname, './src'), 90 | 'public': path.resolve(__dirname, './public'), 91 | } 92 | 93 | cfg.plugins.push( 94 | // new webpack.IgnorePlugin(/^\.\/locale$/, /moment$/), 95 | ) 96 | 97 | // console.log(cfg.optimization) 98 | // console.log(cfg.optimization.splitChunks.cacheGroups) 99 | }, 100 | }, 101 | 102 | // Full list of options: https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js#Property%3A-devServer 103 | devServer: { 104 | server: { 105 | type: 'http' 106 | }, 107 | port: 8080, 108 | open: true // opens browser window automatically 109 | }, 110 | 111 | // https://v2.quasar.dev/quasar-cli-webpack/quasar-config-js#Property%3A-framework 112 | framework: { 113 | config: {}, 114 | 115 | // iconSet: 'material-icons', // Quasar icon set 116 | // lang: 'en-US', // Quasar language pack 117 | 118 | // For special cases outside of where the auto-import strategy can have an impact 119 | // (like functional components as one of the examples), 120 | // you can manually specify Quasar components/directives to be available everywhere: 121 | // 122 | // components: [], 123 | // directives: [], 124 | 125 | // Quasar plugins 126 | plugins: [ 127 | 'Loading', 128 | 'Dialog', 129 | 'Notify', 130 | ] 131 | }, 132 | 133 | // animations: 'all', // --- includes all animations 134 | // https://quasar.dev/options/animations 135 | animations: [], 136 | 137 | // https://v2.quasar.dev/quasar-cli-webpack/developing-ssr/configuring-ssr 138 | ssr: { 139 | pwa: false, 140 | 141 | // manualStoreHydration: true, 142 | // manualPostHydrationTrigger: true, 143 | 144 | prodPort: 3000, // The default port that the production server should use 145 | // (gets superseded if process.env.PORT is specified at runtime) 146 | 147 | maxAge: 1000 * 60 * 60 * 24 * 30, 148 | // Tell browser when a file from the server should expire from cache (in ms) 149 | 150 | 151 | chainWebpackWebserver (/* chain */) {}, 152 | 153 | 154 | middlewares: [ 155 | ctx.prod ? 'compression' : '', 156 | 'render' // keep this as last one 157 | ] 158 | }, 159 | 160 | // https://v2.quasar.dev/quasar-cli-webpack/developing-pwa/configuring-pwa 161 | pwa: { 162 | workboxPluginMode: 'GenerateSW', // 'GenerateSW' or 'InjectManifest' 163 | workboxOptions: {}, // only for GenerateSW 164 | 165 | // for the custom service worker ONLY (/src-pwa/custom-service-worker.[js|ts]) 166 | // if using workbox in InjectManifest mode 167 | 168 | chainWebpackCustomSW (/* chain */) {}, 169 | 170 | 171 | manifest: { 172 | name: `hb-store-cdn-server`, 173 | short_name: `hb-store-cdn-server`, 174 | description: `Local HB-Store Server`, 175 | display: 'standalone', 176 | orientation: 'portrait', 177 | background_color: '#ffffff', 178 | theme_color: '#027be3', 179 | icons: [ 180 | { 181 | src: 'icons/icon-128x128.png', 182 | sizes: '128x128', 183 | type: 'image/png' 184 | }, 185 | { 186 | src: 'icons/icon-192x192.png', 187 | sizes: '192x192', 188 | type: 'image/png' 189 | }, 190 | { 191 | src: 'icons/icon-256x256.png', 192 | sizes: '256x256', 193 | type: 'image/png' 194 | }, 195 | { 196 | src: 'icons/icon-384x384.png', 197 | sizes: '384x384', 198 | type: 'image/png' 199 | }, 200 | { 201 | src: 'icons/icon-512x512.png', 202 | sizes: '512x512', 203 | type: 'image/png' 204 | } 205 | ] 206 | } 207 | }, 208 | 209 | // Full list of options: https://v2.quasar.dev/quasar-cli-webpack/developing-cordova-apps/configuring-cordova 210 | cordova: { 211 | // noIosLegacyBuildFlag: true, // uncomment only if you know what you are doing 212 | }, 213 | 214 | // Full list of options: https://v2.quasar.dev/quasar-cli-webpack/developing-capacitor-apps/configuring-capacitor 215 | capacitor: { 216 | hideSplashscreen: true 217 | }, 218 | 219 | // Full list of options: https://v2.quasar.dev/quasar-cli-webpack/developing-electron-apps/configuring-electron 220 | electron: { 221 | bundler: 'builder', // 'packager' or 'builder' 222 | 223 | packager: { 224 | // https://github.com/electron-userland/electron-packager/blob/master/docs/api.md#options 225 | 226 | // OS X / Mac App Store 227 | // appBundleId: '', 228 | // appCategoryType: '', 229 | // osxSign: '', 230 | // protocol: 'myapp://path', 231 | 232 | // Windows only 233 | // win32metadata: { ... } 234 | }, 235 | 236 | builder: { 237 | // https://www.electron.build/configuration/configuration 238 | appId: 'gkiokan.net.hb-store-cdn-server', 239 | directories: { 240 | output: "release" 241 | }, 242 | publish: { 243 | "provider": "github", 244 | "owner": "gkiokan", 245 | "repo": "hb-store-cdn-server", 246 | "vPrefixedTagName": true 247 | }, 248 | "win": { 249 | "target": [ 250 | { 251 | "target": "zip", 252 | "arch": [ "x64", "ia32" ] 253 | }, 254 | { 255 | "target": "portable", 256 | "arch": [ "x64", "ia32" ] 257 | } 258 | ] 259 | }, 260 | "linux": { 261 | "icon": "icons/icon.png", 262 | "target": [ 263 | { 264 | "target": "AppImage", 265 | "arch": [ "x64" ] 266 | }, 267 | { 268 | "target": "snap" 269 | }, 270 | { 271 | "target": "deb" 272 | } 273 | ] 274 | }, 275 | "portable": { 276 | "artifactName": "${productName}.exe", 277 | "unpackDirName": "HBStoreCDNServer" 278 | } 279 | }, 280 | 281 | // "chain" is a webpack-chain object https://github.com/neutrinojs/webpack-chain 282 | 283 | chainWebpackMain (/* chain */) {}, 284 | 285 | 286 | 287 | chainWebpackPreload (/* chain */) {}, 288 | 289 | } 290 | } 291 | }); 292 | -------------------------------------------------------------------------------- /src-electron/src/server.js: -------------------------------------------------------------------------------- 1 | import { app, BrowserWindow } from 'electron' 2 | 3 | import express from 'express' 4 | import http from 'http' 5 | import fg from 'fast-glob' 6 | import path from 'path' 7 | import hb from './hb' 8 | import db from './db' 9 | import pkgInfo from 'ps4-pkg-info' 10 | import { getPs4PkgInfo } from "@njzy/ps4-pkg-info" 11 | import md5File from 'md5-file' 12 | import normalize from 'normalize-path' 13 | 14 | export default { 15 | ip: null, 16 | port: null, 17 | basePath: null, 18 | files: [], 19 | host: { 20 | app: null, 21 | server: null, 22 | router: null, 23 | }, 24 | 25 | getWindow(){ 26 | let win = BrowserWindow.getFocusedWindow(); 27 | 28 | if(!win){ 29 | let all = BrowserWindow.getAllWindows() 30 | win = all[0] 31 | } 32 | 33 | return win 34 | }, 35 | 36 | getBaseURI(){ 37 | return 'http://' + this.ip + ':' + this.port 38 | }, 39 | 40 | setConfig(config){ 41 | this.ip = config.ip 42 | this.port = config.port 43 | this.basePath = config.basePath 44 | }, 45 | 46 | error(err=null){ 47 | // deprecated 48 | // const win = BrowserWindow.getFocusedWindow(); 49 | this.getWindow().webContents.send('error', err) 50 | this.log(err) 51 | }, 52 | 53 | log(msg=null){ 54 | this.getWindow().webContents.send('log', msg) 55 | console.log("Server:: " + msg) 56 | }, 57 | 58 | notify(msg=null){ 59 | this.getWindow().webContents.send('notify', msg) 60 | this.log(msg) 61 | }, 62 | 63 | sendFiles(){ 64 | this.getWindow().webContents.send('server-files', this.files) 65 | }, 66 | 67 | setState(state=null){ 68 | this.getWindow().send('server-state', state) 69 | this.log("Set Server State to " + state) 70 | }, 71 | 72 | updatePS4IP(ip){ 73 | this.getWindow().send('update-ps4-ip', ip) 74 | this.log("I guess we have a ps4 IP here " + ip) 75 | }, 76 | 77 | addCORSHandler(){ 78 | this.host.app.use((req, res, next) => { 79 | res.setHeader('Access-Control-Allow-Origin', '*') 80 | res.setHeader('Access-Control-Allow-Methods', 'GET, POST, OPTIONS, PUT, PATCH, DELETE'); 81 | // res.setHeader('Access-Control-Allow-Headers', 'X-Requested-With,content-type'); 82 | // res.setHeader('Access-Control-Allow-Credentials', true); 83 | next() 84 | }) 85 | }, 86 | 87 | addRouterMiddleware(){ 88 | this.host.app.use((req, res, next) => { 89 | this.host.router(req, res, next) 90 | }) 91 | }, 92 | 93 | createPaths(){ 94 | this.log("Server is ready to create paths") 95 | db.renewDB() 96 | this.files = [] 97 | this.host.router = new express.Router() 98 | this.addHearthbeatEndpoint() 99 | this.addFilesFromBasePath() 100 | }, 101 | 102 | async rescanFolder(config){ 103 | console.log("Trigger re-scan") 104 | this.setConfig(config) 105 | this.createPaths() 106 | this.notify("Re-scaned BasePath") 107 | }, 108 | 109 | addHearthbeatEndpoint(){ 110 | this.log("Create Hearthbeat endpoint") 111 | this.host.router.get('/hb', function(request, response){ 112 | response.status(200).json({ 113 | remoteAddress: request.connection.remoteAddress, 114 | remotePort: request.connection.remotePort, 115 | localAddress: request.connection.localAddress, 116 | localPort: request.connection.localPort, 117 | message: "Hearthbeat of HB-Store CDN Server is working" 118 | }) 119 | }) 120 | 121 | // sample icon0.png 122 | this.host.router.get('/icon0.png', function(request, response){ 123 | let image = path.resolve(__dirname, process.env.QUASAR_PUBLIC_FOLDER) + '/icon0.png' 124 | response.status(200).download(image, 'icon0.png') 125 | }) 126 | 127 | // storage database 128 | this.host.router.get('/store.db', (request, response) => { 129 | console.log("HB-Store Download store.db Request", request) 130 | console.log("PS4 IP", request.ip ) 131 | 132 | // for local tests this throws 133 | try { 134 | var r = /\b\d{1,3}\.\d{1,3}\.\d{1,3}\.\d{1,3}\b/ 135 | let ip = request.ip 136 | let cleanedIP = ip.match(r)[0] 137 | this.updatePS4IP(cleanedIP) 138 | } 139 | catch( e ){ 140 | 141 | } 142 | 143 | let store = db.getStorePath() 144 | console.log(store) 145 | response.status(200).download(store, 'store.db') 146 | }) 147 | 148 | // check the storage checksum 149 | this.host.router.get('/api.php', function(request, response){ 150 | if('db_check_hash' in request.query){ 151 | let hash = md5File.sync(db.getStorePath()) 152 | response.status(200).json({ 153 | hash, 154 | params: request.query, 155 | }) 156 | } 157 | }) 158 | 159 | // number of downloads? 160 | this.host.router.get('/download.php', function(request, response){ 161 | response.status(200).json({ 162 | number_of_downloads: "1337", 163 | }) 164 | }) 165 | 166 | // load server binaries 167 | for (const asset of [ 168 | 'homebrew.elf', 169 | 'homebrew.elf.sig', 170 | 'remote.md5', 171 | 'store.prx', 172 | 'store.prx.sig', 173 | ]) 174 | this.host.router.get('/update/' + asset, function(request, response){ 175 | let file = app.getPath('userData') + '/bin/' + asset 176 | response.status(200).download(file, asset) 177 | }) 178 | }, 179 | 180 | async addFilesFromBasePath(){ 181 | this.log("Search for pkg files in basePath at " + this.basePath) 182 | let patchedBasePath = normalize(this.basePath) 183 | let toRemoveBasePath = (patchedBasePath.charAt(0) == "/") ? patchedBasePath.substr(1).replace(/[^a-zA-Z0-9-_./]/g, '') : patchedBasePath.replace(/[^a-zA-Z0-9-_./]/g, '') 184 | 185 | let files = fg.sync([patchedBasePath + '/**/*.pkg']) 186 | this.log("Found " + files.length + " files in basePath") 187 | 188 | // loop for files and map the files to a file object 189 | let base = this.getBaseURI() 190 | let i = 1 191 | for (const file of files){ 192 | // console.log("Start file ", file) 193 | try { 194 | // let data = await pkgInfo.extract(file) 195 | let data = await getPs4PkgInfo(file, { generateBase64Icon: true }) 196 | .catch( e => { 197 | this.error("Error in PKG Extraction: "+ e + '; File: ' + file) 198 | throw e 199 | }) 200 | // console.log(data) 201 | let item = hb.createItem(data, file, i) 202 | item = hb.removeBasePath(item, toRemoveBasePath) 203 | item = hb.addImages(item, base) 204 | item = this.addFileEndpoint(item, base) 205 | 206 | this.files.push(item) 207 | // console.log(item) 208 | i = i+1 209 | } 210 | catch(e){ 211 | console.log("Error", e) 212 | } 213 | 214 | // console.log("End file ", file) 215 | // console.log("====") 216 | } 217 | 218 | db.addAllItems(this.files) 219 | // console.log("=====================================") 220 | // console.log("patched file 0 ", this.files[0] ) 221 | // console.log("=====================================") 222 | 223 | this.sendFiles() 224 | }, 225 | 226 | addFileEndpoint(item, base){ 227 | this.host.router.get(`/${item.patchedFilename}`, function(request, response){ 228 | response.status(200).download(item.path, item.filename) 229 | }) 230 | 231 | this.host.router.get(`/${item.patchedFilename}/icon0.png`, function(request, response){ 232 | let imgData = item.icon0.replace(/^data:image\/png;base64,/, ''); 233 | let img = Buffer.from(imgData, 'base64') 234 | 235 | response.writeHead(200, { 236 | 'Content-Type': 'image/png', 237 | 'Content-Length': img.length 238 | }) 239 | 240 | response.end(img) 241 | }) 242 | 243 | item.package = base + '/' + item.patchedFilename 244 | 245 | return item 246 | }, 247 | 248 | createServer(){ 249 | const app = express(); 250 | this.host.app = app 251 | this.host.router = express.Router() 252 | this.log("Server created") 253 | }, 254 | 255 | async start(config){ 256 | this.setConfig(config) 257 | 258 | if(!this.host.app){ 259 | this.createServer() 260 | } 261 | 262 | // console.log(this.ip, this.ip.length, this.port, this.port.length) 263 | if(this.ip.length == 0 || this.port.length == 0){ 264 | this.error("Server cannot start. Please configure IP and Port") 265 | // this.$message({ type: 'warning', message: error }); 266 | return 267 | } 268 | 269 | this.host.server = await this.host.app.listen(this.port, () => { 270 | this.notify('Server is running on ' + this.ip + ' at port ' + this.port) 271 | this.setState('running') 272 | 273 | this.addCORSHandler() 274 | this.addRouterMiddleware() 275 | this.createPaths() 276 | }) 277 | .on('error', (e) => { 278 | // console.log({ ...e }) 279 | this.setState('stopped') 280 | 281 | if(e.code === 'EADDRINUSE'){ 282 | let error = "Port " + this.port + " is already in use.
Choose another port and restart the Server" 283 | this.error(error) 284 | } 285 | else { 286 | this.error('Error in listening on ' + this.ip + ' at port ' + this.port + ". Error: " + e.code) 287 | } 288 | }) 289 | }, 290 | 291 | async stop(){ 292 | this.log('Closing Server') 293 | 294 | if(this.host.server) 295 | await this.host.server.close(() => { 296 | this.log('Server closed') 297 | this.setState('stopped') 298 | }) 299 | else 300 | this.error("Server can not be closed. Server Object does't exist") 301 | }, 302 | 303 | restart(config){ 304 | this.log("Server restarting triggered") 305 | this.stop() 306 | this.start(config) 307 | }, 308 | 309 | } 310 | -------------------------------------------------------------------------------- /public/kofi.svg: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | Artboard 5 | Created with Sketch. 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 | -------------------------------------------------------------------------------- /src-electron/src/baseImage.js: -------------------------------------------------------------------------------- 1 | export default "" 2 | --------------------------------------------------------------------------------