Skip to content

Commit

Permalink
chore: added frontend proxy, added translation logic for weblate to i18n
Browse files Browse the repository at this point in the history
  • Loading branch information
web-mi committed Jan 19, 2024
1 parent 8c100d5 commit 40f2679
Show file tree
Hide file tree
Showing 47 changed files with 6,761 additions and 600 deletions.
32 changes: 31 additions & 1 deletion .env.sample
Original file line number Diff line number Diff line change
Expand Up @@ -21,4 +21,34 @@ CSRF_WHITELIST_HEADER_FOR_LOCAL_DEVELOPMENT=X-WHITELIST-HEADER
#AUDIO_FILE_INCOMING_NOTIFICATION=/assets/audio/incomingNotification.mp3

# Disable 2FA
REACT_APP_DISABLE_2FA_DUTY=0
REACT_APP_DISABLE_2FA_DUTY=0

### Weblate
# Weblate host
REACT_APP_WEBLATE_HOST=
# Path to api
REACT_APP_WEBLATE_PATH=/weblate/api
# Name of weblate project
REACT_APP_WEBLATE_PROJECT=
# Weblate api key - Attention! if provided it will be used for frontend requests and is visible to everyone
REACT_APP_WEBLATE_API_KEY=
# Min percentage - percentage which must be translated in one language until it gets visible in the frontend
REACT_APP_WEBLATE_MIN_PERCENT=
# Disable localStorage caching
REACT_APP_TRANSLATION_DISABLE_CACHE=
# Time until frontend cache gets invalidated in minutes
REACT_APP_TRANSLATION_CACHE_TIME=

### NodeJS
# Storage path - if set relative to project root translation files will be stored temporary until LOCALIZATION_CACHE_TIME expired
STORAGE_PATH=./.storage

# Localize cache time - in minutes - default 120 minutes
LOCALIZATION_CACHE_TIME=

# Weblate host for proxy
LOCALIZATION_WEBLATE_HOST=https://weblate.example.com/
# If weblate is provided under subdirectory
LOCALIZATION_WEBLATE_PATH=/weblate
# Weblate api key - must be provided to skip request limits
LOCALIZATION_WEBLATE_API_KEY=
3 changes: 3 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

# dependencies
/node_modules
/proxy/node_modules
/.pnp
.pnp.js

Expand All @@ -12,6 +13,7 @@

# production
/build
/dist

# misc
.DS_Store
Expand All @@ -22,6 +24,7 @@
.env.test.local
.env.production.local
.idea
.storage

# cypress
browserstack.json
Expand Down
44 changes: 40 additions & 4 deletions Dockerfile
Original file line number Diff line number Diff line change
@@ -1,4 +1,40 @@
ARG DOCKER_MATRIX=ghcr.io
FROM $DOCKER_MATRIX/onlineberatung/onlineberatung-nginx/onlineberatung-nginx:dockerimage.v.005-main
COPY build /usr/share/nginx/html
COPY nginx.conf /etc/nginx/conf.d/default.conf
ARG NODE_VERSION=14.21.1
ARG PORT=80

FROM node:$NODE_VERSION as proxyBuild

WORKDIR /app
COPY proxy /app

ENV NODE_ENV=development
ENV PORT=$PORT

# Currently nothing to build inside
# RUN npm run install
# RUN npm run build
# RUN rm /app/node_modules

FROM node:$NODE_VERSION as frontendBuild

WORKDIR /app
COPY . /app

ENV NODE_ENV=development
ENV PORT=$PORT

RUN npm install
RUN npm run build

# Prod build
FROM node:$NODE_VERSION
WORKDIR /app
EXPOSE $PORT
COPY --from=proxyBuild /app ./
COPY --from=frontendBuild /app/build ./build

ENV NODE_ENV=production
ENV PORT=$PORT

RUN npm install

