mirror of
https://github.com/outline/outline.git
synced 2026-04-24 11:20:11 -05:00
1285efc49a
* feat: i18n
* Changing language single source of truth from TEAM to USER
* Changes according to @tommoor comments on PR
* Changed package.json for build:i18n and translation label
* Finished 1st MVP of i18n for outline
* new translation labels & Portuguese from Portugal translation
* Fixes from PR request
* Described language dropdown as an experimental feature
* Set keySeparator to false in order to cowork with html keys
* Added useTranslation to Breadcrumb
* Repositioned <strong> element
* Removed extra space from TemplatesMenu
* Fortified the test suite for i18n
* Fixed trans component problematic
* Check if selected language is available
* Update yarn.lock
* Removed unused Trans
* Removing debug variable from i18n init
* Removed debug variable
* test: update snapshots
* flow: Remove decorator usage to get proper flow typing
It's a shame, but hopefully we'll move to Typescript in the next 6 months and we can forget this whole Flow mistake ever happened
* translate: Drafts
* More translatable strings
* Mo translation strings
* translation: Search
* async translations loading
* cache translations in client
* Revert "cache translations in client"
This reverts commit 08fb61ce36.
* Revert localStorage cache for cache headers
* Update Crowdin configuration file
* Moved translation files to locales folder and fixed english text
* Added CONTRIBUTING File for CrowdIn
* chore: Move translations again to please CrowdIn
* fix: loading paths
chore: Add strings for editor
* fix: Improve validation on documents.import endpoint
* test: mock bull
* fix: Unknown mimetype should fallback to Markdown parsing if markdown extension (#1678)
* closes #1675
* Update CONTRIBUTING
* chore: Add link to translation portal from app UI
* refactor: Centralize language config
* fix: Ensure creation of i18n directory in build
* feat: Add language prompt
* chore: Improve contributing guidelines, add link from README
* chore: Normalize tab header casing
* chore: More string externalization
* fix: Language prompt in dark mode
Co-authored-by: André Glatzl <andreglatzl@gmail.com>
118 lines
2.8 KiB
JavaScript
118 lines
2.8 KiB
JavaScript
// @flow
|
|
import fs from "fs";
|
|
import path from "path";
|
|
import util from "util";
|
|
import Koa from "koa";
|
|
import Router from "koa-router";
|
|
import sendfile from "koa-sendfile";
|
|
import serve from "koa-static";
|
|
import { languages } from "../shared/i18n";
|
|
import environment from "./env";
|
|
import { NotFoundError } from "./errors";
|
|
import apexRedirect from "./middlewares/apexRedirect";
|
|
import { opensearchResponse } from "./utils/opensearch";
|
|
import { robotsResponse } from "./utils/robots";
|
|
|
|
const isProduction = process.env.NODE_ENV === "production";
|
|
const koa = new Koa();
|
|
const router = new Router();
|
|
const readFile = util.promisify(fs.readFile);
|
|
|
|
const readIndexFile = async (ctx) => {
|
|
if (isProduction) {
|
|
return readFile(path.join(__dirname, "../app/index.html"));
|
|
}
|
|
|
|
const middleware = ctx.devMiddleware;
|
|
await new Promise((resolve) => middleware.waitUntilValid(resolve));
|
|
|
|
return new Promise((resolve, reject) => {
|
|
middleware.fileSystem.readFile(
|
|
`${ctx.webpackConfig.output.path}/index.html`,
|
|
(err, result) => {
|
|
if (err) {
|
|
return reject(err);
|
|
}
|
|
resolve(result);
|
|
}
|
|
);
|
|
});
|
|
};
|
|
|
|
const renderApp = async (ctx, next) => {
|
|
if (ctx.request.path === "/realtime/") {
|
|
return next();
|
|
}
|
|
|
|
const page = await readIndexFile(ctx);
|
|
const env = `
|
|
window.env = ${JSON.stringify(environment)};
|
|
`;
|
|
ctx.body = page
|
|
.toString()
|
|
.replace(/\/\/inject-env\/\//g, env)
|
|
.replace(/\/\/inject-sentry-dsn\/\//g, process.env.SENTRY_DSN || "")
|
|
.replace(/\/\/inject-slack-app-id\/\//g, process.env.SLACK_APP_ID || "");
|
|
};
|
|
|
|
// serve static assets
|
|
koa.use(
|
|
serve(path.resolve(__dirname, "../../public"), {
|
|
maxage: 60 * 60 * 24 * 30 * 1000,
|
|
})
|
|
);
|
|
|
|
router.get("/_health", (ctx) => (ctx.body = "OK"));
|
|
|
|
if (process.env.NODE_ENV === "production") {
|
|
router.get("/static/*", async (ctx) => {
|
|
ctx.set({
|
|
"Cache-Control": `max-age=${356 * 24 * 60 * 60}`,
|
|
});
|
|
|
|
await sendfile(ctx, path.join(__dirname, "../app/", ctx.path.substring(8)));
|
|
});
|
|
}
|
|
|
|
router.get("/locales/:lng.json", async (ctx) => {
|
|
let { lng } = ctx.params;
|
|
|
|
if (!languages.includes(lng)) {
|
|
throw new NotFoundError();
|
|
}
|
|
|
|
if (process.env.NODE_ENV === "production") {
|
|
ctx.set({
|
|
"Cache-Control": `max-age=${7 * 24 * 60 * 60}`,
|
|
});
|
|
}
|
|
|
|
await sendfile(
|
|
ctx,
|
|
path.join(__dirname, "../shared/i18n/locales", lng, "translation.json")
|
|
);
|
|
});
|
|
|
|
router.get("/robots.txt", (ctx) => {
|
|
ctx.body = robotsResponse(ctx);
|
|
});
|
|
|
|
router.get("/opensearch.xml", (ctx) => {
|
|
ctx.type = "text/xml";
|
|
ctx.body = opensearchResponse();
|
|
});
|
|
|
|
router.get("/share/*", (ctx, next) => {
|
|
ctx.remove("X-Frame-Options");
|
|
return renderApp(ctx, next);
|
|
});
|
|
|
|
// catch all for application
|
|
router.get("*", renderApp);
|
|
|
|
// middleware
|
|
koa.use(apexRedirect());
|
|
koa.use(router.routes());
|
|
|
|
export default koa;
|