Hangar/frontend/nuxt.config.ts

293 lines
9.0 KiB
TypeScript
Raw Normal View History

import * as fs from 'fs';
2021-01-31 10:00:11 +08:00
import { NuxtConfig } from '@nuxt/types';
2021-03-28 06:41:20 +08:00
import colors from 'vuetify/lib/util/colors';
import { LocaleMessages } from 'vue-i18n';
import { LocaleObject } from '@nuxtjs/i18n';
2021-04-04 07:53:58 +08:00
require('events').EventEmitter.defaultMaxListeners = 20;
2021-01-21 11:36:18 +08:00
require('dotenv').config();
2021-02-08 05:36:36 +08:00
const proxyHost = process.env.proxyHost || 'http://localhost:8080';
const oauthHost = process.env.oauthHost || 'http://localhost:4444';
const authHost = process.env.authHost || 'http://localhost:3001';
2021-03-16 03:31:16 +08:00
const publicHost = process.env.PUBLIC_HOST || 'http://localhost:3000';
2021-02-08 07:02:17 +08:00
const host = process.env.host || 'localhost';
2021-03-17 01:43:58 +08:00
const nodeEnv = process.env.NODE_ENV;
const publicPath = process.env.PUBLIC_PATH || '/_nuxt/';
const locales = setupLocales();
2021-06-07 05:45:56 +08:00
// noinspection JSUnusedGlobalSymbols
2021-01-21 11:36:18 +08:00
export default {
telemetry: false,
modern: 'server',
// PWA module configuration: https://go.nuxtjs.dev/pwa
pwa: {
manifest: {
name: 'Hangar | PaperMC',
short_name: 'Hangar',
description: 'Plugin repository for Paper plugins and more!',
lang: 'en',
},
workbox: {
enabled: false,
},
},
2021-01-21 11:36:18 +08:00
// Global page headers: https://go.nuxtjs.dev/config-head
head: {
2021-03-29 04:10:41 +08:00
htmlAttrs: {
dir: 'ltr',
},
titleTemplate: (titleChunk) => (titleChunk ? `${titleChunk} | Hangar` : 'Hangar'),
2021-01-21 11:36:18 +08:00
meta: [
{ hid: 'charset', charset: 'utf-8' },
2021-01-21 11:36:18 +08:00
{ name: 'viewport', content: 'width=device-width, initial-scale=1' },
{ property: 'twitter:site', name: 'twitter:site', hid: 'twitter:site', content: 'hangar' },
{ property: 'twitter:card', name: 'twitter:card', hid: 'twitter:card', content: 'summary' },
2021-01-21 11:36:18 +08:00
],
noscript: [{ innerHTML: 'We are sorry, but Hangar requires JavaScript to work properly.' }],
2021-01-21 11:36:18 +08:00
},
2021-03-17 01:43:58 +08:00
env: {
proxyHost,
oauthHost,
2021-03-17 01:43:58 +08:00
authHost,
publicHost,
host,
nodeEnv,
},
2021-01-21 11:36:18 +08:00
// Global CSS: https://go.nuxtjs.dev/config-css
2021-02-12 11:33:01 +08:00
css: ['~/assets/main.scss'],
2021-01-21 11:36:18 +08:00
// Plugins to run before rendering page: https://go.nuxtjs.dev/config-plugins
plugins: ['~/plugins/api.ts', '~/plugins/utils.ts', '~/plugins/auth.ts', '~/plugins/perms.ts', '~/plugins/seo.ts'],
2021-01-21 11:36:18 +08:00
// Auto import components: https://go.nuxtjs.dev/config-components
components: false,
2021-01-21 11:36:18 +08:00
// Modules for dev and build (recommended): https://go.nuxtjs.dev/config-modules
buildModules: [
// https://go.nuxtjs.dev/typescript
'@nuxt/typescript-build',
// https://go.nuxtjs.dev/vuetify
'@nuxtjs/vuetify',
2021-01-31 10:00:11 +08:00
// https://go.nuxtjs.dev/eslint
'@nuxtjs/eslint-module',
2021-02-06 16:29:16 +08:00
// https://go.nuxtjs.dev/pwa
'@nuxtjs/pwa',
2021-01-22 04:32:08 +08:00
'@nuxtjs/dotenv',
2021-01-21 11:36:18 +08:00
],
// Modules: https://go.nuxtjs.dev/config-modules
modules: [
// https://go.nuxtjs.dev/axios
'@nuxtjs/axios',
'cookie-universal-nuxt',
2021-01-22 03:47:58 +08:00
'@nuxtjs/proxy',
'@nuxtjs/i18n',
'@dansmaculotte/nuxt-security',
2021-01-21 11:36:18 +08:00
],
// Axios module configuration: https://go.nuxtjs.dev/config-axios
2021-01-23 13:24:48 +08:00
axios: {},
2021-01-21 11:36:18 +08:00
// Vuetify module configuration: https://go.nuxtjs.dev/config-vuetify
vuetify: {
customVariables: ['~/assets/variables.scss'],
optionsPath: '~/plugins/vuetify.ts',
2021-01-31 01:27:36 +08:00
treeShake: true,
2021-01-21 11:36:18 +08:00
},
// Build Configuration: https://go.nuxtjs.dev/config-build
2021-01-31 01:27:36 +08:00
build: {
publicPath,
transpile: ['lodash-es'],
parallel: true,
cache: true,
babel: {
presets({ envName }) {
const envTargets = {
client: { browsers: ['last 2 versions'], ie: 11 },
server: { node: 'current' },
modern: { esmodules: true },
};
return [
[
'@nuxt/babel-preset-app',
{
targets: envTargets[envName],
corejs: 3,
},
],
];
},
},
2021-01-31 01:27:36 +08:00
},
2021-01-22 03:47:58 +08:00
2021-01-22 11:11:24 +08:00
router: {
middleware: ['routePermissions'],
2021-01-22 11:11:24 +08:00
},
2021-01-31 03:55:24 +08:00
proxy: [
2021-02-08 06:06:06 +08:00
// backend
proxyHost + '/api/',
proxyHost + '/signup',
proxyHost + '/login',
proxyHost + '/logout',
proxyHost + '/handle-logout',
proxyHost + '/refresh',
proxyHost + '/invalidate',
proxyHost + '/v2/api-docs/',
2021-04-04 07:53:58 +08:00
proxyHost + '/robots.txt',
proxyHost + '/sitemap.xml',
proxyHost + '/global-sitemap.xml',
proxyHost + '/*/sitemap.xml',
2021-04-04 13:17:05 +08:00
proxyHost + '/statusz',
2021-02-08 06:06:06 +08:00
// auth
authHost + '/avatar',
authHost + '/oauth/logout',
oauthHost + '/oauth2',
2021-01-31 03:55:24 +08:00
],
2021-01-22 03:47:58 +08:00
2021-01-23 05:06:44 +08:00
i18n: {
vueI18nLoader: true,
strategy: 'no_prefix',
defaultLocale: 'en',
locales: locales.map((msgs) => {
// we have to cast here since they might be booleans
return {
code: msgs.meta.code as string,
iso: msgs.meta.iso as string,
name: msgs.meta.name as string,
icon: msgs.meta.icon as string,
} as LocaleObject;
}),
2021-01-23 05:06:44 +08:00
vueI18n: {
locale: 'en',
fallbackLocale: 'en',
messages: locales.reduce((obj, item) => {
return {
...obj,
[item.meta.code as string]: item,
};
}, {}) as LocaleMessages,
2021-01-23 05:06:44 +08:00
},
detectBrowserLanguage: {
useCookie: true,
cookieKey: 'i18n_redirected',
redirectOn: 'root',
},
2021-01-31 10:00:11 +08:00
},
2021-01-23 05:06:44 +08:00
security: {
dev: true,
hsts: {
maxAge: 15552000,
includeSubDomains: true,
preload: true,
},
2021-04-06 00:49:46 +08:00
csp: {
directives: {
defaultSrc: [
"'self'",
'https://google-analytics.com',
'https://fonts.gstatic.com',
'https://fonts.googleapis.com',
'https://static.cloudflareinsights.com',
'https://cdnjs.cloudflare.com',
],
styleSrc: ["'self'", 'https://fonts.googleapis.com', 'cdn.jsdelivr.net', "'unsafe-inline'", 'https://cdn.crowdin.com'],
fontSrc: ['fonts.gstatic.com', 'cdn.jsdelivr.net'],
scriptSrc: [
"'self'" /* , "'nonce-{nonce}'" */,
"'unsafe-inline'",
"'unsafe-eval'",
'https://static.cloudflareinsights.com',
'https://cdn.crowdin.com',
],
imgSrc: [
2021-04-06 00:49:46 +08:00
"'self'",
'https://www.google-analytics.com',
'https://www.gravatar.com',
authHost,
'data: papermc.io docs.papermc.io',
2021-04-06 00:49:46 +08:00
'https:', // ppl can use images in descriptions, we would need an image proxy or smth
],
frameSrc: ["'self'", 'http://localhost/', 'https://papermc.io/', 'https://hangar.crowdin.com', 'https://www.youtube-nocookie.com'],
manifestSrc: ["'self'"],
connectSrc: [
"'self'",
'https://www.google-analytics.com',
'https://stats.g.doubleclick.net',
'https://hangar.crowdin.com',
'http://localhost:3001',
'https://hangar-auth.benndorf.dev',
],
mediaSrc: ["'self'"],
objectSrc: ["'none'"],
baseUri: ["'none'"],
2021-04-06 00:49:46 +08:00
},
reportOnly: false,
2021-04-06 00:49:46 +08:00
},
referrer: 'same-origin',
additionalHeaders: true,
2021-04-06 00:49:46 +08:00
},
2021-01-22 03:47:58 +08:00
server: {
port: 3000,
host,
2021-01-22 03:47:58 +08:00
},
2021-01-31 02:50:12 +08:00
loading: {
2021-03-28 06:41:20 +08:00
color: colors.blue.lighten2,
2021-01-31 02:50:12 +08:00
continuous: true,
},
2021-03-16 03:31:16 +08:00
publicRuntimeConfig: {
axios: {
browserBaseURL: publicHost,
},
},
privateRuntimeConfig: {
axios: {
baseURL: proxyHost,
},
},
2021-01-31 10:00:11 +08:00
} as NuxtConfig;
function setupLocales() {
const locales = fs
.readdirSync('locales')
.filter((f) => f.endsWith('.ts'))
.map((f) => require('./locales/' + f).default);
const english = locales.find((l) => l.meta?.code === 'en');
if (!english) {
console.error("didn't find english?!");
}
locales.forEach((l) => {
if (process.env.TRANSLATION_MODE !== 'true' && english) {
strip(l, english);
}
console.log('Found locale ' + l.meta?.name + ' with example string ' + l.general?.close);
});
return locales;
}
function strip(other: any, english: any) {
for (const k in other) {
if (typeof other[k] === 'object' && other[k] !== null) {
strip(other[k], english[k]);
} else if (Object.prototype.hasOwnProperty.call(other, k)) {
if (!other[k]) {
other[k] = english[k];
}
}
}
}