CMD ["npm", "run", "start"]
5 changes: 1 addition & 4 deletions config/env.js
Original file line number Diff line number Diff line change
Expand Up @@ -84,10 +84,7 @@ function getClientEnvironment(publicUrl) {
// and `sockPort` options in webpack-dev-server.
WDS_SOCKET_HOST: process.env.WDS_SOCKET_HOST,
WDS_SOCKET_PATH: process.env.WDS_SOCKET_PATH,
WDS_SOCKET_PORT: process.env.WDS_SOCKET_PORT,
// Whether or not react-refresh is enabled.
// It is defined here so it is available in the webpackHotDevClient.
FAST_REFRESH: process.env.FAST_REFRESH !== 'false'
WDS_SOCKET_PORT: process.env.WDS_SOCKET_PORT
}
);
// Stringify all values so we can feed into webpack DefinePlugin
Expand Down
3 changes: 3 additions & 0 deletions config/paths.js
Original file line number Diff line number Diff line change
Expand Up @@ -72,6 +72,9 @@ module.exports = {
appNodeModules: resolveApp('node_modules'),
appWebpackCache: resolveApp('node_modules/.cache'),
appTsBuildInfoFile: resolveApp('node_modules/.cache/tsconfig.tsbuildinfo'),
storagePath: process.env.STORAGE_PATH
? resolveApp(process.env.STORAGE_PATH)
: null,
swSrc: resolveModule(resolveApp, 'src/service-worker'),
publicUrlOrPath
};
Expand Down
133 changes: 29 additions & 104 deletions config/webpack.config.js
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,6 @@ const InlineChunkHtmlPlugin = require('react-dev-utils/InlineChunkHtmlPlugin');
const TerserPlugin = require('terser-webpack-plugin');
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
const CssMinimizerPlugin = require('css-minimizer-webpack-plugin');
const { WebpackManifestPlugin } = require('webpack-manifest-plugin');
const InterpolateHtmlPlugin = require('react-dev-utils/InterpolateHtmlPlugin');
const WorkboxWebpackPlugin = require('workbox-webpack-plugin');
const ModuleScopePlugin = require('react-dev-utils/ModuleScopePlugin');
Expand Down Expand Up @@ -59,11 +58,6 @@ const imageInlineSizeLimit = parseInt(
// Check if TypeScript is setup
const useTypeScript = fs.existsSync(paths.appTsConfig);

// Check if Tailwind config exists
const useTailwind = fs.existsSync(
path.join(paths.appPath, 'tailwind.config.js')
);

// Get the path to the uncompiled service worker (if it exists).
const swSrc = paths.swSrc;

Expand Down Expand Up @@ -134,8 +128,6 @@ module.exports = function (webpackEnv) {
// Get environment variables to inject into our app.
const env = getClientEnvironment(paths.publicUrlOrPath.slice(0, -1));

const shouldUseReactRefresh = env.raw.FAST_REFRESH;

// common function to get style loaders
const getStyleLoaders = (cssOptions, preProcessor) => {
const loaders = [
Expand Down Expand Up @@ -164,36 +156,22 @@ module.exports = function (webpackEnv) {
// https://github.com/facebook/create-react-app/issues/2677
ident: 'postcss',
config: false,
plugins: !useTailwind
? [
'postcss-flexbugs-fixes',
[
'postcss-preset-env',
{
autoprefixer: {
flexbox: 'no-2009'
},
stage: 3
}
],
// Adds PostCSS Normalize as the reset css with default options,
// so that it honors browserslist config in package.json
// which in turn let's users customize the target behavior as per their needs.
'postcss-normalize'
]
: [
'tailwindcss',
'postcss-flexbugs-fixes',
[
'postcss-preset-env',
{
autoprefixer: {
flexbox: 'no-2009'
},
stage: 3
}
]
]
plugins: [
'postcss-flexbugs-fixes',
[
'postcss-preset-env',
{
autoprefixer: {
flexbox: 'no-2009'
},
stage: 3
}
],
// Adds PostCSS Normalize as the reset css with default options,
// so that it honors browserslist config in package.json
// which in turn let's users customize the target behavior as per their needs.
'postcss-normalize'
]
},
sourceMap: isEnvProduction
? shouldUseSourceMap
Expand Down Expand Up @@ -499,7 +477,6 @@ module.exports = function (webpackEnv) {

plugins: [
isEnvDevelopment &&
shouldUseReactRefresh &&
require.resolve('react-refresh/babel')
].filter(Boolean),
// This is a feature of `babel-loader` for webpack (not Babel itself).
Expand Down Expand Up @@ -657,46 +634,19 @@ module.exports = function (webpackEnv) {
chunks: ['app'],
filename: 'beratung-hilfe.html'
}),
new HtmlWebpackPlugin({
title: 'Beratung & Hilfe',
templateParameters: {
type: 'error',
errorType: '400'
},
template: getTemplate('pages/app.html'),
chunks: ['error'],
filename: 'error.400.html'
}),
new HtmlWebpackPlugin({
title: 'Error Page 401',
templateParameters: {
type: 'error',
errorType: '401'
},
template: getTemplate('pages/app.html'),
chunks: ['error'],
filename: 'error.401.html'
}),
new HtmlWebpackPlugin({
title: 'Error Page 404',
templateParameters: {
type: 'error',
errorType: '404'
},
template: getTemplate('pages/app.html'),
chunks: ['error'],
filename: 'error.404.html'
}),
new HtmlWebpackPlugin({
title: 'Error Page 500',
templateParameters: {
type: 'error',
errorType: '500'
},
template: getTemplate('pages/app.html'),
chunks: ['error'],
filename: 'error.500.html'
}),
...['400', '401', '404', '500'].map(
(errorType) =>
new HtmlWebpackPlugin({
title: `Error Page ${errorType}`,
templateParameters: {
type: 'error',
errorType: errorType
},
template: getTemplate('pages/app.html'),
chunks: ['error'],
filename: `error.${errorType}.html`
})
),
new CopyPlugin({
patterns: [
{ from: getTemplate('pages/under-construction.html') }
Expand Down Expand Up @@ -728,7 +678,6 @@ module.exports = function (webpackEnv) {
// Experimental hot reloading for React .
// https://github.com/facebook/react/tree/main/packages/react-refresh
isEnvDevelopment &&
shouldUseReactRefresh &&
new ReactRefreshWebpackPlugin({
overlay: false
}),
Expand All @@ -745,30 +694,6 @@ module.exports = function (webpackEnv) {
'static/css/[name].[contenthash:8].chunk.css',
ignoreOrder: true // Temporary fix of imported css files
}),
// Generate an asset manifest file with the following content:
// - "files" key: Mapping of all asset filenames to their corresponding
// output file so that tools can pick it up without having to parse
// `index.html`
// - "entrypoints" key: Array of files which are included in `index.html`,
// can be used to reconstruct the HTML if necessary
// new WebpackManifestPlugin({
// fileName: 'asset-manifest.json',
// publicPath: paths.publicUrlOrPath,
// generate: (seed, files, entrypoints) => {
// const manifestFiles = files.reduce((manifest, file) => {
// manifest[file.name] = file.path;
// return manifest;
// }, seed);
// const entrypointFiles = Object.values(entrypoints).filter(
// (fileName) => !fileName.endsWith('.map')
// );

// return {
// files: manifestFiles,
// entrypoints: entrypointFiles
// };
// }
// }),
// Moment.js is an extremely popular library that bundles large locale files
// by default due to how webpack interprets its code. This is a practical
// solution that requires the user to opt into importing specific locales.
Expand Down
Loading

0 comments on commit 40f2679

Please sign in to comment.