mirror of https://github.com/ghostfolio/ghostfolio
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
367 lines
10 KiB
367 lines
10 KiB
import { BenchmarkService } from '@ghostfolio/api/app/benchmark/benchmark.service';
|
|
import { PlatformService } from '@ghostfolio/api/app/platform/platform.service';
|
|
import { RedisCacheService } from '@ghostfolio/api/app/redis-cache/redis-cache.service';
|
|
import { UserService } from '@ghostfolio/api/app/user/user.service';
|
|
import { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service';
|
|
import { ExchangeRateDataService } from '@ghostfolio/api/services/exchange-rate-data/exchange-rate-data.service';
|
|
import { PropertyService } from '@ghostfolio/api/services/property/property.service';
|
|
import {
|
|
DEFAULT_CURRENCY,
|
|
PROPERTY_BETTER_UPTIME_MONITOR_ID,
|
|
PROPERTY_COUNTRIES_OF_SUBSCRIBERS,
|
|
PROPERTY_DEMO_USER_ID,
|
|
PROPERTY_IS_READ_ONLY_MODE,
|
|
PROPERTY_SLACK_COMMUNITY_USERS,
|
|
PROPERTY_STRIPE_CONFIG,
|
|
ghostfolioFearAndGreedIndexDataSource
|
|
} from '@ghostfolio/common/config';
|
|
import {
|
|
DATE_FORMAT,
|
|
encodeDataSource,
|
|
extractNumberFromString
|
|
} from '@ghostfolio/common/helper';
|
|
import {
|
|
InfoItem,
|
|
Statistics,
|
|
SubscriptionOffer
|
|
} from '@ghostfolio/common/interfaces';
|
|
import { permissions } from '@ghostfolio/common/permissions';
|
|
import { SubscriptionOfferKey } from '@ghostfolio/common/types';
|
|
|
|
import { Injectable, Logger } from '@nestjs/common';
|
|
import { JwtService } from '@nestjs/jwt';
|
|
import * as cheerio from 'cheerio';
|
|
import { format, subDays } from 'date-fns';
|
|
import got from 'got';
|
|
|
|
@Injectable()
|
|
export class InfoService {
|
|
private static CACHE_KEY_STATISTICS = 'STATISTICS';
|
|
|
|
public constructor(
|
|
private readonly benchmarkService: BenchmarkService,
|
|
private readonly configurationService: ConfigurationService,
|
|
private readonly exchangeRateDataService: ExchangeRateDataService,
|
|
private readonly jwtService: JwtService,
|
|
private readonly platformService: PlatformService,
|
|
private readonly propertyService: PropertyService,
|
|
private readonly redisCacheService: RedisCacheService,
|
|
private readonly userService: UserService
|
|
) {}
|
|
|
|
public async get(): Promise<InfoItem> {
|
|
const info: Partial<InfoItem> = {};
|
|
let isReadOnlyMode: boolean;
|
|
|
|
const globalPermissions: string[] = [];
|
|
|
|
if (this.configurationService.get('ENABLE_FEATURE_FEAR_AND_GREED_INDEX')) {
|
|
if (this.configurationService.get('ENABLE_FEATURE_SUBSCRIPTION')) {
|
|
info.fearAndGreedDataSource = encodeDataSource(
|
|
ghostfolioFearAndGreedIndexDataSource
|
|
);
|
|
} else {
|
|
info.fearAndGreedDataSource = ghostfolioFearAndGreedIndexDataSource;
|
|
}
|
|
|
|
globalPermissions.push(permissions.enableFearAndGreedIndex);
|
|
}
|
|
|
|
if (this.configurationService.get('ENABLE_FEATURE_READ_ONLY_MODE')) {
|
|
isReadOnlyMode = (await this.propertyService.getByKey(
|
|
PROPERTY_IS_READ_ONLY_MODE
|
|
)) as boolean;
|
|
}
|
|
|
|
if (this.configurationService.get('ENABLE_FEATURE_SOCIAL_LOGIN')) {
|
|
globalPermissions.push(permissions.enableSocialLogin);
|
|
}
|
|
|
|
if (this.configurationService.get('ENABLE_FEATURE_STATISTICS')) {
|
|
globalPermissions.push(permissions.enableStatistics);
|
|
}
|
|
|
|
if (this.configurationService.get('ENABLE_FEATURE_SUBSCRIPTION')) {
|
|
globalPermissions.push(permissions.enableSubscription);
|
|
|
|
info.countriesOfSubscribers =
|
|
((await this.propertyService.getByKey(
|
|
PROPERTY_COUNTRIES_OF_SUBSCRIBERS
|
|
)) as string[]) ?? [];
|
|
info.stripePublicKey = this.configurationService.get('STRIPE_PUBLIC_KEY');
|
|
}
|
|
|
|
if (this.configurationService.get('ENABLE_FEATURE_SYSTEM_MESSAGE')) {
|
|
globalPermissions.push(permissions.enableSystemMessage);
|
|
}
|
|
|
|
const [
|
|
benchmarks,
|
|
demoAuthToken,
|
|
isUserSignupEnabled,
|
|
platforms,
|
|
statistics,
|
|
subscriptionOffers
|
|
] = await Promise.all([
|
|
this.benchmarkService.getBenchmarkAssetProfiles(),
|
|
this.getDemoAuthToken(),
|
|
this.propertyService.isUserSignupEnabled(),
|
|
this.platformService.getPlatforms({
|
|
orderBy: { name: 'asc' }
|
|
}),
|
|
this.getStatistics(),
|
|
this.getSubscriptionOffers()
|
|
]);
|
|
|
|
if (isUserSignupEnabled) {
|
|
globalPermissions.push(permissions.createUserAccount);
|
|
}
|
|
|
|
return {
|
|
...info,
|
|
benchmarks,
|
|
demoAuthToken,
|
|
globalPermissions,
|
|
isReadOnlyMode,
|
|
platforms,
|
|
statistics,
|
|
subscriptionOffers,
|
|
baseCurrency: DEFAULT_CURRENCY,
|
|
currencies: this.exchangeRateDataService.getCurrencies()
|
|
};
|
|
}
|
|
|
|
private async countActiveUsers(aDays: number) {
|
|
return this.userService.count({
|
|
where: {
|
|
AND: [
|
|
{
|
|
NOT: {
|
|
Analytics: null
|
|
}
|
|
},
|
|
{
|
|
Analytics: {
|
|
updatedAt: {
|
|
gt: subDays(new Date(), aDays)
|
|
}
|
|
}
|
|
}
|
|
]
|
|
}
|
|
});
|
|
}
|
|
|
|
private async countDockerHubPulls(): Promise<number> {
|
|
try {
|
|
const abortController = new AbortController();
|
|
|
|
setTimeout(() => {
|
|
abortController.abort();
|
|
}, this.configurationService.get('REQUEST_TIMEOUT'));
|
|
|
|
const { pull_count } = await got(
|
|
`https://hub.docker.com/v2/repositories/ghostfolio/ghostfolio`,
|
|
{
|
|
headers: { 'User-Agent': 'request' },
|
|
// @ts-ignore
|
|
signal: abortController.signal
|
|
}
|
|
).json<any>();
|
|
|
|
return pull_count;
|
|
} catch (error) {
|
|
Logger.error(error, 'InfoService - DockerHub');
|
|
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
private async countGitHubContributors(): Promise<number> {
|
|
try {
|
|
const abortController = new AbortController();
|
|
|
|
setTimeout(() => {
|
|
abortController.abort();
|
|
}, this.configurationService.get('REQUEST_TIMEOUT'));
|
|
|
|
const { body } = await got('https://github.com/ghostfolio/ghostfolio', {
|
|
// @ts-ignore
|
|
signal: abortController.signal
|
|
});
|
|
|
|
const $ = cheerio.load(body);
|
|
|
|
return extractNumberFromString({
|
|
value: $(
|
|
`a[href="/ghostfolio/ghostfolio/graphs/contributors"] .Counter`
|
|
).text()
|
|
});
|
|
} catch (error) {
|
|
Logger.error(error, 'InfoService - GitHub');
|
|
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
private async countGitHubStargazers(): Promise<number> {
|
|
try {
|
|
const abortController = new AbortController();
|
|
|
|
setTimeout(() => {
|
|
abortController.abort();
|
|
}, this.configurationService.get('REQUEST_TIMEOUT'));
|
|
|
|
const { stargazers_count } = await got(
|
|
`https://api.github.com/repos/ghostfolio/ghostfolio`,
|
|
{
|
|
headers: { 'User-Agent': 'request' },
|
|
// @ts-ignore
|
|
signal: abortController.signal
|
|
}
|
|
).json<any>();
|
|
|
|
return stargazers_count;
|
|
} catch (error) {
|
|
Logger.error(error, 'InfoService - GitHub');
|
|
|
|
return undefined;
|
|
}
|
|
}
|
|
|
|
private async countNewUsers(aDays: number) {
|
|
return this.userService.count({
|
|
where: {
|
|
AND: [
|
|
{
|
|
NOT: {
|
|
Analytics: null
|
|
}
|
|
},
|
|
{
|
|
createdAt: {
|
|
gt: subDays(new Date(), aDays)
|
|
}
|
|
}
|
|
]
|
|
}
|
|
});
|
|
}
|
|
|
|
private async countSlackCommunityUsers() {
|
|
return (await this.propertyService.getByKey(
|
|
PROPERTY_SLACK_COMMUNITY_USERS
|
|
)) as string;
|
|
}
|
|
|
|
private async getDemoAuthToken() {
|
|
const demoUserId = (await this.propertyService.getByKey(
|
|
PROPERTY_DEMO_USER_ID
|
|
)) as string;
|
|
|
|
if (demoUserId) {
|
|
return this.jwtService.sign({
|
|
id: demoUserId
|
|
});
|
|
}
|
|
|
|
return undefined;
|
|
}
|
|
|
|
private async getStatistics() {
|
|
if (!this.configurationService.get('ENABLE_FEATURE_STATISTICS')) {
|
|
return undefined;
|
|
}
|
|
|
|
let statistics: Statistics;
|
|
|
|
try {
|
|
statistics = JSON.parse(
|
|
await this.redisCacheService.get(InfoService.CACHE_KEY_STATISTICS)
|
|
);
|
|
|
|
if (statistics) {
|
|
return statistics;
|
|
}
|
|
} catch {}
|
|
|
|
const activeUsers1d = await this.countActiveUsers(1);
|
|
const activeUsers30d = await this.countActiveUsers(30);
|
|
const newUsers30d = await this.countNewUsers(30);
|
|
|
|
const dockerHubPulls = await this.countDockerHubPulls();
|
|
const gitHubContributors = await this.countGitHubContributors();
|
|
const gitHubStargazers = await this.countGitHubStargazers();
|
|
const slackCommunityUsers = await this.countSlackCommunityUsers();
|
|
const uptime = await this.getUptime();
|
|
|
|
statistics = {
|
|
activeUsers1d,
|
|
activeUsers30d,
|
|
dockerHubPulls,
|
|
gitHubContributors,
|
|
gitHubStargazers,
|
|
newUsers30d,
|
|
slackCommunityUsers,
|
|
uptime
|
|
};
|
|
|
|
await this.redisCacheService.set(
|
|
InfoService.CACHE_KEY_STATISTICS,
|
|
JSON.stringify(statistics)
|
|
);
|
|
|
|
return statistics;
|
|
}
|
|
|
|
private async getSubscriptionOffers(): Promise<{
|
|
[offer in SubscriptionOfferKey]: SubscriptionOffer;
|
|
}> {
|
|
if (!this.configurationService.get('ENABLE_FEATURE_SUBSCRIPTION')) {
|
|
return undefined;
|
|
}
|
|
|
|
return (
|
|
((await this.propertyService.getByKey(PROPERTY_STRIPE_CONFIG)) as any) ??
|
|
{}
|
|
);
|
|
}
|
|
|
|
private async getUptime(): Promise<number> {
|
|
{
|
|
try {
|
|
const monitorId = (await this.propertyService.getByKey(
|
|
PROPERTY_BETTER_UPTIME_MONITOR_ID
|
|
)) as string;
|
|
|
|
const abortController = new AbortController();
|
|
|
|
setTimeout(() => {
|
|
abortController.abort();
|
|
}, this.configurationService.get('REQUEST_TIMEOUT'));
|
|
|
|
const { data } = await got(
|
|
`https://uptime.betterstack.com/api/v2/monitors/${monitorId}/sla?from=${format(
|
|
subDays(new Date(), 90),
|
|
DATE_FORMAT
|
|
)}&to${format(new Date(), DATE_FORMAT)}`,
|
|
{
|
|
headers: {
|
|
Authorization: `Bearer ${this.configurationService.get(
|
|
'API_KEY_BETTER_UPTIME'
|
|
)}`
|
|
},
|
|
// @ts-ignore
|
|
signal: abortController.signal
|
|
}
|
|
).json<any>();
|
|
|
|
return data.attributes.availability / 100;
|
|
} catch (error) {
|
|
Logger.error(error, 'InfoService - Better Stack');
|
|
|
|
return undefined;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|