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 { ConfigurationService } from '@ghostfolio/api/services/configuration/configuration.service'; import { ExchangeRateDataService } from '@ghostfolio/api/services/exchange-rate-data/exchange-rate-data.service'; import { PrismaService } from '@ghostfolio/api/services/prisma/prisma.service'; import { PropertyService } from '@ghostfolio/api/services/property/property.service'; import { TagService } from '@ghostfolio/api/services/tag/tag.service'; import { 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, PROPERTY_SYSTEM_MESSAGE, ghostfolioFearAndGreedIndexDataSource } from '@ghostfolio/common/config'; import { DATE_FORMAT, encodeDataSource, extractNumberFromString } from '@ghostfolio/common/helper'; import { InfoItem, Statistics, Subscription } from '@ghostfolio/common/interfaces'; import { permissions } from '@ghostfolio/common/permissions'; import { SubscriptionOffer } from '@ghostfolio/common/types'; import { Injectable, Logger } from '@nestjs/common'; import { JwtService } from '@nestjs/jwt'; import * as bent from 'bent'; import * as cheerio from 'cheerio'; import { format, subDays } from 'date-fns'; @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 prismaService: PrismaService, private readonly propertyService: PropertyService, private readonly redisCacheService: RedisCacheService, private readonly tagService: TagService ) {} public async get(): Promise { const info: Partial = {}; let isReadOnlyMode: boolean; const platforms = ( await this.platformService.getPlatforms({ orderBy: { name: 'asc' } }) ).map(({ id, name }) => { return { id, name }; }); let systemMessage: string; const globalPermissions: string[] = []; if (this.configurationService.get('ENABLE_FEATURE_BLOG')) { globalPermissions.push(permissions.enableBlog); } 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); systemMessage = (await this.propertyService.getByKey( PROPERTY_SYSTEM_MESSAGE )) as string; } const isUserSignupEnabled = await this.propertyService.isUserSignupEnabled(); if (isUserSignupEnabled) { globalPermissions.push(permissions.createUserAccount); } const [benchmarks, demoAuthToken, statistics, subscriptions, tags] = await Promise.all([ this.benchmarkService.getBenchmarkAssetProfiles(), this.getDemoAuthToken(), this.getStatistics(), this.getSubscriptions(), this.tagService.get() ]); return { ...info, benchmarks, demoAuthToken, globalPermissions, isReadOnlyMode, platforms, statistics, subscriptions, systemMessage, tags, baseCurrency: this.configurationService.get('BASE_CURRENCY'), currencies: this.exchangeRateDataService.getCurrencies() }; } private async countActiveUsers(aDays: number) { return await this.prismaService.user.count({ orderBy: { Analytics: { updatedAt: 'desc' } }, where: { AND: [ { NOT: { Analytics: null } }, { Analytics: { updatedAt: { gt: subDays(new Date(), aDays) } } } ] } }); } private async countDockerHubPulls(): Promise { try { const get = bent( `https://hub.docker.com/v2/repositories/ghostfolio/ghostfolio`, 'GET', 'json', 200, { 'User-Agent': 'request' } ); const { pull_count } = await get(); return pull_count; } catch (error) { Logger.error(error, 'InfoService'); return undefined; } } private async countGitHubContributors(): Promise { try { const get = bent( 'https://github.com/ghostfolio/ghostfolio', 'GET', 'string', 200, {} ); const html = await get(); const $ = cheerio.load(html); return extractNumberFromString( $( `a[href="/ghostfolio/ghostfolio/graphs/contributors"] .Counter` ).text() ); } catch (error) { Logger.error(error, 'InfoService'); return undefined; } } private async countGitHubStargazers(): Promise { try { const get = bent( `https://api.github.com/repos/ghostfolio/ghostfolio`, 'GET', 'json', 200, { 'User-Agent': 'request' } ); const { stargazers_count } = await get(); return stargazers_count; } catch (error) { Logger.error(error, 'InfoService'); return undefined; } } private async countNewUsers(aDays: number) { return await this.prismaService.user.count({ orderBy: { createdAt: 'desc' }, 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 getSubscriptions(): Promise<{ [offer in SubscriptionOffer]: Subscription; }> { if (!this.configurationService.get('ENABLE_FEATURE_SUBSCRIPTION')) { return undefined; } const stripeConfig = (await this.prismaService.property.findUnique({ where: { key: PROPERTY_STRIPE_CONFIG } })) ?? { value: '{}' }; return JSON.parse(stripeConfig.value); } private async getUptime(): Promise { { try { const monitorId = (await this.propertyService.getByKey( PROPERTY_BETTER_UPTIME_MONITOR_ID )) as string; const get = bent( `https://betteruptime.com/api/v2/monitors/${monitorId}/sla?from=${format( subDays(new Date(), 90), DATE_FORMAT )}&to${format(new Date(), DATE_FORMAT)}`, 'GET', 'json', 200, { Authorization: `Bearer ${this.configurationService.get( 'BETTER_UPTIME_API_KEY' )}` } ); const { data } = await get(); return data.attributes.availability / 100; } catch (error) { Logger.error(error, 'InfoService'); return undefined; } } } }