├── 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 |
2 |
3 |
4 |
5 |
6 | Logs, latest on top
7 |
8 |
9 |
10 |
{{ log.time }}
11 |
12 |
13 |
14 |
15 |
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 |
2 |
3 |
4 |
5 | 404
6 |
7 |
8 |
9 | Oops. Nothing here...
10 |
11 |
12 |
21 |
22 |
23 |
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 |
2 |
3 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
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 |
2 |
8 |
12 |
13 |
14 |
15 |
16 | {{ title }}
17 | {{ caption }}
18 |
19 |
20 |
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 |
2 |
3 |
4 |
5 | HB-Store CDN Server
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
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 |
2 |
3 |
4 |
5 |
6 |
7 |
8 | List of local serving files will be here ...
9 |
10 |
11 | Found {{ files.length }} files
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
![]()
21 |
22 |
23 | {{ item.id }}
Is Backport
24 |
{{ item.name }}
25 | Version: {{ item.version }}
26 | Size: {{ item.Size }}
27 |
28 |
29 |
30 |
31 |
32 |
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 | [](https://ko-fi.com/M4M082WK8)
3 | [](#)
4 | [](#)
5 | [](#)
6 | [](#)
7 | [](#)
8 |
9 | This is the new HB-Store CDN Tool for you.
10 |
11 | 
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 | [](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 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
14 |
15 |
16 |
HB-Store CDN Server
17 | Version {{ $root.version }}
18 | Made by Gkiokan
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
HB-Store (ps4 app)
28 | Created by LightningMods
29 |
30 |
31 |
PKG-Zone
32 | Official HB-Store
33 | Owned by LightningMods
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
59 |
60 |
62 |
--------------------------------------------------------------------------------
/src/layouts/MainLayout.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
19 |
20 |
21 | Use this as CDN: {{ getCDNAdress }}
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 | Made by Gkiokan | Exclusive for HB-Store
35 |
36 |
37 | v{{ $root.version }}
38 |
39 |
40 |
41 |
42 |
43 |
44 |
72 |
73 |
106 |
--------------------------------------------------------------------------------
/src/components/HBOperations.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 | Get store.log
11 |
12 |
13 |
14 |
15 |
16 | Get loader.log
17 |
18 |
19 |
20 |
21 |
22 | Get itemzflow.log
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 | Clear store.log on PS4
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 | Update CDN to My Server
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 | Restore CDN to pkg-zone.com
55 |
56 |
57 |
58 |
59 |
60 |
61 |
62 |
63 |
64 |
65 |
114 |
115 |
117 |
--------------------------------------------------------------------------------
/src/assets/quasar-logo-vertical.svg:
--------------------------------------------------------------------------------
1 |
--------------------------------------------------------------------------------
/src/App.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
199 |
--------------------------------------------------------------------------------
/src/components/ServerBinariesDownload.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
Current Version {{ binaryVersion }}
13 |
New Version {{ newUpdateAvailableVersion }}
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 | {{ asset.name }}
22 |
23 |
24 |
25 |
26 |
28 |
29 |
30 |
31 |
32 |
33 |
172 |
173 |
175 |
--------------------------------------------------------------------------------
/src/components/Config.vue:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 |
23 |
24 |
25 |
26 |
27 |
28 |
29 |
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 |
54 |
55 |
56 |
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 |
--------------------------------------------------------------------------------
/src-electron/src/baseImage.js:
--------------------------------------------------------------------------------
1 | export default ""
2 |
--------------------------------------------------------------------------------