Browse Source

Merge upstream/main

pull/5981/head
Germán Martín 5 days ago
parent
commit
accce9bec4
  1. 11
      CHANGELOG.md
  2. 21
      apps/api/src/app/admin/admin.service.ts
  3. 2
      apps/api/src/app/subscription/subscription.service.ts
  4. 7
      apps/api/src/interceptors/redact-values-in-response/redact-values-in-response.interceptor.ts
  5. 6
      apps/api/src/interceptors/transform-data-source-in-request/transform-data-source-in-request.interceptor.ts
  6. 6
      apps/api/src/interceptors/transform-data-source-in-response/transform-data-source-in-response.interceptor.ts
  7. 2
      apps/api/src/services/data-provider/data-enhancer/yahoo-finance/yahoo-finance.service.ts
  8. 28
      apps/api/src/services/data-provider/financial-modeling-prep/financial-modeling-prep.service.ts
  9. 102
      apps/client/src/app/pages/blog/blog-page.routes.ts
  10. 6
      apps/client/src/app/pages/resources/resources-page.routes.ts
  11. 3
      libs/common/src/lib/interfaces/responses/data-provider-ghostfolio-asset-profile-response.interface.ts
  12. 75
      libs/common/src/lib/interfaces/simplewebauthn.interface.ts
  13. 3
      libs/common/src/lib/types/subscription-offer-key.type.ts
  14. 4
      libs/common/src/lib/validators/is-currency-code.ts
  15. 6
      libs/ui/src/lib/assistant/interfaces/interfaces.ts
  16. 2
      libs/ui/src/lib/entity-logo/entity-logo.component.html
  17. 20
      package-lock.json
  18. 6
      package.json

11
CHANGELOG.md

@ -11,6 +11,13 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
- Added _OpenID Connect_ (`OIDC`) as a new login provider (experimental) - Added _OpenID Connect_ (`OIDC`) as a new login provider (experimental)
#### Changed
- Upgraded `envalid` from version `8.1.0` to `8.1.1`
- Upgraded `prettier` from version `3.7.3` to `3.7.4`
## 2.221.0 - 2025-12-01
### Changed ### Changed
- Refactored the API query parameters in various data provider services - Refactored the API query parameters in various data provider services
@ -21,6 +28,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
### Fixed ### Fixed
- Improved the country weightings in the _Financial Modeling Prep_ service - Improved the country weightings in the _Financial Modeling Prep_ service
- Improved the search functionality by name in the _Financial Modeling Prep_ service
- Resolved an issue in the user endpoint where the list was returning empty in the admin control panel’s users section
## 2.220.0 - 2025-11-29 ## 2.220.0 - 2025-11-29
@ -2256,7 +2265,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
### Fixed ### Fixed
- Fixed an issue in the portfolio summary with the currency conversion of fees - Fixed an issue in the portfolio summary with the currency conversion of fees
- Fixed an issue in the the search for a holding - Fixed an issue in the search for a holding
- Removed the show condition of the experimental features setting in the user settings - Removed the show condition of the experimental features setting in the user settings
## 2.95.0 - 2024-07-12 ## 2.95.0 - 2024-07-12

21
apps/api/src/app/admin/admin.service.ts

