Compare commits
6 Commits
443ff55295
...
4368beebff
Author | SHA1 | Date | |
---|---|---|---|
4368beebff | |||
|
7ca0720244 | ||
|
6bdfd8984f | ||
|
52081d6741 | ||
|
cb1f488eb4 | ||
|
25d0c1c8a0 |
@ -5,10 +5,11 @@ 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
|
## 2.145.0 - 2025-03-09
|
||||||
|
|
||||||
### Added
|
### Added
|
||||||
|
|
||||||
|
- Extended the export functionality by the account balances
|
||||||
- Added a _Copy portfolio data to clipboard for AI prompt_ action to the analysis page (experimental)
|
- Added a _Copy portfolio data to clipboard for AI prompt_ action to the analysis page (experimental)
|
||||||
|
|
||||||
### Changed
|
### Changed
|
||||||
@ -17,6 +18,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
|
|||||||
- Improved the language localization for German (`de`)
|
- Improved the language localization for German (`de`)
|
||||||
- Upgraded `@simplewebauthn/browser` and `@simplewebauthn/server` from version `9.0` to `13.1`
|
- Upgraded `@simplewebauthn/browser` and `@simplewebauthn/server` from version `9.0` to `13.1`
|
||||||
|
|
||||||
|
### Fixed
|
||||||
|
|
||||||
|
- Fixed an issue to get dividends in the _Financial Modeling Prep_ service
|
||||||
|
- Fixed an issue to get historical market data in the _Financial Modeling Prep_ service
|
||||||
|
- Fixed an issue with serving _Storybook_
|
||||||
|
|
||||||
## 2.144.0 - 2025-03-06
|
## 2.144.0 - 2025-03-06
|
||||||
|
|
||||||
### Fixed
|
### Fixed
|
||||||
|
@ -7,7 +7,13 @@ import { Filter } from '@ghostfolio/common/interfaces';
|
|||||||
|
|
||||||
import { Injectable } from '@nestjs/common';
|
import { Injectable } from '@nestjs/common';
|
||||||
import { EventEmitter2 } from '@nestjs/event-emitter';
|
import { EventEmitter2 } from '@nestjs/event-emitter';
|
||||||
import { Account, Order, Platform, Prisma } from '@prisma/client';
|
import {
|
||||||
|
Account,
|
||||||
|
AccountBalance,
|
||||||
|
Order,
|
||||||
|
Platform,
|
||||||
|
Prisma
|
||||||
|
} from '@prisma/client';
|
||||||
import { Big } from 'big.js';
|
import { Big } from 'big.js';
|
||||||
import { format } from 'date-fns';
|
import { format } from 'date-fns';
|
||||||
import { groupBy } from 'lodash';
|
import { groupBy } from 'lodash';
|
||||||
@ -56,13 +62,19 @@ export class AccountService {
|
|||||||
orderBy?: Prisma.AccountOrderByWithRelationInput;
|
orderBy?: Prisma.AccountOrderByWithRelationInput;
|
||||||
}): Promise<
|
}): Promise<
|
||||||
(Account & {
|
(Account & {
|
||||||
|
balances?: AccountBalance[];
|
||||||
Order?: Order[];
|
Order?: Order[];
|
||||||
Platform?: Platform;
|
Platform?: Platform;
|
||||||
})[]
|
})[]
|
||||||
> {
|
> {
|
||||||
const { include = {}, skip, take, cursor, where, orderBy } = params;
|
const { include = {}, skip, take, cursor, where, orderBy } = params;
|
||||||
|
|
||||||
include.balances = { orderBy: { date: 'desc' }, take: 1 };
|
const isBalancesIncluded = !!include.balances;
|
||||||
|
|
||||||
|
include.balances = {
|
||||||
|
orderBy: { date: 'desc' },
|
||||||
|
...(isBalancesIncluded ? {} : { take: 1 })
|
||||||
|
};
|
||||||
|
|
||||||
const accounts = await this.prismaService.account.findMany({
|
const accounts = await this.prismaService.account.findMany({
|
||||||
cursor,
|
cursor,
|
||||||
@ -76,7 +88,9 @@ export class AccountService {
|
|||||||
return accounts.map((account) => {
|
return accounts.map((account) => {
|
||||||
account = { ...account, balance: account.balances[0]?.value ?? 0 };
|
account = { ...account, balance: account.balances[0]?.value ?? 0 };
|
||||||
|
|
||||||
delete account.balances;
|
if (!isBalancesIncluded) {
|
||||||
|
delete account.balances;
|
||||||
|
}
|
||||||
|
|
||||||
return account;
|
return account;
|
||||||
});
|
});
|
||||||
|
@ -31,6 +31,7 @@ export class ExportService {
|
|||||||
const accounts = (
|
const accounts = (
|
||||||
await this.accountService.accounts({
|
await this.accountService.accounts({
|
||||||
include: {
|
include: {
|
||||||
|
balances: true,
|
||||||
Platform: true
|
Platform: true
|
||||||
},
|
},
|
||||||
orderBy: {
|
orderBy: {
|
||||||
@ -41,6 +42,7 @@ export class ExportService {
|
|||||||
).map(
|
).map(
|
||||||
({
|
({
|
||||||
balance,
|
balance,
|
||||||
|
balances,
|
||||||
comment,
|
comment,
|
||||||
currency,
|
currency,
|
||||||
id,
|
id,
|
||||||
@ -55,6 +57,9 @@ export class ExportService {
|
|||||||
|
|
||||||
return {
|
return {
|
||||||
balance,
|
balance,
|
||||||
|
balances: balances.map(({ date, value }) => {
|
||||||
|
return { date: date.toISOString(), value };
|
||||||
|
}),
|
||||||
comment,
|
comment,
|
||||||
currency,
|
currency,
|
||||||
id,
|
id,
|
||||||
|
@ -129,6 +129,7 @@ export const HtmlTemplateMiddleware = async (
|
|||||||
|
|
||||||
if (
|
if (
|
||||||
path.startsWith('/api/') ||
|
path.startsWith('/api/') ||
|
||||||
|
path.startsWith('/development/storybook') ||
|
||||||
isFileRequest(path) ||
|
isFileRequest(path) ||
|
||||||
!environment.production
|
!environment.production
|
||||||
) {
|
) {
|
||||||
|
@ -244,7 +244,7 @@ export class FinancialModelingPrepService implements DataProviderInterface {
|
|||||||
[date: string]: IDataProviderHistoricalResponse;
|
[date: string]: IDataProviderHistoricalResponse;
|
||||||
} = {};
|
} = {};
|
||||||
|
|
||||||
const { historical } = await fetch(
|
const { historical = [] } = await fetch(
|
||||||
`${this.URL}/historical-price-full/stock_dividend/${symbol}?apikey=${this.apiKey}`,
|
`${this.URL}/historical-price-full/stock_dividend/${symbol}?apikey=${this.apiKey}`,
|
||||||
{
|
{
|
||||||
signal: AbortSignal.timeout(requestTimeout)
|
signal: AbortSignal.timeout(requestTimeout)
|
||||||
@ -305,7 +305,7 @@ export class FinancialModelingPrepService implements DataProviderInterface {
|
|||||||
? addYears(currentFrom, MAX_YEARS_PER_REQUEST)
|
? addYears(currentFrom, MAX_YEARS_PER_REQUEST)
|
||||||
: to;
|
: to;
|
||||||
|
|
||||||
const { historical } = await fetch(
|
const { historical = [] } = await fetch(
|
||||||
`${this.URL}/historical-price-full/${symbol}?apikey=${this.apiKey}&from=${format(currentFrom, DATE_FORMAT)}&to=${format(currentTo, DATE_FORMAT)}`,
|
`${this.URL}/historical-price-full/${symbol}?apikey=${this.apiKey}&from=${format(currentFrom, DATE_FORMAT)}&to=${format(currentTo, DATE_FORMAT)}`,
|
||||||
{
|
{
|
||||||
signal: AbortSignal.timeout(requestTimeout)
|
signal: AbortSignal.timeout(requestTimeout)
|
||||||
|
@ -40,8 +40,8 @@ export class GfApiPageComponent implements OnInit {
|
|||||||
this.apiKey = prompt($localize`Please enter your Ghostfolio API key:`);
|
this.apiKey = prompt($localize`Please enter your Ghostfolio API key:`);
|
||||||
|
|
||||||
this.dividends$ = this.fetchDividends({ symbol: 'KO' });
|
this.dividends$ = this.fetchDividends({ symbol: 'KO' });
|
||||||
this.historicalData$ = this.fetchHistoricalData({ symbol: 'AAPL.US' });
|
this.historicalData$ = this.fetchHistoricalData({ symbol: 'AAPL' });
|
||||||
this.quotes$ = this.fetchQuotes({ symbols: ['AAPL.US', 'VOO.US'] });
|
this.quotes$ = this.fetchQuotes({ symbols: ['AAPL', 'VOO.US'] });
|
||||||
this.status$ = this.fetchStatus();
|
this.status$ = this.fetchStatus();
|
||||||
this.symbols$ = this.fetchSymbols({ query: 'apple' });
|
this.symbols$ = this.fetchSymbols({ query: 'apple' });
|
||||||
}
|
}
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
import { Account, Order, Platform, Tag } from '@prisma/client';
|
import { Account, Order, Platform, Tag } from '@prisma/client';
|
||||||
|
|
||||||
export interface Export {
|
export interface Export {
|
||||||
accounts: Omit<Account, 'createdAt' | 'updatedAt' | 'userId'>[];
|
accounts: (Omit<Account, 'createdAt' | 'updatedAt' | 'userId'> & {
|
||||||
|
balances: { date: string; value: number }[];
|
||||||
|
})[];
|
||||||
activities: (Omit<
|
activities: (Omit<
|
||||||
Order,
|
Order,
|
||||||
| 'accountUserId'
|
| 'accountUserId'
|
||||||
|
4
package-lock.json
generated
4
package-lock.json
generated
@ -1,12 +1,12 @@
|
|||||||
{
|
{
|
||||||
"name": "ghostfolio",
|
"name": "ghostfolio",
|
||||||
"version": "2.144.0",
|
"version": "2.145.0",
|
||||||
"lockfileVersion": 3,
|
"lockfileVersion": 3,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"packages": {
|
"packages": {
|
||||||
"": {
|
"": {
|
||||||
"name": "ghostfolio",
|
"name": "ghostfolio",
|
||||||
"version": "2.144.0",
|
"version": "2.145.0",
|
||||||
"hasInstallScript": true,
|
"hasInstallScript": true,
|
||||||
"license": "AGPL-3.0",
|
"license": "AGPL-3.0",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "ghostfolio",
|
"name": "ghostfolio",
|
||||||
"version": "2.144.0",
|
"version": "2.145.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",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user