Compare commits
3 Commits
44850e0802
...
9db8c5ccef
Author | SHA1 | Date | |
---|---|---|---|
9db8c5ccef | |||
|
589eefaa76 | ||
|
b260c4f450 |
@ -5,6 +5,12 @@ All notable changes to this project will be documented in this file.
|
|||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.0.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
|
## Unreleased
|
||||||
|
|
||||||
|
### Added
|
||||||
|
|
||||||
|
- Added a _Copy portfolio data to clipboard for AI prompt_ action to the analysis page (experimental)
|
||||||
|
|
||||||
## 2.144.0 - 2025-03-06
|
## 2.144.0 - 2025-03-06
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
@ -6,9 +6,9 @@ import {
|
|||||||
} from '@ghostfolio/common/config';
|
} from '@ghostfolio/common/config';
|
||||||
import { AiPromptResponse } from '@ghostfolio/common/interfaces';
|
import { AiPromptResponse } from '@ghostfolio/common/interfaces';
|
||||||
import { permissions } from '@ghostfolio/common/permissions';
|
import { permissions } from '@ghostfolio/common/permissions';
|
||||||
import type { RequestWithUser } from '@ghostfolio/common/types';
|
import type { AiPromptMode, RequestWithUser } from '@ghostfolio/common/types';
|
||||||
|
|
||||||
import { Controller, Get, Inject, UseGuards } from '@nestjs/common';
|
import { Controller, Get, Inject, Param, UseGuards } from '@nestjs/common';
|
||||||
import { REQUEST } from '@nestjs/core';
|
import { REQUEST } from '@nestjs/core';
|
||||||
import { AuthGuard } from '@nestjs/passport';
|
import { AuthGuard } from '@nestjs/passport';
|
||||||
|
|
||||||
@ -21,11 +21,14 @@ export class AiController {
|
|||||||
@Inject(REQUEST) private readonly request: RequestWithUser
|
@Inject(REQUEST) private readonly request: RequestWithUser
|
||||||
) {}
|
) {}
|
||||||
|
|
||||||
@Get('prompt')
|
@Get('prompt/:mode')
|
||||||
@HasPermission(permissions.readAiPrompt)
|
@HasPermission(permissions.readAiPrompt)
|
||||||
@UseGuards(AuthGuard('jwt'), HasPermissionGuard)
|
@UseGuards(AuthGuard('jwt'), HasPermissionGuard)
|
||||||
public async getPrompt(): Promise<AiPromptResponse> {
|
public async getPrompt(
|
||||||
|
@Param('mode') mode: AiPromptMode
|
||||||
|
): Promise<AiPromptResponse> {
|
||||||
const prompt = await this.aiService.getPrompt({
|
const prompt = await this.aiService.getPrompt({
|
||||||
|
mode,
|
||||||
impersonationId: undefined,
|
impersonationId: undefined,
|
||||||
languageCode:
|
languageCode:
|
||||||
this.request.user.Settings.settings.language ?? DEFAULT_LANGUAGE_CODE,
|
this.request.user.Settings.settings.language ?? DEFAULT_LANGUAGE_CODE,
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { PortfolioService } from '@ghostfolio/api/app/portfolio/portfolio.service';
|
import { PortfolioService } from '@ghostfolio/api/app/portfolio/portfolio.service';
|
||||||
|
import type { AiPromptMode } from '@ghostfolio/common/types';
|
||||||
|
|
||||||
import { Injectable } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
|
|
||||||
@ -9,11 +10,13 @@ export class AiService {
|
|||||||
public async getPrompt({
|
public async getPrompt({
|
||||||
impersonationId,
|
impersonationId,
|
||||||
languageCode,
|
languageCode,
|
||||||
|
mode,
|
||||||
userCurrency,
|
userCurrency,
|
||||||
userId
|
userId
|
||||||
}: {
|
}: {
|
||||||
impersonationId: string;
|
impersonationId: string;
|
||||||
languageCode: string;
|
languageCode: string;
|
||||||
|
mode: AiPromptMode;
|
||||||
userCurrency: string;
|
userCurrency: string;
|
||||||
userId: string;
|
userId: string;
|
||||||
}) {
|
}) {
|
||||||
@ -43,6 +46,10 @@ export class AiService {
|
|||||||
)
|
)
|
||||||
];
|
];
|
||||||
|
|
||||||
|
if (mode === 'portfolio') {
|
||||||
|
return holdingsTable.join('\n');
|
||||||
|
}
|
||||||
|
|
||||||
return [
|
return [
|
||||||
`You are a neutral financial assistant. Please analyze the following investment portfolio (base currency being ${userCurrency}) in simple words.`,
|
`You are a neutral financial assistant. Please analyze the following investment portfolio (base currency being ${userCurrency}) in simple words.`,
|
||||||
...holdingsTable,
|
...holdingsTable,
|
||||||
|
@ -13,7 +13,7 @@ import {
|
|||||||
User
|
User
|
||||||
} from '@ghostfolio/common/interfaces';
|
} from '@ghostfolio/common/interfaces';
|
||||||
import { hasPermission, permissions } from '@ghostfolio/common/permissions';
|
import { hasPermission, permissions } from '@ghostfolio/common/permissions';
|
||||||
import { GroupBy } from '@ghostfolio/common/types';
|
import type { AiPromptMode, GroupBy } from '@ghostfolio/common/types';
|
||||||
import { translate } from '@ghostfolio/ui/i18n';
|
import { translate } from '@ghostfolio/ui/i18n';
|
||||||
|
|
||||||
import { Clipboard } from '@angular/cdk/clipboard';
|
import { Clipboard } from '@angular/cdk/clipboard';
|
||||||
@ -171,9 +171,9 @@ export class AnalysisPageComponent implements OnDestroy, OnInit {
|
|||||||
this.fetchDividendsAndInvestments();
|
this.fetchDividendsAndInvestments();
|
||||||
}
|
}
|
||||||
|
|
||||||
public onCopyPromptToClipboard() {
|
public onCopyPromptToClipboard(mode: AiPromptMode) {
|
||||||
this.dataService
|
this.dataService
|
||||||
.fetchPrompt()
|
.fetchPrompt(mode)
|
||||||
.pipe(takeUntil(this.unsubscribeSubject))
|
.pipe(takeUntil(this.unsubscribeSubject))
|
||||||
.subscribe(({ prompt }) => {
|
.subscribe(({ prompt }) => {
|
||||||
this.clipboard.copy(prompt);
|
this.clipboard.copy(prompt);
|
||||||
|
@ -16,7 +16,7 @@
|
|||||||
<button
|
<button
|
||||||
mat-menu-item
|
mat-menu-item
|
||||||
[disabled]="!hasPermissionToReadAiPrompt"
|
[disabled]="!hasPermissionToReadAiPrompt"
|
||||||
(click)="onCopyPromptToClipboard()"
|
(click)="onCopyPromptToClipboard('portfolio')"
|
||||||
>
|
>
|
||||||
<span class="align-items-center d-flex">
|
<span class="align-items-center d-flex">
|
||||||
@if (user?.subscription?.type === 'Basic') {
|
@if (user?.subscription?.type === 'Basic') {
|
||||||
@ -24,7 +24,25 @@
|
|||||||
} @else {
|
} @else {
|
||||||
<ion-icon class="mr-2" name="copy-outline" />
|
<ion-icon class="mr-2" name="copy-outline" />
|
||||||
}
|
}
|
||||||
<ng-container i18n>Copy AI prompt to clipboard</ng-container>
|
<ng-container i18n
|
||||||
|
>Copy portfolio data to clipboard for AI prompt</ng-container
|
||||||
|
>
|
||||||
|
</span>
|
||||||
|
</button>
|
||||||
|
<button
|
||||||
|
mat-menu-item
|
||||||
|
[disabled]="!hasPermissionToReadAiPrompt"
|
||||||
|
(click)="onCopyPromptToClipboard('analysis')"
|
||||||
|
>
|
||||||
|
<span class="align-items-center d-flex">
|
||||||
|
@if (user?.subscription?.type === 'Basic') {
|
||||||
|
<gf-premium-indicator class="mr-2" />
|
||||||
|
} @else {
|
||||||
|
<ion-icon class="mr-2" name="copy-outline" />
|
||||||
|
}
|
||||||
|
<ng-container i18n
|
||||||
|
>Copy AI prompt to clipboard for analysis</ng-container
|
||||||
|
>
|
||||||
</span>
|
</span>
|
||||||
</button>
|
</button>
|
||||||
</mat-menu>
|
</mat-menu>
|
||||||
|
@ -46,7 +46,12 @@ import {
|
|||||||
User
|
User
|
||||||
} from '@ghostfolio/common/interfaces';
|
} from '@ghostfolio/common/interfaces';
|
||||||
import { filterGlobalPermissions } from '@ghostfolio/common/permissions';
|
import { filterGlobalPermissions } from '@ghostfolio/common/permissions';
|
||||||
import { AccountWithValue, DateRange, GroupBy } from '@ghostfolio/common/types';
|
import type {
|
||||||
|
AccountWithValue,
|
||||||
|
AiPromptMode,
|
||||||
|
DateRange,
|
||||||
|
GroupBy
|
||||||
|
} from '@ghostfolio/common/types';
|
||||||
import { translate } from '@ghostfolio/ui/i18n';
|
import { translate } from '@ghostfolio/ui/i18n';
|
||||||
|
|
||||||
import { HttpClient, HttpParams } from '@angular/common/http';
|
import { HttpClient, HttpParams } from '@angular/common/http';
|
||||||
@ -650,8 +655,8 @@ export class DataService {
|
|||||||
return this.http.get<PortfolioReportResponse>('/api/v1/portfolio/report');
|
return this.http.get<PortfolioReportResponse>('/api/v1/portfolio/report');
|
||||||
}
|
}
|
||||||
|
|
||||||
public fetchPrompt() {
|
public fetchPrompt(mode: AiPromptMode) {
|
||||||
return this.http.get<AiPromptResponse>('/api/v1/ai/prompt');
|
return this.http.get<AiPromptResponse>(`/api/v1/ai/prompt/${mode}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
public fetchPublicPortfolio(aAccessId: string) {
|
public fetchPublicPortfolio(aAccessId: string) {
|
||||||
|
@ -82,6 +82,16 @@ export const personalFinanceTools: Product[] = [
|
|||||||
regions: ['Global'],
|
regions: ['Global'],
|
||||||
slogan: 'Take control of your financial future'
|
slogan: 'Take control of your financial future'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
founded: 2017,
|
||||||
|
hasFreePlan: true,
|
||||||
|
hasSelfHostingAbility: false,
|
||||||
|
key: 'coinstats',
|
||||||
|
name: 'CoinStats',
|
||||||
|
origin: 'Armenia',
|
||||||
|
pricingPerYear: '$168',
|
||||||
|
slogan: 'Manage All Your Wallets & Exchanges From One Place'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
founded: 2013,
|
founded: 2013,
|
||||||
hasFreePlan: true,
|
hasFreePlan: true,
|
||||||
@ -154,6 +164,7 @@ export const personalFinanceTools: Product[] = [
|
|||||||
name: 'Delta Investment Tracker',
|
name: 'Delta Investment Tracker',
|
||||||
note: 'Acquired by eToro',
|
note: 'Acquired by eToro',
|
||||||
origin: 'Belgium',
|
origin: 'Belgium',
|
||||||
|
pricingPerYear: '$150',
|
||||||
slogan: 'The app to track all your investments. Make smart moves only.'
|
slogan: 'The app to track all your investments. Make smart moves only.'
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -252,6 +263,13 @@ export const personalFinanceTools: Product[] = [
|
|||||||
slogan:
|
slogan:
|
||||||
'The most convenient mobile application for personal finance accounting'
|
'The most convenient mobile application for personal finance accounting'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
founded: 2022,
|
||||||
|
key: 'fincake',
|
||||||
|
name: 'Fincake',
|
||||||
|
origin: 'British Virgin Islands',
|
||||||
|
slogan: 'Easy-to-use Portfolio Tracker'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
founded: 2023,
|
founded: 2023,
|
||||||
hasFreePlan: true,
|
hasFreePlan: true,
|
||||||
@ -340,6 +358,15 @@ export const personalFinanceTools: Product[] = [
|
|||||||
pricingPerYear: '€119',
|
pricingPerYear: '€119',
|
||||||
slogan: 'ETF portfolios made simple'
|
slogan: 'ETF portfolios made simple'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
founded: 2018,
|
||||||
|
hasFreePlan: true,
|
||||||
|
hasSelfHostingAbility: false,
|
||||||
|
key: 'koinly',
|
||||||
|
name: 'Koinly',
|
||||||
|
origin: 'Singapore',
|
||||||
|
slogan: 'Track all your crypto wallets in one place'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
founded: 2016,
|
founded: 2016,
|
||||||
hasFreePlan: true,
|
hasFreePlan: true,
|
||||||
@ -469,6 +496,16 @@ export const personalFinanceTools: Product[] = [
|
|||||||
slogan:
|
slogan:
|
||||||
'Track your equity, fund, investment trust, ETF and pension investments in one place.'
|
'Track your equity, fund, investment trust, ETF and pension investments in one place.'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
founded: 2020,
|
||||||
|
hasFreePlan: true,
|
||||||
|
hasSelfHostingAbility: false,
|
||||||
|
key: 'nansen',
|
||||||
|
name: 'Crypto Portfolio Tracker by Nansen',
|
||||||
|
origin: 'Singapore',
|
||||||
|
pricingPerYear: '$1188',
|
||||||
|
slogan: 'Your Complete Crypto Portfolio, Reimagined'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
founded: 2017,
|
founded: 2017,
|
||||||
hasFreePlan: false,
|
hasFreePlan: false,
|
||||||
@ -634,6 +671,16 @@ export const personalFinanceTools: Product[] = [
|
|||||||
pricingPerYear: '€80',
|
pricingPerYear: '€80',
|
||||||
slogan: 'Stock Portfolio Tracker'
|
slogan: 'Stock Portfolio Tracker'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
founded: 2014,
|
||||||
|
hasFreePlan: true,
|
||||||
|
hasSelfHostingAbility: false,
|
||||||
|
key: 'simply-wallstreet',
|
||||||
|
name: 'Stock Portfolio Tracker & Visualizer by Simply Wall St',
|
||||||
|
origin: 'Australia',
|
||||||
|
pricingPerYear: '$120',
|
||||||
|
slogan: 'Smart portfolio tracker for informed investors'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
founded: 2021,
|
founded: 2021,
|
||||||
hasFreePlan: true,
|
hasFreePlan: true,
|
||||||
@ -706,6 +753,16 @@ export const personalFinanceTools: Product[] = [
|
|||||||
slogan:
|
slogan:
|
||||||
'Your financial life in a spreadsheet, automatically updated each day'
|
'Your financial life in a spreadsheet, automatically updated each day'
|
||||||
},
|
},
|
||||||
|
{
|
||||||
|
founded: 2011,
|
||||||
|
hasFreePlan: false,
|
||||||
|
hasSelfHostingAbility: false,
|
||||||
|
key: 'tradervue',
|
||||||
|
name: 'Tradervue',
|
||||||
|
origin: 'United States',
|
||||||
|
pricingPerYear: '$360',
|
||||||
|
slogan: 'The Trading Journal to Improve Your Trading Performance'
|
||||||
|
},
|
||||||
{
|
{
|
||||||
hasFreePlan: true,
|
hasFreePlan: true,
|
||||||
hasSelfHostingAbility: false,
|
hasSelfHostingAbility: false,
|
||||||
|
1
libs/common/src/lib/types/ai-prompt-mode.type.ts
Normal file
1
libs/common/src/lib/types/ai-prompt-mode.type.ts
Normal file
@ -0,0 +1 @@
|
|||||||
|
export type AiPromptMode = 'analysis' | 'portfolio';
|
@ -2,6 +2,7 @@ import type { AccessType } from './access-type.type';
|
|||||||
import type { AccessWithGranteeUser } from './access-with-grantee-user.type';
|
import type { AccessWithGranteeUser } from './access-with-grantee-user.type';
|
||||||
import type { AccountWithPlatform } from './account-with-platform.type';
|
import type { AccountWithPlatform } from './account-with-platform.type';
|
||||||
import type { AccountWithValue } from './account-with-value.type';
|
import type { AccountWithValue } from './account-with-value.type';
|
||||||
|
import type { AiPromptMode } from './ai-prompt-mode.type';
|
||||||
import type { BenchmarkTrend } from './benchmark-trend.type';
|
import type { BenchmarkTrend } from './benchmark-trend.type';
|
||||||
import type { ColorScheme } from './color-scheme.type';
|
import type { ColorScheme } from './color-scheme.type';
|
||||||
import type { DateRange } from './date-range.type';
|
import type { DateRange } from './date-range.type';
|
||||||
@ -24,6 +25,7 @@ export type {
|
|||||||
AccessWithGranteeUser,
|
AccessWithGranteeUser,
|
||||||
AccountWithPlatform,
|
AccountWithPlatform,
|
||||||
AccountWithValue,
|
AccountWithValue,
|
||||||
|
AiPromptMode,
|
||||||
BenchmarkTrend,
|
BenchmarkTrend,
|
||||||
ColorScheme,
|
ColorScheme,
|
||||||
DateRange,
|
DateRange,
|
||||||
|
@ -70,9 +70,11 @@ const locales = {
|
|||||||
'South America': $localize`South America`,
|
'South America': $localize`South America`,
|
||||||
|
|
||||||
// Countries
|
// Countries
|
||||||
|
Armenia: $localize`Armenia`,
|
||||||
Australia: $localize`Australia`,
|
Australia: $localize`Australia`,
|
||||||
Austria: $localize`Austria`,
|
Austria: $localize`Austria`,
|
||||||
Belgium: $localize`Belgium`,
|
Belgium: $localize`Belgium`,
|
||||||
|
'British Virgin Islands': $localize`British Virgin Islands`,
|
||||||
Bulgaria: $localize`Bulgaria`,
|
Bulgaria: $localize`Bulgaria`,
|
||||||
Canada: $localize`Canada`,
|
Canada: $localize`Canada`,
|
||||||
'Czech Republic': $localize`Czech Republic`,
|
'Czech Republic': $localize`Czech Republic`,
|
||||||
@ -86,6 +88,7 @@ const locales = {
|
|||||||
'New Zealand': $localize`New Zealand`,
|
'New Zealand': $localize`New Zealand`,
|
||||||
Poland: $localize`Poland`,
|
Poland: $localize`Poland`,
|
||||||
Romania: $localize`Romania`,
|
Romania: $localize`Romania`,
|
||||||
|
Singapore: $localize`Singapore`,
|
||||||
'South Africa': $localize`South Africa`,
|
'South Africa': $localize`South Africa`,
|
||||||
Switzerland: $localize`Switzerland`,
|
Switzerland: $localize`Switzerland`,
|
||||||
Thailand: $localize`Thailand`,
|
Thailand: $localize`Thailand`,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user