Feature/add endpoint to localize site.webmanifest (#4450)
* Add endpoint to localize site.webmanifest * Refactor rootUrl * Update changelog --------- Co-authored-by: Thomas Kaul <4159106+dtslvr@users.noreply.github.com>
This commit is contained in:
parent
198f73db00
commit
a9c3224856
@ -10,6 +10,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
||||
### Added
|
||||
|
||||
- Added support for filtering in the _Copy AI prompt to clipboard_ actions on the analysis page (experimental)
|
||||
- Added an endpoint to localize the `site.webmanifest`
|
||||
- Added the _Storybook_ path to the `sitemap.xml` file
|
||||
|
||||
### Changed
|
||||
|
@ -32,6 +32,7 @@ import { AuthModule } from './auth/auth.module';
|
||||
import { CacheModule } from './cache/cache.module';
|
||||
import { AiModule } from './endpoints/ai/ai.module';
|
||||
import { ApiKeysModule } from './endpoints/api-keys/api-keys.module';
|
||||
import { AssetsModule } from './endpoints/assets/assets.module';
|
||||
import { BenchmarksModule } from './endpoints/benchmarks/benchmarks.module';
|
||||
import { GhostfolioModule } from './endpoints/data-providers/ghostfolio/ghostfolio.module';
|
||||
import { MarketDataModule } from './endpoints/market-data/market-data.module';
|
||||
@ -61,6 +62,7 @@ import { UserModule } from './user/user.module';
|
||||
AiModule,
|
||||
ApiKeysModule,
|
||||
AssetModule,
|
||||
AssetsModule,
|
||||
AuthDeviceModule,
|
||||
AuthModule,
|
||||
BenchmarksModule,
|
||||
|
46
apps/api/src/app/endpoints/assets/assets.controller.ts
Normal file
46
apps/api/src/app/endpoints/assets/assets.controller.ts
Normal file
@ -0,0 +1,46 @@
|
||||
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service';
|
||||
import { interpolate } from '@ghostfolio/common/helper';
|
||||
|
||||
import {
|
||||
Controller,
|
||||
Get,
|
||||
Param,
|
||||
Res,
|
||||
Version,
|
||||
VERSION_NEUTRAL
|
||||
} from '@nestjs/common';
|
||||
import { Response } from 'express';
|
||||
import { readFileSync } from 'fs';
|
||||
import { join } from 'path';
|
||||
|
||||
@Controller('assets')
|
||||
export class AssetsController {
|
||||
private webManifest = '';
|
||||
|
||||
public constructor(
|
||||
public readonly configurationService: ConfigurationService
|
||||
) {
|
||||
try {
|
||||
this.webManifest = readFileSync(
|
||||
join(__dirname, 'assets', 'site.webmanifest'),
|
||||
'utf8'
|
||||
);
|
||||
} catch {}
|
||||
}
|
||||
|
||||
@Get('/:languageCode/site.webmanifest')
|
||||
@Version(VERSION_NEUTRAL)
|
||||
public getWebManifest(
|
||||
@Param('languageCode') languageCode: string,
|
||||
@Res() response: Response
|
||||
): void {
|
||||
const rootUrl = this.configurationService.get('ROOT_URL');
|
||||
const webManifest = interpolate(this.webManifest, {
|
||||
languageCode,
|
||||
rootUrl
|
||||
});
|
||||
|
||||
response.setHeader('Content-Type', 'application/json');
|
||||
response.send(webManifest);
|
||||
}
|
||||
}
|
11
apps/api/src/app/endpoints/assets/assets.module.ts
Normal file
11
apps/api/src/app/endpoints/assets/assets.module.ts
Normal file
@ -0,0 +1,11 @@
|
||||
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service';
|
||||
|
||||
import { Module } from '@nestjs/common';
|
||||
|
||||
import { AssetsController } from './assets.controller';
|
||||
|
||||
@Module({
|
||||
controllers: [AssetsController],
|
||||
providers: [ConfigurationService]
|
||||
})
|
||||
export class AssetsModule {}
|
@ -9,8 +9,8 @@ import { personalFinanceTools } from '@ghostfolio/common/personal-finance-tools'
|
||||
import { Controller, Get, Res, VERSION_NEUTRAL, Version } from '@nestjs/common';
|
||||
import { format } from 'date-fns';
|
||||
import { Response } from 'express';
|
||||
import * as fs from 'fs';
|
||||
import * as path from 'path';
|
||||
import { readFileSync } from 'fs';
|
||||
import { join } from 'path';
|
||||
|
||||
@Controller('sitemap.xml')
|
||||
export class SitemapController {
|
||||
@ -20,8 +20,8 @@ export class SitemapController {
|
||||
private readonly configurationService: ConfigurationService
|
||||
) {
|
||||
try {
|
||||
this.sitemapXml = fs.readFileSync(
|
||||
path.join(__dirname, 'assets', 'sitemap.xml'),
|
||||
this.sitemapXml = readFileSync(
|
||||
join(__dirname, 'assets', 'sitemap.xml'),
|
||||
'utf8'
|
||||
);
|
||||
} catch {}
|
||||
|
@ -25,7 +25,7 @@
|
||||
"name": "Ghostfolio",
|
||||
"orientation": "portrait",
|
||||
"short_name": "Ghostfolio",
|
||||
"start_url": "/en/",
|
||||
"start_url": "/${languageCode}/",
|
||||
"theme_color": "#FFFFFF",
|
||||
"url": "https://ghostfol.io"
|
||||
"url": "${rootUrl}"
|
||||
}
|
@ -1,4 +1,7 @@
|
||||
import { DEFAULT_HOST, DEFAULT_PORT } from '@ghostfolio/common/config';
|
||||
|
||||
export const environment = {
|
||||
production: true,
|
||||
rootUrl: `http://${DEFAULT_HOST}:${DEFAULT_PORT}`,
|
||||
version: `${require('../../../../package.json').version}`
|
||||
};
|
||||
|
@ -1,4 +1,7 @@
|
||||
import { DEFAULT_HOST } from '@ghostfolio/common/config';
|
||||
|
||||
export const environment = {
|
||||
production: false,
|
||||
rootUrl: `https://${DEFAULT_HOST}:4200`,
|
||||
version: 'dev'
|
||||
};
|
||||
|
@ -1,4 +1,8 @@
|
||||
import { STORYBOOK_PATH } from '@ghostfolio/common/config';
|
||||
import {
|
||||
DEFAULT_HOST,
|
||||
DEFAULT_PORT,
|
||||
STORYBOOK_PATH
|
||||
} from '@ghostfolio/common/config';
|
||||
|
||||
import {
|
||||
Logger,
|
||||
@ -75,8 +79,8 @@ async function bootstrap() {
|
||||
|
||||
app.use(HtmlTemplateMiddleware);
|
||||
|
||||
const HOST = configService.get<string>('HOST') || '0.0.0.0';
|
||||
const PORT = configService.get<number>('PORT') || 3333;
|
||||
const HOST = configService.get<string>('HOST') || DEFAULT_HOST;
|
||||
const PORT = configService.get<number>('PORT') || DEFAULT_PORT;
|
||||
|
||||
await app.listen(PORT, HOST, () => {
|
||||
logLogo();
|
||||
|
@ -2,7 +2,6 @@ import { environment } from '@ghostfolio/api/environments/environment';
|
||||
import { I18nService } from '@ghostfolio/api/services/i18n/i18n.service';
|
||||
import {
|
||||
DEFAULT_LANGUAGE_CODE,
|
||||
DEFAULT_ROOT_URL,
|
||||
STORYBOOK_PATH,
|
||||
SUPPORTED_LANGUAGE_CODES
|
||||
} from '@ghostfolio/common/config';
|
||||
@ -126,7 +125,7 @@ export const HtmlTemplateMiddleware = async (
|
||||
}
|
||||
|
||||
const currentDate = format(new Date(), DATE_FORMAT);
|
||||
const rootUrl = process.env.ROOT_URL || DEFAULT_ROOT_URL;
|
||||
const rootUrl = process.env.ROOT_URL || environment.rootUrl;
|
||||
|
||||
if (
|
||||
path.startsWith('/api/') ||
|
||||
|
@ -1,11 +1,13 @@
|
||||
import { environment } from '@ghostfolio/api/environments/environment';
|
||||
import { Environment } from '@ghostfolio/api/services/interfaces/environment.interface';
|
||||
import {
|
||||
CACHE_TTL_NO_CACHE,
|
||||
DEFAULT_HOST,
|
||||
DEFAULT_PORT,
|
||||
DEFAULT_PROCESSOR_GATHER_ASSET_PROFILE_CONCURRENCY,
|
||||
DEFAULT_PROCESSOR_GATHER_HISTORICAL_MARKET_DATA_CONCURRENCY,
|
||||
DEFAULT_PROCESSOR_PORTFOLIO_SNAPSHOT_COMPUTATION_CONCURRENCY,
|
||||
DEFAULT_PROCESSOR_PORTFOLIO_SNAPSHOT_COMPUTATION_TIMEOUT,
|
||||
DEFAULT_ROOT_URL
|
||||
DEFAULT_PROCESSOR_PORTFOLIO_SNAPSHOT_COMPUTATION_TIMEOUT
|
||||
} from '@ghostfolio/common/config';
|
||||
|
||||
import { Injectable } from '@nestjs/common';
|
||||
@ -49,11 +51,11 @@ export class ConfigurationService {
|
||||
GOOGLE_SHEETS_ACCOUNT: str({ default: '' }),
|
||||
GOOGLE_SHEETS_ID: str({ default: '' }),
|
||||
GOOGLE_SHEETS_PRIVATE_KEY: str({ default: '' }),
|
||||
HOST: host({ default: '0.0.0.0' }),
|
||||
HOST: host({ default: DEFAULT_HOST }),
|
||||
JWT_SECRET_KEY: str({}),
|
||||
MAX_ACTIVITIES_TO_IMPORT: num({ default: Number.MAX_SAFE_INTEGER }),
|
||||
MAX_CHART_ITEMS: num({ default: 365 }),
|
||||
PORT: port({ default: 3333 }),
|
||||
PORT: port({ default: DEFAULT_PORT }),
|
||||
PROCESSOR_GATHER_ASSET_PROFILE_CONCURRENCY: num({
|
||||
default: DEFAULT_PROCESSOR_GATHER_ASSET_PROFILE_CONCURRENCY
|
||||
}),
|
||||
@ -71,7 +73,9 @@ export class ConfigurationService {
|
||||
REDIS_PASSWORD: str({ default: '' }),
|
||||
REDIS_PORT: port({ default: 6379 }),
|
||||
REQUEST_TIMEOUT: num({ default: ms('3 seconds') }),
|
||||
ROOT_URL: url({ default: DEFAULT_ROOT_URL }),
|
||||
ROOT_URL: url({
|
||||
default: environment.rootUrl
|
||||
}),
|
||||
STRIPE_PUBLIC_KEY: str({ default: '' }),
|
||||
STRIPE_SECRET_KEY: str({ default: '' }),
|
||||
TWITTER_ACCESS_TOKEN: str({ default: 'dummyAccessToken' }),
|
||||
|
@ -6,13 +6,7 @@
|
||||
"name": "app",
|
||||
"installMode": "prefetch",
|
||||
"resources": {
|
||||
"files": [
|
||||
"/favicon.ico",
|
||||
"/index.html",
|
||||
"/assets/site.webmanifest",
|
||||
"/*.css",
|
||||
"/*.js"
|
||||
]
|
||||
"files": ["/favicon.ico", "/index.html", "/*.css", "/*.js"]
|
||||
}
|
||||
},
|
||||
{
|
||||
|
@ -146,9 +146,6 @@
|
||||
{
|
||||
"command": "shx cp apps/client/src/assets/robots.txt dist/apps/client"
|
||||
},
|
||||
{
|
||||
"command": "shx cp apps/client/src/assets/site.webmanifest dist/apps/client"
|
||||
},
|
||||
{
|
||||
"command": "shx cp node_modules/ionicons/dist/index.js dist/apps/client"
|
||||
},
|
||||
|
@ -45,7 +45,10 @@
|
||||
sizes="16x16"
|
||||
type="image/png"
|
||||
/>
|
||||
<link href="../assets/site.webmanifest" rel="manifest" />
|
||||
<link
|
||||
href="../api/assets/${languageCode}/site.webmanifest"
|
||||
rel="manifest"
|
||||
/>
|
||||
</head>
|
||||
<body>
|
||||
<gf-root></gf-root>
|
||||
|
@ -48,13 +48,14 @@ export const PORTFOLIO_SNAPSHOT_COMPUTATION_QUEUE_PRIORITY_LOW =
|
||||
|
||||
export const DEFAULT_CURRENCY = 'USD';
|
||||
export const DEFAULT_DATE_FORMAT_MONTH_YEAR = 'MMM yyyy';
|
||||
export const DEFAULT_HOST = '0.0.0.0';
|
||||
export const DEFAULT_LANGUAGE_CODE = 'en';
|
||||
export const DEFAULT_PAGE_SIZE = 50;
|
||||
export const DEFAULT_PORT = 3333;
|
||||
export const DEFAULT_PROCESSOR_GATHER_ASSET_PROFILE_CONCURRENCY = 1;
|
||||
export const DEFAULT_PROCESSOR_GATHER_HISTORICAL_MARKET_DATA_CONCURRENCY = 1;
|
||||
export const DEFAULT_PROCESSOR_PORTFOLIO_SNAPSHOT_COMPUTATION_CONCURRENCY = 1;
|
||||
export const DEFAULT_PROCESSOR_PORTFOLIO_SNAPSHOT_COMPUTATION_TIMEOUT = 30000;
|
||||
export const DEFAULT_ROOT_URL = 'https://localhost:4200';
|
||||
|
||||
// USX is handled separately
|
||||
export const DERIVED_CURRENCIES = [
|
||||
|
Loading…
x
Reference in New Issue
Block a user