@ -532,12 +532,7 @@ export class AdminService {
this.countUsersWithAnalytics(), this.countUsersWithAnalytics(),
this.getUsersWithAnalytics({ this.getUsersWithAnalytics({
skip, skip,
take, take
where: {
NOT: {
analytics: null
}
}
}) })
]); ]);
@ -855,6 +850,20 @@ export class AdminService {
} }
} }
]; ];
const noAnalyticsCondition: Prisma.UserWhereInput['NOT'] = {
analytics: null
};
if (where) {
if (where.NOT) {
where.NOT = { ...where.NOT, ...noAnalyticsCondition };
} else {
where.NOT = noAnalyticsCondition;
}
} else {
where = { NOT: noAnalyticsCondition };
}
} }
const usersWithAnalytics = await this.prismaService.user.findMany({ const usersWithAnalytics = await this.prismaService.user.findMany({

2
apps/api/src/app/subscription/subscription.service.ts

@ -179,6 +179,8 @@ export class SubscriptionService {
offerKey = 'renewal-early-bird-2023'; offerKey = 'renewal-early-bird-2023';
} else if (isBefore(createdAt, parseDate('2024-01-01'))) { } else if (isBefore(createdAt, parseDate('2024-01-01'))) {
offerKey = 'renewal-early-bird-2024'; offerKey = 'renewal-early-bird-2024';
} else if (isBefore(createdAt, parseDate('2025-12-01'))) {
offerKey = 'renewal-early-bird-2025';
} }
const offer = await this.getSubscriptionOffer({ const offer = await this.getSubscriptionOffer({

7
apps/api/src/interceptors/redact-values-in-response/redact-values-in-response.interceptor.ts

@ -16,9 +16,10 @@ import { Observable } from 'rxjs';
import { map } from 'rxjs/operators'; import { map } from 'rxjs/operators';
@Injectable() @Injectable()
export class RedactValuesInResponseInterceptor<T> export class RedactValuesInResponseInterceptor<T> implements NestInterceptor<
implements NestInterceptor<T, any> T,
{ any
> {
public intercept( public intercept(
context: ExecutionContext, context: ExecutionContext,
next: CallHandler<T> next: CallHandler<T>

6
apps/api/src/interceptors/transform-data-source-in-request/transform-data-source-in-request.interceptor.ts

@ -11,9 +11,9 @@ import { DataSource } from '@prisma/client';
import { Observable } from 'rxjs'; import { Observable } from 'rxjs';
@Injectable() @Injectable()
export class TransformDataSourceInRequestInterceptor<T> export class TransformDataSourceInRequestInterceptor<
implements NestInterceptor<T, any> T
{ > implements NestInterceptor<T, any> {
public constructor( public constructor(
private readonly configurationService: ConfigurationService private readonly configurationService: ConfigurationService
) {} ) {}

6
apps/api/src/interceptors/transform-data-source-in-response/transform-data-source-in-response.interceptor.ts

@ -13,9 +13,9 @@ import { Observable } from 'rxjs';
import { map } from 'rxjs/operators'; import { map } from 'rxjs/operators';
@Injectable() @Injectable()
export class TransformDataSourceInResponseInterceptor<T> export class TransformDataSourceInResponseInterceptor<
implements NestInterceptor<T, any> T
{ > implements NestInterceptor<T, any> {
private encodedDataSourceMap: { private encodedDataSourceMap: {
[dataSource: string]: string; [dataSource: string]: string;
} = {}; } = {};

2
apps/api/src/services/data-provider/data-enhancer/yahoo-finance/yahoo-finance.service.ts

@ -317,7 +317,7 @@ export class YahooFinanceDataEnhancerService implements DataEnhancerInterface {
return { assetClass, assetSubClass }; return { assetClass, assetSubClass };
} }
private parseSector(aString: string): string { private parseSector(aString: string) {
let sector = UNKNOWN_KEY; let sector = UNKNOWN_KEY;
switch (aString) { switch (aString) {

28
apps/api/src/services/data-provider/financial-modeling-prep/financial-modeling-prep.service.ts

@ -41,6 +41,7 @@ import {
isSameDay, isSameDay,
parseISO parseISO
} from 'date-fns'; } from 'date-fns';
import { uniqBy } from 'lodash';
@Injectable() @Injectable()
export class FinancialModelingPrepService implements DataProviderInterface { export class FinancialModelingPrepService implements DataProviderInterface {
@ -549,14 +550,27 @@ export class FinancialModelingPrepService implements DataProviderInterface {
apikey: this.apiKey apikey: this.apiKey
}); });
const result = await fetch( const [nameResults, symbolResults] = await Promise.all([
`${this.getUrl({ version: 'stable' })}/search-symbol?${queryParams.toString()}`, fetch(
{ `${this.getUrl({ version: 'stable' })}/search-name?${queryParams.toString()}`,
signal: AbortSignal.timeout( {
this.configurationService.get('REQUEST_TIMEOUT') signal: AbortSignal.timeout(requestTimeout)
) }
).then((res) => res.json()),
fetch(
`${this.getUrl({ version: 'stable' })}/search-symbol?${queryParams.toString()}`,
{
signal: AbortSignal.timeout(requestTimeout)
}
).then((res) => res.json())
]);
const result = uniqBy(
[...nameResults, ...symbolResults],
({ exchange, symbol }) => {
return `${exchange}-${symbol}`;
} }
).then((res) => res.json()); );
items = result items = result
.filter(({ exchange, symbol }) => { .filter(({ exchange, symbol }) => {

102
apps/client/src/app/pages/blog/blog-page.routes.ts

@ -34,117 +34,117 @@ export const routes: Routes = [
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/01/ghostfolio-first-months-in-open-source', path: '2022/01/ghostfolio-first-months-in-open-source',
loadComponent: () => loadComponent: () =>
import( import('./2022/01/first-months-in-open-source/first-months-in-open-source-page.component').then(
'./2022/01/first-months-in-open-source/first-months-in-open-source-page.component' (c) => c.FirstMonthsInOpenSourcePageComponent
).then((c) => c.FirstMonthsInOpenSourcePageComponent), ),
title: 'First months in Open Source' title: 'First months in Open Source'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/07/ghostfolio-meets-internet-identity', path: '2022/07/ghostfolio-meets-internet-identity',
loadComponent: () => loadComponent: () =>
import( import('./2022/07/ghostfolio-meets-internet-identity/ghostfolio-meets-internet-identity-page.component').then(
'./2022/07/ghostfolio-meets-internet-identity/ghostfolio-meets-internet-identity-page.component' (c) => c.GhostfolioMeetsInternetIdentityPageComponent
).then((c) => c.GhostfolioMeetsInternetIdentityPageComponent), ),
title: 'Ghostfolio meets Internet Identity' title: 'Ghostfolio meets Internet Identity'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/07/how-do-i-get-my-finances-in-order', path: '2022/07/how-do-i-get-my-finances-in-order',
loadComponent: () => loadComponent: () =>
import( import('./2022/07/how-do-i-get-my-finances-in-order/how-do-i-get-my-finances-in-order-page.component').then(
'./2022/07/how-do-i-get-my-finances-in-order/how-do-i-get-my-finances-in-order-page.component' (c) => c.HowDoIGetMyFinancesInOrderPageComponent
).then((c) => c.HowDoIGetMyFinancesInOrderPageComponent), ),
title: 'How do I get my finances in order?' title: 'How do I get my finances in order?'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/08/500-stars-on-github', path: '2022/08/500-stars-on-github',
loadComponent: () => loadComponent: () =>
import( import('./2022/08/500-stars-on-github/500-stars-on-github-page.component').then(
'./2022/08/500-stars-on-github/500-stars-on-github-page.component' (c) => c.FiveHundredStarsOnGitHubPageComponent
).then((c) => c.FiveHundredStarsOnGitHubPageComponent), ),
title: '500 Stars on GitHub' title: '500 Stars on GitHub'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/10/hacktoberfest-2022', path: '2022/10/hacktoberfest-2022',
loadComponent: () => loadComponent: () =>
import( import('./2022/10/hacktoberfest-2022/hacktoberfest-2022-page.component').then(
'./2022/10/hacktoberfest-2022/hacktoberfest-2022-page.component' (c) => c.Hacktoberfest2022PageComponent
).then((c) => c.Hacktoberfest2022PageComponent), ),
title: 'Hacktoberfest 2022' title: 'Hacktoberfest 2022'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/11/black-friday-2022', path: '2022/11/black-friday-2022',
loadComponent: () => loadComponent: () =>
import( import('./2022/11/black-friday-2022/black-friday-2022-page.component').then(
'./2022/11/black-friday-2022/black-friday-2022-page.component' (c) => c.BlackFriday2022PageComponent
).then((c) => c.BlackFriday2022PageComponent), ),
title: 'Black Friday 2022' title: 'Black Friday 2022'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2022/12/the-importance-of-tracking-your-personal-finances', path: '2022/12/the-importance-of-tracking-your-personal-finances',
loadComponent: () => loadComponent: () =>
import( import('./2022/12/the-importance-of-tracking-your-personal-finances/the-importance-of-tracking-your-personal-finances-page.component').then(
'./2022/12/the-importance-of-tracking-your-personal-finances/the-importance-of-tracking-your-personal-finances-page.component' (c) => c.TheImportanceOfTrackingYourPersonalFinancesPageComponent
).then((c) => c.TheImportanceOfTrackingYourPersonalFinancesPageComponent), ),
title: 'The importance of tracking your personal finances' title: 'The importance of tracking your personal finances'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/01/ghostfolio-auf-sackgeld-vorgestellt', path: '2023/01/ghostfolio-auf-sackgeld-vorgestellt',
loadComponent: () => loadComponent: () =>
import( import('./2023/01/ghostfolio-auf-sackgeld-vorgestellt/ghostfolio-auf-sackgeld-vorgestellt-page.component').then(
'./2023/01/ghostfolio-auf-sackgeld-vorgestellt/ghostfolio-auf-sackgeld-vorgestellt-page.component' (c) => c.GhostfolioAufSackgeldVorgestelltPageComponent
).then((c) => c.GhostfolioAufSackgeldVorgestelltPageComponent), ),
title: 'Ghostfolio auf Sackgeld.com vorgestellt' title: 'Ghostfolio auf Sackgeld.com vorgestellt'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/02/ghostfolio-meets-umbrel', path: '2023/02/ghostfolio-meets-umbrel',
loadComponent: () => loadComponent: () =>
import( import('./2023/02/ghostfolio-meets-umbrel/ghostfolio-meets-umbrel-page.component').then(
'./2023/02/ghostfolio-meets-umbrel/ghostfolio-meets-umbrel-page.component' (c) => c.GhostfolioMeetsUmbrelPageComponent
).then((c) => c.GhostfolioMeetsUmbrelPageComponent), ),
title: 'Ghostfolio meets Umbrel' title: 'Ghostfolio meets Umbrel'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/03/ghostfolio-reaches-1000-stars-on-github', path: '2023/03/ghostfolio-reaches-1000-stars-on-github',
loadComponent: () => loadComponent: () =>
import( import('./2023/03/1000-stars-on-github/1000-stars-on-github-page.component').then(
'./2023/03/1000-stars-on-github/1000-stars-on-github-page.component' (c) => c.ThousandStarsOnGitHubPageComponent
).then((c) => c.ThousandStarsOnGitHubPageComponent), ),
title: 'Ghostfolio reaches 1’000 Stars on GitHub' title: 'Ghostfolio reaches 1’000 Stars on GitHub'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/05/unlock-your-financial-potential-with-ghostfolio', path: '2023/05/unlock-your-financial-potential-with-ghostfolio',
loadComponent: () => loadComponent: () =>
import( import('./2023/05/unlock-your-financial-potential-with-ghostfolio/unlock-your-financial-potential-with-ghostfolio-page.component').then(
'./2023/05/unlock-your-financial-potential-with-ghostfolio/unlock-your-financial-potential-with-ghostfolio-page.component' (c) => c.UnlockYourFinancialPotentialWithGhostfolioPageComponent
).then((c) => c.UnlockYourFinancialPotentialWithGhostfolioPageComponent), ),
title: 'Unlock your Financial Potential with Ghostfolio' title: 'Unlock your Financial Potential with Ghostfolio'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/07/exploring-the-path-to-fire', path: '2023/07/exploring-the-path-to-fire',
loadComponent: () => loadComponent: () =>
import( import('./2023/07/exploring-the-path-to-fire/exploring-the-path-to-fire-page.component').then(
'./2023/07/exploring-the-path-to-fire/exploring-the-path-to-fire-page.component' (c) => c.ExploringThePathToFirePageComponent
).then((c) => c.ExploringThePathToFirePageComponent), ),
title: 'Exploring the Path to FIRE' title: 'Exploring the Path to FIRE'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/08/ghostfolio-joins-oss-friends', path: '2023/08/ghostfolio-joins-oss-friends',
loadComponent: () => loadComponent: () =>
import( import('./2023/08/ghostfolio-joins-oss-friends/ghostfolio-joins-oss-friends-page.component').then(
'./2023/08/ghostfolio-joins-oss-friends/ghostfolio-joins-oss-friends-page.component' (c) => c.GhostfolioJoinsOssFriendsPageComponent
).then((c) => c.GhostfolioJoinsOssFriendsPageComponent), ),
title: 'Ghostfolio joins OSS Friends' title: 'Ghostfolio joins OSS Friends'
}, },
{ {
@ -160,9 +160,9 @@ export const routes: Routes = [
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/09/hacktoberfest-2023', path: '2023/09/hacktoberfest-2023',
loadComponent: () => loadComponent: () =>
import( import('./2023/09/hacktoberfest-2023/hacktoberfest-2023-page.component').then(
'./2023/09/hacktoberfest-2023/hacktoberfest-2023-page.component' (c) => c.Hacktoberfest2023PageComponent
).then((c) => c.Hacktoberfest2023PageComponent), ),
title: 'Hacktoberfest 2023' title: 'Hacktoberfest 2023'
}, },
{ {
@ -178,18 +178,18 @@ export const routes: Routes = [
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2023/11/hacktoberfest-2023-debriefing', path: '2023/11/hacktoberfest-2023-debriefing',
loadComponent: () => loadComponent: () =>
import( import('./2023/11/hacktoberfest-2023-debriefing/hacktoberfest-2023-debriefing-page.component').then(
'./2023/11/hacktoberfest-2023-debriefing/hacktoberfest-2023-debriefing-page.component' (c) => c.Hacktoberfest2023DebriefingPageComponent
).then((c) => c.Hacktoberfest2023DebriefingPageComponent), ),
title: 'Hacktoberfest 2023 Debriefing' title: 'Hacktoberfest 2023 Debriefing'
}, },
{ {
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2024/09/hacktoberfest-2024', path: '2024/09/hacktoberfest-2024',
loadComponent: () => loadComponent: () =>
import( import('./2024/09/hacktoberfest-2024/hacktoberfest-2024-page.component').then(
'./2024/09/hacktoberfest-2024/hacktoberfest-2024-page.component' (c) => c.Hacktoberfest2024PageComponent
).then((c) => c.Hacktoberfest2024PageComponent), ),
title: 'Hacktoberfest 2024' title: 'Hacktoberfest 2024'
}, },
{ {
@ -205,9 +205,9 @@ export const routes: Routes = [
canActivate: [AuthGuard], canActivate: [AuthGuard],
path: '2025/09/hacktoberfest-2025', path: '2025/09/hacktoberfest-2025',
loadComponent: () => loadComponent: () =>
import( import('./2025/09/hacktoberfest-2025/hacktoberfest-2025-page.component').then(
'./2025/09/hacktoberfest-2025/hacktoberfest-2025-page.component' (c) => c.Hacktoberfest2025PageComponent
).then((c) => c.Hacktoberfest2025PageComponent), ),
title: 'Hacktoberfest 2025' title: 'Hacktoberfest 2025'
}, },
{ {

6
apps/client/src/app/pages/resources/resources-page.routes.ts

@ -33,9 +33,9 @@ export const routes: Routes = [
{ {
path: publicRoutes.resources.subRoutes.personalFinanceTools.path, path: publicRoutes.resources.subRoutes.personalFinanceTools.path,
loadChildren: () => loadChildren: () =>
import( import('./personal-finance-tools/personal-finance-tools-page.routes').then(
'./personal-finance-tools/personal-finance-tools-page.routes' (m) => m.routes
).then((m) => m.routes) )
} }
], ],
path: '', path: '',

3
libs/common/src/lib/interfaces/responses/data-provider-ghostfolio-asset-profile-response.interface.ts

@ -1,4 +1,3 @@
import { SymbolProfile } from '@prisma/client'; import { SymbolProfile } from '@prisma/client';
export interface DataProviderGhostfolioAssetProfileResponse export interface DataProviderGhostfolioAssetProfileResponse extends Partial<SymbolProfile> {}
extends Partial<SymbolProfile> {}

75
libs/common/src/lib/interfaces/simplewebauthn.interface.ts

@ -3,8 +3,7 @@ export interface AuthenticatorAssertionResponse extends AuthenticatorResponse {
readonly signature: ArrayBuffer; readonly signature: ArrayBuffer;
readonly userHandle: ArrayBuffer | null; readonly userHandle: ArrayBuffer | null;
} }
export interface AuthenticatorAttestationResponse export interface AuthenticatorAttestationResponse extends AuthenticatorResponse {
extends AuthenticatorResponse {
readonly attestationObject: ArrayBuffer; readonly attestationObject: ArrayBuffer;
} }
export interface AuthenticationExtensionsClientInputs { export interface AuthenticationExtensionsClientInputs {
@ -57,8 +56,7 @@ export interface PublicKeyCredentialRequestOptions {
timeout?: number; timeout?: number;
userVerification?: UserVerificationRequirement; userVerification?: UserVerificationRequirement;
} }
export interface PublicKeyCredentialUserEntity export interface PublicKeyCredentialUserEntity extends PublicKeyCredentialEntity {
extends PublicKeyCredentialEntity {
displayName: string; displayName: string;
id: BufferSource; id: BufferSource;
} }
@ -99,11 +97,10 @@ export declare type BufferSource = ArrayBufferView | ArrayBuffer;
export declare type PublicKeyCredentialType = 'public-key'; export declare type PublicKeyCredentialType = 'public-key';
export declare type UvmEntry = number[]; export declare type UvmEntry = number[];
export interface PublicKeyCredentialCreationOptionsJSON export interface PublicKeyCredentialCreationOptionsJSON extends Omit<
extends Omit< PublicKeyCredentialCreationOptions,
PublicKeyCredentialCreationOptions, 'challenge' | 'user' | 'excludeCredentials'
'challenge' | 'user' | 'excludeCredentials' > {
> {
user: PublicKeyCredentialUserEntityJSON; user: PublicKeyCredentialUserEntityJSON;
challenge: Base64URLString; challenge: Base64URLString;
excludeCredentials: PublicKeyCredentialDescriptorJSON[]; excludeCredentials: PublicKeyCredentialDescriptorJSON[];
@ -113,21 +110,24 @@ export interface PublicKeyCredentialCreationOptionsJSON
* A variant of PublicKeyCredentialRequestOptions suitable for JSON transmission to the browser to * A variant of PublicKeyCredentialRequestOptions suitable for JSON transmission to the browser to
* (eventually) get passed into navigator.credentials.get(...) in the browser. * (eventually) get passed into navigator.credentials.get(...) in the browser.
*/ */
export interface PublicKeyCredentialRequestOptionsJSON export interface PublicKeyCredentialRequestOptionsJSON extends Omit<
extends Omit< PublicKeyCredentialRequestOptions,
PublicKeyCredentialRequestOptions, 'challenge' | 'allowCredentials'
'challenge' | 'allowCredentials' > {
> {
challenge: Base64URLString; challenge: Base64URLString;
allowCredentials?: PublicKeyCredentialDescriptorJSON[]; allowCredentials?: PublicKeyCredentialDescriptorJSON[];
extensions?: AuthenticationExtensionsClientInputs; extensions?: AuthenticationExtensionsClientInputs;
} }
export interface PublicKeyCredentialDescriptorJSON export interface PublicKeyCredentialDescriptorJSON extends Omit<
extends Omit<PublicKeyCredentialDescriptor, 'id'> { PublicKeyCredentialDescriptor,
'id'
> {
id: Base64URLString; id: Base64URLString;
} }
export interface PublicKeyCredentialUserEntityJSON export interface PublicKeyCredentialUserEntityJSON extends Omit<
extends Omit<PublicKeyCredentialUserEntity, 'id'> { PublicKeyCredentialUserEntity,
'id'
> {
id: string; id: string;
} }
/** /**
@ -140,11 +140,10 @@ export interface AttestationCredential extends PublicKeyCredential {
* A slightly-modified AttestationCredential to simplify working with ArrayBuffers that * A slightly-modified AttestationCredential to simplify working with ArrayBuffers that
* are Base64URL-encoded in the browser so that they can be sent as JSON to the server. * are Base64URL-encoded in the browser so that they can be sent as JSON to the server.
*/ */
export interface AttestationCredentialJSON export interface AttestationCredentialJSON extends Omit<
extends Omit< AttestationCredential,
AttestationCredential, 'response' | 'rawId' | 'getClientExtensionResults'
'response' | 'rawId' | 'getClientExtensionResults' > {
> {
rawId: Base64URLString; rawId: Base64URLString;
response: AuthenticatorAttestationResponseJSON; response: AuthenticatorAttestationResponseJSON;
clientExtensionResults: AuthenticationExtensionsClientOutputs; clientExtensionResults: AuthenticationExtensionsClientOutputs;
@ -160,11 +159,10 @@ export interface AssertionCredential extends PublicKeyCredential {
* A slightly-modified AssertionCredential to simplify working with ArrayBuffers that * A slightly-modified AssertionCredential to simplify working with ArrayBuffers that
* are Base64URL-encoded in the browser so that they can be sent as JSON to the server. * are Base64URL-encoded in the browser so that they can be sent as JSON to the server.
*/ */
export interface AssertionCredentialJSON export interface AssertionCredentialJSON extends Omit<
extends Omit< AssertionCredential,
AssertionCredential, 'response' | 'rawId' | 'getClientExtensionResults'
'response' | 'rawId' | 'getClientExtensionResults' > {
> {
rawId: Base64URLString; rawId: Base64URLString;
response: AuthenticatorAssertionResponseJSON; response: AuthenticatorAssertionResponseJSON;
clientExtensionResults: AuthenticationExtensionsClientOutputs; clientExtensionResults: AuthenticationExtensionsClientOutputs;
@ -173,11 +171,10 @@ export interface AssertionCredentialJSON
* A slightly-modified AuthenticatorAttestationResponse to simplify working with ArrayBuffers that * A slightly-modified AuthenticatorAttestationResponse to simplify working with ArrayBuffers that
* are Base64URL-encoded in the browser so that they can be sent as JSON to the server. * are Base64URL-encoded in the browser so that they can be sent as JSON to the server.
*/ */
export interface AuthenticatorAttestationResponseJSON export interface AuthenticatorAttestationResponseJSON extends Omit<
extends Omit< AuthenticatorAttestationResponseFuture,
AuthenticatorAttestationResponseFuture, 'clientDataJSON' | 'attestationObject'
'clientDataJSON' | 'attestationObject' > {
> {
clientDataJSON: Base64URLString; clientDataJSON: Base64URLString;
attestationObject: Base64URLString; attestationObject: Base64URLString;
} }
@ -185,11 +182,10 @@ export interface AuthenticatorAttestationResponseJSON
* A slightly-modified AuthenticatorAssertionResponse to simplify working with ArrayBuffers that * A slightly-modified AuthenticatorAssertionResponse to simplify working with ArrayBuffers that
* are Base64URL-encoded in the browser so that they can be sent as JSON to the server. * are Base64URL-encoded in the browser so that they can be sent as JSON to the server.
*/ */
export interface AuthenticatorAssertionResponseJSON export interface AuthenticatorAssertionResponseJSON extends Omit<
extends Omit< AuthenticatorAssertionResponse,
AuthenticatorAssertionResponse, 'authenticatorData' | 'clientDataJSON' | 'signature' | 'userHandle'
'authenticatorData' | 'clientDataJSON' | 'signature' | 'userHandle' > {
> {
authenticatorData: Base64URLString; authenticatorData: Base64URLString;
clientDataJSON: Base64URLString; clientDataJSON: Base64URLString;
signature: Base64URLString; signature: Base64URLString;
@ -217,8 +213,7 @@ export declare type Base64URLString = string;
* *
* Properties marked optional are not supported in all browsers. * Properties marked optional are not supported in all browsers.
*/ */
export interface AuthenticatorAttestationResponseFuture export interface AuthenticatorAttestationResponseFuture extends AuthenticatorAttestationResponse {
extends AuthenticatorAttestationResponse {
getTransports?: () => AuthenticatorTransport[]; getTransports?: () => AuthenticatorTransport[];
getAuthenticatorData?: () => ArrayBuffer; getAuthenticatorData?: () => ArrayBuffer;
getPublicKey?: () => ArrayBuffer; getPublicKey?: () => ArrayBuffer;

3
libs/common/src/lib/types/subscription-offer-key.type.ts

@ -2,4 +2,5 @@ export type SubscriptionOfferKey =
| 'default' | 'default'
| 'renewal' | 'renewal'
| 'renewal-early-bird-2023' | 'renewal-early-bird-2023'
| 'renewal-early-bird-2024'; | 'renewal-early-bird-2024'
| 'renewal-early-bird-2025';

4
libs/common/src/lib/validators/is-currency-code.ts

@ -21,9 +21,7 @@ export function IsCurrencyCode(validationOptions?: ValidationOptions) {
} }
@ValidatorConstraint({ async: false }) @ValidatorConstraint({ async: false })
export class IsExtendedCurrencyConstraint export class IsExtendedCurrencyConstraint implements ValidatorConstraintInterface {
implements ValidatorConstraintInterface
{
public defaultMessage() { public defaultMessage() {
return '$property must be a valid ISO4217 currency code'; return '$property must be a valid ISO4217 currency code';
} }

6
libs/ui/src/lib/assistant/interfaces/interfaces.ts

@ -3,8 +3,10 @@ import { AccountWithValue, DateRange } from '@ghostfolio/common/types';
import { SearchMode } from '../enums/search-mode'; import { SearchMode } from '../enums/search-mode';
export interface AccountSearchResultItem export interface AccountSearchResultItem extends Pick<
extends Pick<AccountWithValue, 'id' | 'name'> { AccountWithValue,
'id' | 'name'
> {
mode: SearchMode.ACCOUNT; mode: SearchMode.ACCOUNT;
routerLink: string[]; routerLink: string[];
} }

2
libs/ui/src/lib/entity-logo/entity-logo.component.html

@ -1,6 +1,6 @@
@if (src) { @if (src) {
<img <img
onerror="this.style.display='none'" onerror="this.style.display = 'none'"
[ngClass]="{ large: size === 'large' }" [ngClass]="{ large: size === 'large' }"
[src]="src" [src]="src"
[title]="tooltip ? tooltip : ''" [title]="tooltip ? tooltip : ''"

20
package-lock.json

@ -1,12 +1,12 @@
{ {
"name": "ghostfolio", "name": "ghostfolio",
"version": "2.220.0", "version": "2.221.0",
"lockfileVersion": 3, "lockfileVersion": 3,
"requires": true, "requires": true,
"packages": { "packages": {
"": { "": {
"name": "ghostfolio", "name": "ghostfolio",
"version": "2.220.0", "version": "2.221.0",
"hasInstallScript": true, "hasInstallScript": true,
"license": "AGPL-3.0", "license": "AGPL-3.0",
"dependencies": { "dependencies": {
@ -62,7 +62,7 @@
"date-fns": "4.1.0", "date-fns": "4.1.0",
"dotenv": "17.2.3", "dotenv": "17.2.3",
"dotenv-expand": "12.0.3", "dotenv-expand": "12.0.3",
"envalid": "8.1.0", "envalid": "8.1.1",
"fuse.js": "7.1.0", "fuse.js": "7.1.0",
"google-spreadsheet": "3.2.0", "google-spreadsheet": "3.2.0",
"helmet": "7.0.0", "helmet": "7.0.0",
@ -142,7 +142,7 @@
"jest-environment-jsdom": "29.7.0", "jest-environment-jsdom": "29.7.0",
"jest-preset-angular": "14.6.0", "jest-preset-angular": "14.6.0",
"nx": "21.5.1", "nx": "21.5.1",
"prettier": "3.7.3", "prettier": "3.7.4",
"prettier-plugin-organize-attributes": "1.0.0", "prettier-plugin-organize-attributes": "1.0.0",
"prisma": "6.19.0", "prisma": "6.19.0",
"react": "18.2.0", "react": "18.2.0",
@ -21358,9 +21358,9 @@
} }
}, },
"node_modules/envalid": { "node_modules/envalid": {
"version": "8.1.0", "version": "8.1.1",
"resolved": "https://registry.npmjs.org/envalid/-/envalid-8.1.0.tgz", "resolved": "https://registry.npmjs.org/envalid/-/envalid-8.1.1.tgz",
"integrity": "sha512-OT6+qVhKVyCidaGoXflb2iK1tC8pd0OV2Q+v9n33wNhUJ+lus+rJobUj4vJaQBPxPZ0vYrPGuxdrenyCAIJcow==", "integrity": "sha512-vOUfHxAFFvkBjbVQbBfgnCO9d3GcNfMMTtVfgqSU2rQGMFEVqWy9GBuoSfHnwGu7EqR0/GeukQcL3KjFBaga9w==",
"license": "MIT", "license": "MIT",
"dependencies": { "dependencies": {
"tslib": "2.8.1" "tslib": "2.8.1"
@ -35792,9 +35792,9 @@
} }
}, },
"node_modules/prettier": { "node_modules/prettier": {
"version": "3.7.3", "version": "3.7.4",
"resolved": "https://registry.npmjs.org/prettier/-/prettier-3.7.3.tgz", "resolved": "https://registry.npmjs.org/prettier/-/prettier-3.7.4.tgz",
"integrity": "sha512-QgODejq9K3OzoBbuyobZlUhznP5SKwPqp+6Q6xw6o8gnhr4O85L2U915iM2IDcfF2NPXVaM9zlo9tdwipnYwzg==", "integrity": "sha512-v6UNi1+3hSlVvv8fSaoUbggEM5VErKmmpGA7Pl3HF8V6uKY7rvClBOJlH6yNwQtfTueNkGVpOv/mtWL9L4bgRA==",
"dev": true, "dev": true,
"license": "MIT", "license": "MIT",
"bin": { "bin": {

6
package.json

@ -1,6 +1,6 @@
{ {
"name": "ghostfolio", "name": "ghostfolio",
"version": "2.220.0", "version": "2.221.0",
"homepage": "https://ghostfol.io", "homepage": "https://ghostfol.io",
"license": "AGPL-3.0", "license": "AGPL-3.0",
"repository": "https://github.com/ghostfolio/ghostfolio", "repository": "https://github.com/ghostfolio/ghostfolio",
@ -106,7 +106,7 @@
"date-fns": "4.1.0", "date-fns": "4.1.0",
"dotenv": "17.2.3", "dotenv": "17.2.3",
"dotenv-expand": "12.0.3", "dotenv-expand": "12.0.3",
"envalid": "8.1.0", "envalid": "8.1.1",
"fuse.js": "7.1.0", "fuse.js": "7.1.0",
"google-spreadsheet": "3.2.0", "google-spreadsheet": "3.2.0",
"helmet": "7.0.0", "helmet": "7.0.0",
@ -186,7 +186,7 @@
"jest-environment-jsdom": "29.7.0", "jest-environment-jsdom": "29.7.0",
"jest-preset-angular": "14.6.0", "jest-preset-angular": "14.6.0",
"nx": "21.5.1", "nx": "21.5.1",
"prettier": "3.7.3", "prettier": "3.7.4",
"prettier-plugin-organize-attributes": "1.0.0", "prettier-plugin-organize-attributes": "1.0.0",
"prisma": "6.19.0", "prisma": "6.19.0",
"react": "18.2.0", "react": "18.2.0",

Loading…
Cancel
Save