├── mobile ├── .idea │ ├── .name │ ├── codeStyles │ │ ├── codeStyleConfig.xml │ │ └── Project.xml │ ├── compiler.xml │ ├── vcs.xml │ ├── gradle.xml │ └── misc.xml ├── app │ ├── src │ │ └── main │ │ │ ├── res │ │ │ ├── values │ │ │ │ ├── strings.xml │ │ │ │ ├── colors.xml │ │ │ │ └── themes.xml │ │ │ ├── mipmap-hdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-mdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-xhdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-xxhdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-xxxhdpi │ │ │ │ ├── ic_launcher.png │ │ │ │ └── ic_launcher_round.png │ │ │ ├── mipmap-anydpi-v26 │ │ │ │ ├── ic_launcher.xml │ │ │ │ └── ic_launcher_round.xml │ │ │ ├── drawable-v24 │ │ │ │ └── ic_launcher_foreground.xml │ │ │ ├── values-night │ │ │ │ └── themes.xml │ │ │ ├── drawable │ │ │ │ └── ic_launcher_background.xml │ │ │ └── layout │ │ │ │ └── activity_main.xml │ │ │ ├── java │ │ │ └── dev │ │ │ │ └── kognise │ │ │ │ └── ministalker │ │ │ │ ├── BootHandler.kt │ │ │ │ ├── MainActivity.kt │ │ │ │ └── Worker.kt │ │ │ └── AndroidManifest.xml │ ├── proguard-rules.pro │ └── build.gradle ├── gradle │ └── wrapper │ │ ├── gradle-wrapper.jar │ │ └── gradle-wrapper.properties ├── gradle.properties ├── build.gradle ├── settings.gradle ├── gradlew.bat └── gradlew ├── .dockerignore ├── .prettierrc.json ├── Dockerfile ├── extension ├── manifest.json └── background.js ├── stalkerd ├── Cargo.toml ├── src │ ├── input.rs │ ├── processes.rs │ └── main.rs └── Cargo.lock ├── tsconfig.json ├── src ├── polling.ts ├── modules │ ├── toggl.ts │ ├── lastfm.ts │ ├── fsp.ts │ └── calendar.ts ├── env.ts ├── util.ts └── index.ts ├── .gitignore ├── fly.toml ├── package.json ├── prisma └── schema.prisma ├── LICENSE ├── README.md ├── dash.html └── yarn.lock /mobile/.idea/.name: -------------------------------------------------------------------------------- 1 | MiniStalker -------------------------------------------------------------------------------- /.dockerignore: -------------------------------------------------------------------------------- 1 | node_modules 2 | dist 3 | stalkerd 4 | extension 5 | mobile 6 | .env -------------------------------------------------------------------------------- /mobile/app/src/main/res/values/strings.xml: -------------------------------------------------------------------------------- 1 | 2 | MiniStalker 3 | -------------------------------------------------------------------------------- /mobile/gradle/wrapper/gradle-wrapper.jar: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/gradle/wrapper/gradle-wrapper.jar -------------------------------------------------------------------------------- /.prettierrc.json: -------------------------------------------------------------------------------- 1 | { 2 | "semi": false, 3 | "singleQuote": true, 4 | "useTabs": true, 5 | "printWidth": 120, 6 | "trailingComma": "none" 7 | } -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-hdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-hdpi/ic_launcher.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-mdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-mdpi/ic_launcher.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-xhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-xhdpi/ic_launcher.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-xxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-xxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-xxxhdpi/ic_launcher.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-hdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-hdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-mdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-mdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-xhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/kognise/stalker/HEAD/mobile/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.png -------------------------------------------------------------------------------- /mobile/gradle.properties: -------------------------------------------------------------------------------- 1 | org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8 2 | android.useAndroidX=true 3 | kotlin.code.style=official 4 | android.nonTransitiveRClass=true -------------------------------------------------------------------------------- /Dockerfile: -------------------------------------------------------------------------------- 1 | FROM node:alpine 2 | 3 | COPY package.json ./ 4 | COPY yarn.lock ./ 5 | RUN yarn install 6 | 7 | COPY . . 8 | RUN yarn build 9 | 10 | EXPOSE 3000 11 | CMD [ "yarn", "start" ] -------------------------------------------------------------------------------- /mobile/.idea/codeStyles/codeStyleConfig.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 5 | -------------------------------------------------------------------------------- /mobile/.idea/compiler.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /mobile/.idea/vcs.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | -------------------------------------------------------------------------------- /mobile/app/src/main/res/values/colors.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | #FF000000 4 | #FFFFFFFF 5 | #FFFF0000 6 | -------------------------------------------------------------------------------- /mobile/gradle/wrapper/gradle-wrapper.properties: -------------------------------------------------------------------------------- 1 | #Wed Apr 27 23:02:31 CDT 2022 2 | distributionBase=GRADLE_USER_HOME 3 | distributionUrl=https\://services.gradle.org/distributions/gradle-7.2-bin.zip 4 | distributionPath=wrapper/dists 5 | zipStorePath=wrapper/dists 6 | zipStoreBase=GRADLE_USER_HOME 7 | -------------------------------------------------------------------------------- /mobile/build.gradle: -------------------------------------------------------------------------------- 1 | plugins { 2 | id 'com.android.application' version '7.1.3' apply false 3 | id 'com.android.library' version '7.1.3' apply false 4 | id 'org.jetbrains.kotlin.android' version '1.5.30' apply false 5 | } 6 | 7 | task clean(type: Delete) { 8 | delete rootProject.buildDir 9 | } -------------------------------------------------------------------------------- /extension/manifest.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "Stalker", 3 | "version": "0.1.0", 4 | "description": "A thing that watches everything everything I do", 5 | "manifest_version": 3, 6 | "permissions": [ 7 | "tabs", 8 | "alarms", 9 | "storage" 10 | ], 11 | "background": { 12 | "service_worker": "background.js" 13 | } 14 | } -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /mobile/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | -------------------------------------------------------------------------------- /stalkerd/Cargo.toml: -------------------------------------------------------------------------------- 1 | [package] 2 | name = "stalkerd" 3 | version = "0.1.0" 4 | edition = "2021" 5 | 6 | [dependencies] 7 | daemonize = "0.4.1" 8 | home = "0.5.3" 9 | reqwest = { version = "0.11.10", features = ["blocking", "json"] } 10 | serde = { version = "1.0.136", features = ["derive"] } 11 | sysinfo = "0.23.10" 12 | urlencoding = "2.1.0" 13 | 14 | [features] 15 | -------------------------------------------------------------------------------- /stalkerd/src/input.rs: -------------------------------------------------------------------------------- 1 | #[link(name = "CoreGraphics", kind = "framework")] 2 | extern "C" { 3 | // https://developer.apple.com/documentation/coregraphics/1408794-cgeventsourcecounterforeventtype 4 | pub fn CGEventSourceCounterForEventType(sourceState: u32, eventType: u32) -> u32; 5 | } 6 | 7 | pub fn last_input_tick() -> usize { 8 | unsafe { CGEventSourceCounterForEventType(0, !0) as usize } 9 | } 10 | -------------------------------------------------------------------------------- /mobile/settings.gradle: -------------------------------------------------------------------------------- 1 | pluginManagement { 2 | repositories { 3 | gradlePluginPortal() 4 | google() 5 | mavenCentral() 6 | } 7 | } 8 | dependencyResolutionManagement { 9 | repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS) 10 | repositories { 11 | google() 12 | mavenCentral() 13 | } 14 | } 15 | rootProject.name = "MiniStalker" 16 | include ':app' 17 | -------------------------------------------------------------------------------- /mobile/app/src/main/res/drawable-v24/ic_launcher_foreground.xml: -------------------------------------------------------------------------------- 1 | 6 | 9 | 10 | -------------------------------------------------------------------------------- /tsconfig.json: -------------------------------------------------------------------------------- 1 | { 2 | "compilerOptions": { 3 | "incremental": true, 4 | "target": "es2020", 5 | "module": "es2020", 6 | "moduleResolution": "node", 7 | "esModuleInterop": true, 8 | "allowSyntheticDefaultImports": true, 9 | "forceConsistentCasingInFileNames": true, 10 | "strict": true, 11 | "skipLibCheck": true, 12 | "declaration": false, 13 | "outDir": "dist" 14 | }, 15 | "include": [ 16 | "src/**/*" 17 | ], 18 | } 19 | -------------------------------------------------------------------------------- /mobile/app/proguard-rules.pro: -------------------------------------------------------------------------------- 1 | # If your project uses WebView with JS, uncomment the following 2 | # and specify the fully qualified class name to the JavaScript interface 3 | # class: 4 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview { 5 | # public *; 6 | #} 7 | 8 | # Uncomment this to preserve the line number information for 9 | # debugging stack traces. 10 | #-keepattributes SourceFile,LineNumberTable 11 | 12 | # If you keep the line number information, uncomment this to 13 | # hide the original source file name. 14 | #-renamesourcefileattribute SourceFile -------------------------------------------------------------------------------- /mobile/app/src/main/res/values/themes.xml: -------------------------------------------------------------------------------- 1 | 2 | 13 | -------------------------------------------------------------------------------- /src/polling.ts: -------------------------------------------------------------------------------- 1 | import { getCalendarState } from './modules/calendar.js' 2 | import { getFspState } from './modules/fsp.js' 3 | import { getLastfmState } from './modules/lastfm.js' 4 | import { getTogglState } from './modules/toggl.js' 5 | 6 | export const pollingStateRegistry = { 7 | fsp: { fetcher: getFspState, cron: '10 */15 * * * *' }, 8 | toggl: { fetcher: getTogglState, cron: '0 */5 * * * *' }, 9 | lastfm: { fetcher: getLastfmState, cron: '*/10 * * * * *' }, 10 | calendar: { fetcher: getCalendarState, cron: '10 */15 * * * *' } 11 | } 12 | 13 | export type PollingState = { 14 | [key in keyof typeof pollingStateRegistry]: Awaited> 15 | } 16 | -------------------------------------------------------------------------------- /mobile/app/src/main/res/values-night/themes.xml: -------------------------------------------------------------------------------- 1 | 2 | 13 | -------------------------------------------------------------------------------- /mobile/app/src/main/res/drawable/ic_launcher_background.xml: -------------------------------------------------------------------------------- 1 | 7 | 8 | 9 | 15 | 16 | 17 | 18 | 19 | 20 | 21 | -------------------------------------------------------------------------------- /mobile/.idea/gradle.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 19 | 20 | -------------------------------------------------------------------------------- /mobile/app/src/main/java/dev/kognise/ministalker/BootHandler.kt: -------------------------------------------------------------------------------- 1 | package dev.kognise.ministalker 2 | 3 | import android.content.BroadcastReceiver 4 | import android.content.Context 5 | import android.content.Intent 6 | import android.util.Log 7 | import androidx.work.ExistingWorkPolicy 8 | import androidx.work.OneTimeWorkRequest 9 | import androidx.work.WorkManager 10 | 11 | class BootHandler : BroadcastReceiver() { 12 | override fun onReceive(context: Context, intent: Intent) { 13 | if (intent.action != Intent.ACTION_BOOT_COMPLETED) return 14 | Log.d("BootHandler", "Relaunching after boot!") 15 | 16 | val workManager = WorkManager.getInstance(context) 17 | val workRequest = OneTimeWorkRequest.from(Worker::class.java) 18 | workManager.enqueueUniqueWork("worker", ExistingWorkPolicy.KEEP, workRequest) 19 | } 20 | } -------------------------------------------------------------------------------- /src/modules/toggl.ts: -------------------------------------------------------------------------------- 1 | import fetch from 'node-fetch' 2 | import env from '../env.js' 3 | 4 | interface TogglEntry { 5 | id: number 6 | wid: number 7 | billable: boolean 8 | start: string 9 | duration: number 10 | duronly: boolean 11 | at: string 12 | uid: number 13 | } 14 | 15 | interface TogglCurrentEntry { 16 | data: TogglEntry | null 17 | } 18 | 19 | export interface TogglState { 20 | tracking: boolean 21 | } 22 | 23 | export const getTogglState = async (): Promise => { 24 | const res = await fetch('https://api.track.toggl.com/api/v8/time_entries/current', { 25 | headers: { 26 | Authorization: `Basic ${Buffer.from(`${env.togglApiKey}:api_token`).toString('base64')}` 27 | } 28 | }) 29 | if (!res.ok) throw new Error(`Status ${res.status} while fetching Toggl current entry`) 30 | const json = (await res.json()) as TogglCurrentEntry 31 | return { 32 | tracking: json.data !== null 33 | } 34 | } 35 | -------------------------------------------------------------------------------- /.gitignore: -------------------------------------------------------------------------------- 1 | .DS_Store 2 | 3 | node_modules 4 | .env 5 | dist 6 | tsconfig.tsbuildinfo 7 | 8 | mobile/local.properties 9 | mobile/build 10 | mobile/app/build 11 | mobile/captures 12 | mobile/*.iml 13 | mobile/.gradle 14 | mobile/.externalNativeBuild 15 | mobile/.cxx 16 | 17 | mobile/.idea/**/workspace.xml 18 | mobile/.idea/**/tasks.xml 19 | mobile/.idea/**/usage.statistics.xml 20 | mobile/.idea/**/dictionaries 21 | mobile/.idea/**/shelf 22 | mobile/.idea/**/contentModel.xml 23 | mobile/.idea/**/dataSources/ 24 | mobile/.idea/**/dataSources.ids 25 | mobile/.idea/**/dataSources.local.xml 26 | mobile/.idea/**/sqlDataSources.xml 27 | mobile/.idea/**/dynamic.xml 28 | mobile/.idea/**/uiDesigner.xml 29 | mobile/.idea/**/dbnavigator.xml 30 | mobile/.idea/**/gradle.xml 31 | mobile/.idea/**/libraries 32 | mobile/.idea/modules.xml 33 | mobile/.idea/*.iml 34 | mobile/.idea/assetWizardSettings.xml 35 | mobile/.idea/modules 36 | 37 | stalkerd/target 38 | stalkerd/.password -------------------------------------------------------------------------------- /fly.toml: -------------------------------------------------------------------------------- 1 | app = "stalker" 2 | 3 | kill_signal = "SIGINT" 4 | kill_timeout = 5 5 | processes = [] 6 | 7 | [experimental] 8 | allowed_public_ports = [] 9 | auto_rollback = true 10 | 11 | [[services]] 12 | internal_port = 3000 13 | processes = ["app"] 14 | protocol = "tcp" 15 | script_checks = [] 16 | 17 | [[services.http_checks]] 18 | grace_period = "30s" 19 | interval = 10000 20 | method = "get" 21 | path = "/" 22 | protocol = "http" 23 | restart_limit = 0 24 | timeout = 2000 25 | 26 | [services.concurrency] 27 | hard_limit = 25 28 | soft_limit = 20 29 | type = "connections" 30 | 31 | [[services.ports]] 32 | force_https = true 33 | handlers = ["http"] 34 | port = 80 35 | 36 | [[services.ports]] 37 | handlers = ["tls", "http"] 38 | port = 443 39 | 40 | [[services.tcp_checks]] 41 | grace_period = "1s" 42 | interval = "15s" 43 | restart_limit = 0 44 | timeout = "2s" 45 | -------------------------------------------------------------------------------- /package.json: -------------------------------------------------------------------------------- 1 | { 2 | "name": "stalker", 3 | "version": "0.1.0", 4 | "main": "index.js", 5 | "license": "MIT", 6 | "type": "module", 7 | "scripts": { 8 | "dev": "nodemon src/index.ts --exec ts-node-esm", 9 | "build": "prisma generate && tsc", 10 | "start": "prisma db push --skip-generate && node dist/index.js", 11 | "fmt": "prettier --write ./src" 12 | }, 13 | "dependencies": { 14 | "@fastify/cors": "^7.0.0", 15 | "@prisma/client": "^3.13.0", 16 | "cookie": "^0.5.0", 17 | "cron": "^1.8.2", 18 | "dotenv": "^16.0.0", 19 | "fastify": "^3.28.0", 20 | "node-fetch": "^3.2.3", 21 | "nodemon": "^2.0.15" 22 | }, 23 | "devDependencies": { 24 | "@types/cookie": "^0.5.1", 25 | "@types/cron": "^1.7.3", 26 | "@types/node": "^17.0.29", 27 | "pino-pretty": "^7.6.1", 28 | "prettier": "^2.6.2", 29 | "prisma": "^3.13.0", 30 | "ts-node": "^10.7.0", 31 | "typescript": "^4.6.3" 32 | } 33 | } -------------------------------------------------------------------------------- /prisma/schema.prisma: -------------------------------------------------------------------------------- 1 | generator client { 2 | provider = "prisma-client-js" 3 | } 4 | 5 | datasource db { 6 | provider = "postgresql" 7 | url = env("DATABASE_URL") 8 | } 9 | 10 | model OauthToken { 11 | refreshToken String @id 12 | accessToken String 13 | expiration DateTime? 14 | } 15 | 16 | model Activity { 17 | time DateTime @id @default(now()) 18 | emoji String 19 | label String 20 | } 21 | 22 | model ManualActivity { 23 | id String @id @default(cuid()) 24 | time DateTime 25 | emoji String 26 | label String 27 | } 28 | 29 | model Ping { 30 | key String @id 31 | time DateTime 32 | } 33 | 34 | model List { 35 | id String @id @default(cuid()) 36 | key String 37 | sourceDevice String 38 | time DateTime 39 | list String[] 40 | 41 | @@unique([key, sourceDevice]) 42 | } 43 | 44 | model ZoomUser { 45 | id String @id 46 | lastUpdate DateTime 47 | inMeeting Boolean 48 | } 49 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | Copyright 2022 Kognise 2 | 3 | Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: 4 | 5 | The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. 6 | 7 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. 8 | -------------------------------------------------------------------------------- /src/env.ts: -------------------------------------------------------------------------------- 1 | import dotenv from 'dotenv' 2 | dotenv.config() 3 | 4 | const getVarOrPanic = (key: string): string => { 5 | if (!process.env[key]) { 6 | throw new Error(`Missing environment variable '${key}'`) 7 | } 8 | return process.env[key]! 9 | } 10 | 11 | const env = { 12 | password: getVarOrPanic('PASSWORD'), 13 | slackToken: getVarOrPanic('SLACK_TOKEN'), 14 | fspUsername: getVarOrPanic('FSP_USERNAME'), 15 | fspPassword: getVarOrPanic('FSP_PASSWORD'), 16 | fspOperatorId: parseInt(getVarOrPanic('FSP_OPERATOR_ID')), 17 | togglApiKey: getVarOrPanic('TOGGL_API_KEY'), 18 | lastfmUsername: getVarOrPanic('LASTFM_USERNAME'), 19 | lastfmApiKey: getVarOrPanic('LASTFM_API_KEY'), 20 | calendarIds: getVarOrPanic('CALENDAR_IDS').split(','), 21 | calendarRefreshToken: getVarOrPanic('CALENDAR_REFRESH_TOKEN'), 22 | calendarClientId: getVarOrPanic('CALENDAR_CLIENT_ID'), 23 | calendarClientSecret: getVarOrPanic('CALENDAR_CLIENT_SECRET'), 24 | zoomUserId: getVarOrPanic('ZOOM_USER_ID'), 25 | zoomVerificationToken: getVarOrPanic('ZOOM_VERIFICATION_TOKEN') 26 | } 27 | export default env 28 | -------------------------------------------------------------------------------- /mobile/app/build.gradle: -------------------------------------------------------------------------------- 1 | plugins { 2 | id 'com.android.application' 3 | id 'org.jetbrains.kotlin.android' 4 | } 5 | 6 | android { 7 | compileSdk 32 8 | 9 | defaultConfig { 10 | applicationId "dev.kognise.ministalker" 11 | minSdk 29 12 | targetSdk 32 13 | versionCode 1 14 | versionName "1.0" 15 | } 16 | 17 | buildTypes { 18 | release { 19 | minifyEnabled false 20 | proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro' 21 | } 22 | } 23 | compileOptions { 24 | sourceCompatibility JavaVersion.VERSION_1_8 25 | targetCompatibility JavaVersion.VERSION_1_8 26 | } 27 | kotlinOptions { 28 | jvmTarget = '1.8' 29 | } 30 | } 31 | 32 | dependencies { 33 | implementation 'com.google.android.material:material:1.5.0' 34 | implementation 'androidx.core:core-ktx:1.7.0' 35 | implementation 'androidx.appcompat:appcompat:1.4.1' 36 | implementation 'androidx.constraintlayout:constraintlayout:2.1.3' 37 | implementation 'androidx.work:work-runtime-ktx:2.7.1' 38 | 39 | implementation 'io.ktor:ktor-client-core:2.0.0' 40 | implementation 'io.ktor:ktor-client-cio:2.0.0' 41 | } -------------------------------------------------------------------------------- /extension/background.js: -------------------------------------------------------------------------------- 1 | const getId = async () => { 2 | const { id } = await chrome.storage.local.get(['id']) 3 | if (id) return id 4 | const newId = Date.now().toString(36) + Math.random().toString(36).slice(2) 5 | await chrome.storage.local.set({ id: newId }) 6 | return newId 7 | } 8 | 9 | const update = async () => { 10 | const { password } = await chrome.storage.sync.get(['password']) 11 | if (!password) throw new Error("No password set! Try: chrome.storage.sync.set({ password: '...' })") 12 | const id = await getId() 13 | 14 | const tabs = await chrome.tabs.query({}) 15 | const domains = tabs 16 | .map((tab) => new URL(tab.url)) 17 | .filter((url) => ['http:', 'https:'].includes(url.protocol)) 18 | .map((url) => url.hostname.toLowerCase()) 19 | .map((hostname) => (hostname.startsWith('www.') ? hostname.slice(4) : hostname)) 20 | const uniqueDomains = [...new Set(domains)] 21 | 22 | await fetch(`https://api.kognise.dev/list/domains/${id}`, { 23 | method: 'POST', 24 | headers: { 25 | 'Content-Type': 'application/json', 26 | Authorization: `Bearer ${password}` 27 | }, 28 | body: JSON.stringify({ list: uniqueDomains }) 29 | }) 30 | } 31 | 32 | chrome.runtime.onInstalled.addListener(() => { 33 | update() 34 | chrome.alarms.create({ periodInMinutes: 1 }) 35 | }) 36 | 37 | chrome.tabs.onRemoved.addListener(update) 38 | chrome.alarms.onAlarm.addListener(update) 39 | -------------------------------------------------------------------------------- /mobile/.idea/misc.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 17 | 18 | 19 | 20 | 21 | 22 | 24 | -------------------------------------------------------------------------------- /stalkerd/src/processes.rs: -------------------------------------------------------------------------------- 1 | use sysinfo::{ProcessExt, System, SystemExt}; 2 | 3 | pub fn collect_processes(system: &mut System) -> Vec<&'static str> { 4 | system.refresh_processes(); 5 | 6 | let mut collected = vec![]; 7 | for process in system.processes().values() { 8 | match process.name() { 9 | "Live" => collected.push("ableton"), 10 | "Terminal" => collected.push("terminal"), 11 | "mscore" => collected.push("musescore"), 12 | "Max" => collected.push("max"), 13 | "zoom.us" => collected.push("zoom"), 14 | "Figma" => collected.push("figma"), 15 | "studio" => collected.push("android-studio"), 16 | "Celeste" => collected.push("celeste"), 17 | "factorio" => collected.push("factorio"), 18 | "RimWorld by Ludeon Studios" => collected.push("rimworld"), 19 | "java" => { 20 | if process.cmd().iter().any(|c| c.contains("minecraft")) { 21 | collected.push("minecraft") 22 | } 23 | } 24 | "Electron" => { 25 | if process 26 | .exe() 27 | .ends_with("Visual Studio Code.app/Contents/MacOS/Electron") 28 | { 29 | collected.push("vscode"); 30 | } 31 | } 32 | _ => {} 33 | } 34 | } 35 | collected 36 | } 37 | -------------------------------------------------------------------------------- /mobile/app/src/main/AndroidManifest.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 14 | 15 | 20 | 21 | 22 | 23 | 24 | 25 | 26 | 27 | 28 | 29 | 30 | 31 | 32 | 33 | -------------------------------------------------------------------------------- /mobile/app/src/main/res/layout/activity_main.xml: -------------------------------------------------------------------------------- 1 | 2 | 8 | 9 | 21 | 22 | 34 | 35 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Stalker 😌 2 | 3 | You know that thing on my website that shows what I'm doing? Well, it used to be manually updated, but that got boring. 4 | 5 | Now I have a complex system of janky programs that culminate in a bunch of if-then statements that generate that status! It even updates my Slack. I'm unstoppable. 6 | 7 | ## API 8 | 9 | Consumable at and deployed on the glorious [Fly.io](https://fly.io/). 10 | 11 | **Public endpoints:** 12 | 13 | - `/` - Get current activity and now playing song 14 | - `/history` - Get full activity history 15 | - `/dash` - Access web dashboard (only I can update statuses) 16 | 17 | **Kognise-only endpoints:** 18 | 19 | - `/manual` - Manually update status 20 | - `/manual/clear` - Clear manual status 21 | - `/ping/:key` - Signify activity for a device class 22 | - `/list/:key/:sourceDevice` - Update browser tabs or running apps 23 | - `/zoom` - Zoom webhook updates 24 | 25 | ## Other 26 | 27 | Compile StalkerMini (mobile app) to an APK with Gradle, I just used Android Studio. 28 | 29 | Stalkerd (desktop daemon) needs a `.password` file to compile, which Rust embeds in the binary. Don't share the binary! I made this technical decision because I'm lazy. 30 | 31 | The browser extension is quite cursed. If it doesn't work, you'll have to open the dev console; the error message should explain how to set the password. 32 | 33 | Stalkerd is not cross-platform yet, only supports macOS. 34 | 35 | ## Todo 36 | 37 | - Some way to let me know if something is breaking on the backend 38 | - Figure out why StalkerMini sometimes gets yeeted by the system 39 | - Maybe switch to Spotify API for now playing 40 | - Improve code quality (lmao as if) 41 | -------------------------------------------------------------------------------- /mobile/app/src/main/java/dev/kognise/ministalker/MainActivity.kt: -------------------------------------------------------------------------------- 1 | package dev.kognise.ministalker 2 | 3 | import android.content.Context 4 | import android.content.Intent 5 | import android.content.SharedPreferences 6 | import androidx.appcompat.app.AppCompatActivity 7 | import android.os.Bundle 8 | import android.text.Editable 9 | import android.text.TextWatcher 10 | import android.widget.TextView 11 | import androidx.work.ExistingWorkPolicy 12 | import androidx.work.OneTimeWorkRequest 13 | import androidx.work.WorkManager 14 | 15 | class MainActivity : AppCompatActivity() { 16 | private lateinit var preferences: SharedPreferences 17 | 18 | override fun onCreate(savedInstanceState: Bundle?) { 19 | super.onCreate(savedInstanceState) 20 | setContentView(R.layout.activity_main) 21 | preferences = getSharedPreferences( 22 | "${BuildConfig.APPLICATION_ID}.prefs", 23 | Context.MODE_PRIVATE 24 | ) 25 | 26 | // Sync password field with preferences. 27 | val passwordField = findViewById(R.id.password) 28 | passwordField.text = preferences.getString("password", "")!! 29 | passwordField.addTextChangedListener(textWatcher) 30 | 31 | // Start the background service. 32 | val workManager = WorkManager.getInstance(this) 33 | val workRequest = OneTimeWorkRequest.from(Worker::class.java) 34 | workManager.enqueueUniqueWork("worker", ExistingWorkPolicy.REPLACE, workRequest) 35 | } 36 | 37 | private val textWatcher = object : TextWatcher { 38 | override fun onTextChanged(text: CharSequence?, start: Int, before: Int, count: Int) { 39 | preferences.edit().putString("password", text.toString()).apply() 40 | } 41 | 42 | override fun beforeTextChanged(text: CharSequence?, start: Int, before: Int, after: Int) {} 43 | override fun afterTextChanged(editable: Editable?) {} 44 | } 45 | } -------------------------------------------------------------------------------- /src/modules/lastfm.ts: -------------------------------------------------------------------------------- 1 | import fetch from 'node-fetch' 2 | import env from '../env.js' 3 | import { makeUrl } from '../util.js' 4 | 5 | interface LastfmTrack { 6 | artist: { 7 | mbid: string 8 | '#text': string 9 | } 10 | streamable: string 11 | image: { 12 | size: string 13 | '#text': string 14 | }[] 15 | mbid: string 16 | album: { 17 | mbid: string 18 | '#text': string 19 | } 20 | name: string 21 | '@attr': { 22 | nowplaying: string 23 | } 24 | url: string 25 | } 26 | 27 | interface LastfmRecentTracks { 28 | recenttracks: { 29 | track: LastfmTrack[] 30 | '@attr': { 31 | user: string 32 | totalPages: string 33 | page: string 34 | perPage: string 35 | total: string 36 | } 37 | } 38 | } 39 | 40 | const getRecentTracks = async (username: string, apiKey: string, limit: number): Promise => { 41 | const res = await fetch( 42 | makeUrl(`https://ws.audioscrobbler.com/2.0/`, { 43 | user: username, 44 | api_key: apiKey, 45 | format: 'json', 46 | method: 'user.getrecenttracks', 47 | limit 48 | }) 49 | ) 50 | if (!res.ok) throw new Error(`Status ${res.status} while fetching recent tracks`) 51 | return (await res.json()) as LastfmRecentTracks 52 | } 53 | 54 | export interface LastfmState { 55 | nowPlaying: boolean 56 | track: { 57 | name: string 58 | artist: string 59 | album: string 60 | images: string[] 61 | url: string 62 | } 63 | } 64 | 65 | export const getLastfmState = async (): Promise => { 66 | const recentTracks = await getRecentTracks(env.lastfmUsername, env.lastfmApiKey, 1) 67 | const track = recentTracks.recenttracks.track[0] 68 | if (!track) throw new Error('No Last.fm recent tracks!') 69 | return { 70 | nowPlaying: track['@attr'] ? track['@attr'].nowplaying === 'true' : false, 71 | track: { 72 | name: track.name, 73 | artist: track.artist['#text'], 74 | album: track.album['#text'], 75 | images: track.image.map((image) => image['#text']), 76 | url: track.url 77 | } 78 | } 79 | } 80 | -------------------------------------------------------------------------------- /src/util.ts: -------------------------------------------------------------------------------- 1 | import Prisma from '@prisma/client' 2 | import { preHandlerHookHandler } from 'fastify' 3 | import env from './env.js' 4 | 5 | export const expiration = 2 * 60 * 1000 // 2 minutes 6 | export const zoomExpiration = 12 * 60 * 60 * 1000 // 12 hours 7 | export const sleepExpiration = 60 * 60 * 1000 // 1 hour 8 | export const sleepExpirationDelay = 6 * 60 * 1000 // 6 minutes 9 | 10 | export const makeUrl = (url: string, params: Record = {}): string => { 11 | const newUrl = new URL(url) 12 | for (const [key, value] of Object.entries(params)) { 13 | newUrl.searchParams.set(key, value.toString()) 14 | } 15 | return newUrl.toString() 16 | } 17 | 18 | export const pingDiff = async (key: string): Promise => { 19 | const ping = await prisma.ping.findUnique({ where: { key } }) 20 | return ping ? Date.now() - ping.time.getTime() : Infinity 21 | } 22 | 23 | export const getList = async (key: string): Promise => { 24 | const lists = await prisma.list.findMany({ 25 | where: { 26 | key, 27 | time: { gte: new Date(Date.now() - expiration) } 28 | } 29 | }) 30 | return lists.reduce((acc, { list }) => [...acc, ...list], []) 31 | } 32 | 33 | export const isSleepingHours = (date: Date): boolean => { 34 | const hours = date.getUTCHours() 35 | return hours >= 3 && hours <= 12 36 | } 37 | 38 | export const isInZoomMeeting = async (): Promise => { 39 | const zoomUser = await prisma.zoomUser.findUnique({ where: { id: env.zoomUserId.toLowerCase() } }) 40 | if (!zoomUser) return false 41 | if (Date.now() - zoomUser.lastUpdate.getTime() > zoomExpiration) return false 42 | return zoomUser.inMeeting 43 | } 44 | 45 | export interface Activity { 46 | emoji: string 47 | label: string 48 | } 49 | 50 | export const prisma = new Prisma.PrismaClient() 51 | 52 | export const requirePassword: preHandlerHookHandler = async (req, reply) => { 53 | const prefix = 'Bearer ' 54 | if ( 55 | !req.headers.authorization || 56 | req.headers.authorization.length <= prefix.length || 57 | req.headers.authorization.slice(prefix.length) !== env.password 58 | ) { 59 | reply.status(401) 60 | throw new Error('Incorrect password') 61 | } 62 | } 63 | -------------------------------------------------------------------------------- /mobile/gradlew.bat: -------------------------------------------------------------------------------- 1 | @rem 2 | @rem Copyright 2015 the original author or authors. 3 | @rem 4 | @rem Licensed under the Apache License, Version 2.0 (the "License"); 5 | @rem you may not use this file except in compliance with the License. 6 | @rem You may obtain a copy of the License at 7 | @rem 8 | @rem https://www.apache.org/licenses/LICENSE-2.0 9 | @rem 10 | @rem Unless required by applicable law or agreed to in writing, software 11 | @rem distributed under the License is distributed on an "AS IS" BASIS, 12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 | @rem See the License for the specific language governing permissions and 14 | @rem limitations under the License. 15 | @rem 16 | 17 | @if "%DEBUG%" == "" @echo off 18 | @rem ########################################################################## 19 | @rem 20 | @rem Gradle startup script for Windows 21 | @rem 22 | @rem ########################################################################## 23 | 24 | @rem Set local scope for the variables with windows NT shell 25 | if "%OS%"=="Windows_NT" setlocal 26 | 27 | set DIRNAME=%~dp0 28 | if "%DIRNAME%" == "" set DIRNAME=. 29 | set APP_BASE_NAME=%~n0 30 | set APP_HOME=%DIRNAME% 31 | 32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter. 33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi 34 | 35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" 37 | 38 | @rem Find java.exe 39 | if defined JAVA_HOME goto findJavaFromJavaHome 40 | 41 | set JAVA_EXE=java.exe 42 | %JAVA_EXE% -version >NUL 2>&1 43 | if "%ERRORLEVEL%" == "0" goto execute 44 | 45 | echo. 46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 47 | echo. 48 | echo Please set the JAVA_HOME variable in your environment to match the 49 | echo location of your Java installation. 50 | 51 | goto fail 52 | 53 | :findJavaFromJavaHome 54 | set JAVA_HOME=%JAVA_HOME:"=% 55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe 56 | 57 | if exist "%JAVA_EXE%" goto execute 58 | 59 | echo. 60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% 61 | echo. 62 | echo Please set the JAVA_HOME variable in your environment to match the 63 | echo location of your Java installation. 64 | 65 | goto fail 66 | 67 | :execute 68 | @rem Setup the command line 69 | 70 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar 71 | 72 | 73 | @rem Execute Gradle 74 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* 75 | 76 | :end 77 | @rem End local scope for the variables with windows NT shell 78 | if "%ERRORLEVEL%"=="0" goto mainEnd 79 | 80 | :fail 81 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of 82 | rem the _cmd.exe /c_ return code! 83 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1 84 | exit /b 1 85 | 86 | :mainEnd 87 | if "%OS%"=="Windows_NT" endlocal 88 | 89 | :omega 90 | -------------------------------------------------------------------------------- /stalkerd/src/main.rs: -------------------------------------------------------------------------------- 1 | #![feature(never_type)] 2 | #![feature(try_blocks)] 3 | 4 | mod input; 5 | mod processes; 6 | 7 | use std::fs::File; 8 | use std::thread::sleep; 9 | use std::time::Duration; 10 | 11 | use daemonize::Daemonize; 12 | use home::home_dir; 13 | use reqwest::blocking::Client as HttpClient; 14 | use serde::Serialize; 15 | use sysinfo::System; 16 | use sysinfo::SystemExt; 17 | use urlencoding::encode; 18 | 19 | use crate::input::last_input_tick; 20 | use crate::processes::collect_processes; 21 | 22 | const PASSWORD: &str = include_str!("../.password"); 23 | 24 | #[derive(Serialize)] 25 | struct List<'a> { 26 | list: Vec<&'a str>, 27 | } 28 | 29 | fn main() { 30 | let home = home_dir().expect("Failed to get home directory"); 31 | let stdout = 32 | File::create(home.join(".stalkerd.log")).expect("Failed to open logfile for stdout"); 33 | let stderr = File::options() 34 | .append(true) 35 | .open(home.join(".stalkerd.log")) 36 | .expect("Failed to open logfile for stderr"); 37 | let daemonize = Daemonize::new() 38 | .pid_file(home.join(".stalkerd.pid")) 39 | .stdout(stdout) 40 | .stderr(stderr); 41 | 42 | println!("Starting daemon..."); 43 | match daemonize.start() { 44 | Ok(_) => println!("Daemon started!"), 45 | Err(err) => panic!("Error starting daemon: {}", err), 46 | } 47 | 48 | let mut system = System::default(); 49 | let hostname = system.host_name().expect("No hostname"); 50 | let http = HttpClient::new(); 51 | 52 | let mut tick = 0; 53 | let mut last_apps_empty = false; // To avoid duplicate empty requests 54 | 55 | loop { 56 | let mut delay = Duration::from_secs(60); 57 | 58 | // My rust-analyzer broke with try_blocks for some reason :( 59 | let mut go = || -> Result<(), reqwest::Error> { 60 | let new_tick = last_input_tick(); 61 | if new_tick != tick { 62 | // Ping backend: 63 | tick = new_tick; 64 | http.post("https://api.kognise.dev/ping/desktop") 65 | .bearer_auth(PASSWORD) 66 | .send()?; 67 | 68 | // Update process list: 69 | let apps = collect_processes(&mut system); 70 | let apps_empty = apps.is_empty(); 71 | if !(apps_empty && last_apps_empty) { 72 | http.post(format!( 73 | "https://api.kognise.dev/list/apps/{}", 74 | encode(&hostname) 75 | )) 76 | .bearer_auth(PASSWORD) 77 | .json(&List { list: apps }) 78 | .send()?; 79 | } 80 | last_apps_empty = apps_empty; 81 | } else { 82 | delay = Duration::from_secs(10); 83 | } 84 | Ok(()) 85 | }; 86 | 87 | if let Err(err) = go() { 88 | eprintln!("{}", err); 89 | delay = Duration::from_secs(30); 90 | } 91 | println!("Next tick in {:?}", delay); 92 | sleep(delay); 93 | } 94 | } 95 | -------------------------------------------------------------------------------- /mobile/app/src/main/java/dev/kognise/ministalker/Worker.kt: -------------------------------------------------------------------------------- 1 | package dev.kognise.ministalker 2 | 3 | import android.app.KeyguardManager 4 | import android.app.NotificationChannel 5 | import android.app.NotificationManager 6 | import android.content.Context 7 | import android.os.PowerManager 8 | import android.util.Log 9 | import androidx.core.app.NotificationCompat 10 | import androidx.work.CoroutineWorker 11 | import androidx.work.ForegroundInfo 12 | import androidx.work.WorkerParameters 13 | import io.ktor.client.* 14 | import io.ktor.client.engine.cio.* 15 | import io.ktor.client.request.* 16 | import io.ktor.client.statement.* 17 | import kotlinx.coroutines.delay 18 | 19 | class Worker(context: Context, parameters: WorkerParameters) : CoroutineWorker(context, parameters) { 20 | private val notificationManager = context.getSystemService(Context.NOTIFICATION_SERVICE) as NotificationManager 21 | private val keyguardManager = context.getSystemService(Context.KEYGUARD_SERVICE) as KeyguardManager 22 | private val powerManager = context.getSystemService(Context.POWER_SERVICE) as PowerManager 23 | 24 | private val preferences = context.getSharedPreferences( 25 | "${BuildConfig.APPLICATION_ID}.prefs", 26 | Context.MODE_PRIVATE 27 | ) 28 | 29 | override suspend fun doWork(): Nothing { 30 | setForeground(createForegroundInfo()) 31 | val http = HttpClient(CIO) { expectSuccess = true } 32 | 33 | while (true) { 34 | val password = preferences.getString("password", "")!! 35 | if (password.isEmpty()) { 36 | delay(10000) 37 | continue 38 | } 39 | 40 | var delaySecs: Long = 60 41 | try { 42 | if (isPhoneActive()) { 43 | http.post { 44 | url("https://api.kognise.dev/ping/mobile") 45 | bearerAuth(password) 46 | } 47 | } else { 48 | delaySecs = 10 49 | } 50 | } catch (e: Exception) { 51 | Log.e("Worker", "${e.message}") 52 | delaySecs = 30 53 | } 54 | 55 | Log.d("Worker", "Next tick in ${delaySecs}s") 56 | delay(delaySecs * 1000) 57 | } 58 | } 59 | 60 | private fun isPhoneActive(): Boolean { 61 | return powerManager.isInteractive && !keyguardManager.isKeyguardLocked 62 | } 63 | 64 | private fun createForegroundInfo(): ForegroundInfo { 65 | notificationManager.createNotificationChannel( 66 | NotificationChannel( 67 | "worker", 68 | "Background worker", 69 | NotificationManager.IMPORTANCE_MIN 70 | ).apply { setShowBadge(false) } 71 | ) 72 | val notification = NotificationCompat.Builder(applicationContext, "worker") 73 | .setContentTitle("MiniStalker") 74 | .setContentText("I'm watching you...") 75 | .setSmallIcon(R.drawable.ic_launcher_foreground) 76 | .setOngoing(true) 77 | .build() 78 | 79 | notificationManager.createNotificationChannel( 80 | NotificationChannel( 81 | "default", 82 | "MiniStalker", 83 | NotificationManager.IMPORTANCE_DEFAULT 84 | ) 85 | ) 86 | 87 | val notificationId = (System.currentTimeMillis() % Int.MAX_VALUE).toInt() 88 | return ForegroundInfo(notificationId, notification) 89 | } 90 | } -------------------------------------------------------------------------------- /src/modules/fsp.ts: -------------------------------------------------------------------------------- 1 | // https://www.flightschedulepro.com/ 2 | // A lot of the interfaces are incomplete but they're fine for our purposes. 3 | 4 | import fetch from 'node-fetch' 5 | import cookie from 'cookie' 6 | import env from '../env.js' 7 | import { makeUrl } from '../util.js' 8 | 9 | interface FspApp { 10 | token: string 11 | operatorId: number 12 | } 13 | 14 | interface FspReservationResults { 15 | total: number 16 | pageIndex: number 17 | pageSize: number 18 | results: { 19 | id: string 20 | start: string 21 | foreground: string 22 | background: string 23 | }[] 24 | } 25 | 26 | interface FspReservation { 27 | reservationId: string 28 | reservationNumber: number 29 | start: string 30 | end: string 31 | foreground: string 32 | background: string 33 | estimatedFlightHours: number 34 | instructorPostFlightMinutes: number 35 | instructorPreFlightMinutes: number 36 | flightRoute: string 37 | locationName: string 38 | } 39 | 40 | const login = async (operatorId: number, username: string, password: string): Promise => { 41 | const formData = new URLSearchParams() 42 | formData.set('username', username) 43 | formData.set('password', password) 44 | formData.set('uv_login', '0') 45 | formData.set('uv_ssl', '0') 46 | 47 | const res = await fetch(`https://app.flightschedulepro.com/Account/Login/${operatorId}`, { 48 | method: 'POST', 49 | headers: { 50 | 'Content-Type': 'application/x-www-form-urlencoded' 51 | }, 52 | body: formData, 53 | redirect: 'manual' 54 | }) 55 | const parsed = cookie.parse(res.headers.get('set-cookie') ?? '') 56 | if (!parsed.FspApp) throw new Error(`FSP login failed: no FspApp cookie (status ${res.status})`) 57 | return JSON.parse(parsed.FspApp) 58 | } 59 | 60 | const getReservationResults = async ( 61 | app: FspApp, 62 | pageIndex: number, 63 | pageSize: number 64 | ): Promise => { 65 | const url = makeUrl( 66 | `https://api.flightschedulepro.com/api/V1/operator/${app.operatorId}/dashboard/upcomingreservations`, 67 | { 68 | pageIndex, 69 | pageSize 70 | } 71 | ) 72 | const res = await fetch(url, { 73 | headers: { 74 | Authorization: `Basic ${app.token}` 75 | } 76 | }) 77 | if (!res.ok) throw new Error(`Status ${res.status} while fetching FSP reservation results`) 78 | return (await res.json()) as FspReservationResults 79 | } 80 | 81 | const getReservation = async (app: FspApp, id: string): Promise => { 82 | const url = makeUrl(`https://api.flightschedulepro.com/api/V2/Reservation/${id}`, { operatorId: app.operatorId }) 83 | const res = await fetch(url, { 84 | headers: { 85 | Authorization: `Bearer ${app.token}` 86 | } 87 | }) 88 | if (!res.ok) throw new Error(`Status ${res.status} while fetching FSP reservation`) 89 | return (await res.json()) as FspReservation 90 | } 91 | 92 | export interface FspState { 93 | inReservation: boolean 94 | } 95 | 96 | export const getFspState = async (): Promise => { 97 | const app = await login(env.fspOperatorId, env.fspUsername, env.fspPassword) 98 | const reservations = await getReservationResults(app, 1, 1) 99 | if (!reservations.results[0]) return { inReservation: false } 100 | 101 | const reservation = await getReservation(app, reservations.results[0].id) 102 | const [start, end] = [reservation.start, reservation.end].map((timestamp) => timestamp + '-05:00').map(Date.parse) 103 | const now = Date.now() 104 | 105 | return { 106 | inReservation: now >= start && now <= end 107 | } 108 | } 109 | -------------------------------------------------------------------------------- /src/modules/calendar.ts: -------------------------------------------------------------------------------- 1 | import fetch from 'node-fetch' 2 | import env from '../env.js' 3 | import { makeUrl, prisma } from '../util.js' 4 | 5 | interface EventItem { 6 | kind: string 7 | etag: string 8 | id: string 9 | status: string 10 | htmlLink: string 11 | created: string 12 | updated: string 13 | summary: string 14 | description: string 15 | location: string 16 | creator: { 17 | email: string 18 | self: boolean 19 | } 20 | organizer: { 21 | email: string 22 | self: boolean 23 | } 24 | start: { 25 | dateTime: string 26 | timeZone: string 27 | } 28 | end: { 29 | dateTime: string 30 | timeZone: string 31 | } 32 | iCalUID: string 33 | sequence: number 34 | reminders: { 35 | useDefault: boolean 36 | } 37 | eventType: string 38 | conferenceData?: { 39 | entryPoints: { 40 | entryPointType: string 41 | uri: string 42 | label: string 43 | }[] 44 | conferenceSolution: { 45 | name: string 46 | iconUri: string 47 | } 48 | } 49 | } 50 | 51 | interface TokenGrant { 52 | access_token: string 53 | expires_in: number 54 | token_type: string 55 | refresh_token: string 56 | } 57 | 58 | const updateAccessToken = async (refreshToken: string): Promise => { 59 | const res = await fetch('https://oauth2.googleapis.com/token', { 60 | method: 'POST', 61 | headers: { 62 | 'Content-Type': 'application/json' 63 | }, 64 | body: JSON.stringify({ 65 | client_id: env.calendarClientId, 66 | client_secret: env.calendarClientSecret, 67 | grant_type: 'refresh_token', 68 | refresh_token: refreshToken 69 | }) 70 | }) 71 | if (!res.ok) throw new Error(`Status ${res.status} while refreshing calendar token`) 72 | 73 | const grant = (await res.json()) as TokenGrant 74 | const expiration = new Date(Date.now() + grant.expires_in * 1000 - 1000) 75 | await prisma.oauthToken.upsert({ 76 | where: { refreshToken }, 77 | update: { 78 | accessToken: grant.access_token, 79 | expiration 80 | }, 81 | create: { 82 | refreshToken, 83 | accessToken: grant.access_token, 84 | expiration 85 | } 86 | }) 87 | return grant.access_token 88 | } 89 | 90 | const getCalendarEvents = async ( 91 | calendarId: string, 92 | startTime: Date, 93 | maxResults: number, 94 | accessToken: string 95 | ): Promise => { 96 | const res = await fetch( 97 | makeUrl(`https://www.googleapis.com/calendar/v3/calendars/${encodeURIComponent(calendarId)}/events`, { 98 | maxResults, 99 | orderBy: 'startTime', 100 | singleEvents: true, 101 | timeMin: startTime.toISOString() 102 | }), 103 | { 104 | headers: { 105 | Authorization: `Bearer ${accessToken}` 106 | } 107 | } 108 | ) 109 | if (!res.ok) throw new Error(`Status ${res.status} while fetching calendar events`) 110 | const json = (await res.json()) as { items: EventItem[] } 111 | return json.items 112 | } 113 | 114 | export interface CalendarState { 115 | eventName: string | null 116 | isVideoMeeting: boolean 117 | } 118 | 119 | export const getCalendarState = async (): Promise => { 120 | for (const calendarId of env.calendarIds) { 121 | let events 122 | try { 123 | const token = await prisma.oauthToken.findUnique({ where: { refreshToken: env.calendarRefreshToken } }) 124 | if (!token) throw new Error('No calendar access token found') 125 | events = await getCalendarEvents(calendarId, new Date(), 1, token.accessToken) 126 | } catch (err) { 127 | const accessToken = await updateAccessToken(env.calendarRefreshToken) 128 | events = await getCalendarEvents(calendarId, new Date(), 1, accessToken) 129 | } 130 | 131 | if (events[0]?.status !== 'confirmed') continue 132 | 133 | const [start, end] = [events[0].start.dateTime, events[0].end.dateTime].map(Date.parse) 134 | const now = Date.now() 135 | if (now < start || now > end) continue 136 | 137 | return { 138 | eventName: events[0].summary ?? '(No title)', 139 | isVideoMeeting: 140 | events[0].conferenceData?.entryPoints?.some((entry) => entry.entryPointType === 'video') || 141 | events[0].location?.includes('zoom.us') || 142 | false 143 | } 144 | } 145 | 146 | return { eventName: null, isVideoMeeting: false } 147 | } 148 | -------------------------------------------------------------------------------- /mobile/.idea/codeStyles/Project.xml: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 6 | 7 | 119 | 120 | 122 | 123 | -------------------------------------------------------------------------------- /dash.html: -------------------------------------------------------------------------------- 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | Stalker Dashboard 9 | 10 | 11 | 12 | 13 | 14 | 15 | 16 | 108 | 109 | 110 | 111 |
112 |

113 | status: 114 | 🤔 loading... 115 |

116 | 117 |
118 |

enter password:

119 | 120 |
121 |
122 | 123 | 227 | 228 | 229 | -------------------------------------------------------------------------------- /mobile/gradlew: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | # 4 | # Copyright 2015 the original author or authors. 5 | # 6 | # Licensed under the Apache License, Version 2.0 (the "License"); 7 | # you may not use this file except in compliance with the License. 8 | # You may obtain a copy of the License at 9 | # 10 | # https://www.apache.org/licenses/LICENSE-2.0 11 | # 12 | # Unless required by applicable law or agreed to in writing, software 13 | # distributed under the License is distributed on an "AS IS" BASIS, 14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 15 | # See the License for the specific language governing permissions and 16 | # limitations under the License. 17 | # 18 | 19 | ############################################################################## 20 | ## 21 | ## Gradle start up script for UN*X 22 | ## 23 | ############################################################################## 24 | 25 | # Attempt to set APP_HOME 26 | # Resolve links: $0 may be a link 27 | PRG="$0" 28 | # Need this for relative symlinks. 29 | while [ -h "$PRG" ] ; do 30 | ls=`ls -ld "$PRG"` 31 | link=`expr "$ls" : '.*-> \(.*\)$'` 32 | if expr "$link" : '/.*' > /dev/null; then 33 | PRG="$link" 34 | else 35 | PRG=`dirname "$PRG"`"/$link" 36 | fi 37 | done 38 | SAVED="`pwd`" 39 | cd "`dirname \"$PRG\"`/" >/dev/null 40 | APP_HOME="`pwd -P`" 41 | cd "$SAVED" >/dev/null 42 | 43 | APP_NAME="Gradle" 44 | APP_BASE_NAME=`basename "$0"` 45 | 46 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. 47 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' 48 | 49 | # Use the maximum available, or set MAX_FD != -1 to use that value. 50 | MAX_FD="maximum" 51 | 52 | warn () { 53 | echo "$*" 54 | } 55 | 56 | die () { 57 | echo 58 | echo "$*" 59 | echo 60 | exit 1 61 | } 62 | 63 | # OS specific support (must be 'true' or 'false'). 64 | cygwin=false 65 | msys=false 66 | darwin=false 67 | nonstop=false 68 | case "`uname`" in 69 | CYGWIN* ) 70 | cygwin=true 71 | ;; 72 | Darwin* ) 73 | darwin=true 74 | ;; 75 | MINGW* ) 76 | msys=true 77 | ;; 78 | NONSTOP* ) 79 | nonstop=true 80 | ;; 81 | esac 82 | 83 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar 84 | 85 | 86 | # Determine the Java command to use to start the JVM. 87 | if [ -n "$JAVA_HOME" ] ; then 88 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then 89 | # IBM's JDK on AIX uses strange locations for the executables 90 | JAVACMD="$JAVA_HOME/jre/sh/java" 91 | else 92 | JAVACMD="$JAVA_HOME/bin/java" 93 | fi 94 | if [ ! -x "$JAVACMD" ] ; then 95 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME 96 | 97 | Please set the JAVA_HOME variable in your environment to match the 98 | location of your Java installation." 99 | fi 100 | else 101 | JAVACMD="java" 102 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. 103 | 104 | Please set the JAVA_HOME variable in your environment to match the 105 | location of your Java installation." 106 | fi 107 | 108 | # Increase the maximum file descriptors if we can. 109 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then 110 | MAX_FD_LIMIT=`ulimit -H -n` 111 | if [ $? -eq 0 ] ; then 112 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then 113 | MAX_FD="$MAX_FD_LIMIT" 114 | fi 115 | ulimit -n $MAX_FD 116 | if [ $? -ne 0 ] ; then 117 | warn "Could not set maximum file descriptor limit: $MAX_FD" 118 | fi 119 | else 120 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" 121 | fi 122 | fi 123 | 124 | # For Darwin, add options to specify how the application appears in the dock 125 | if $darwin; then 126 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" 127 | fi 128 | 129 | # For Cygwin or MSYS, switch paths to Windows format before running java 130 | if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then 131 | APP_HOME=`cygpath --path --mixed "$APP_HOME"` 132 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` 133 | 134 | JAVACMD=`cygpath --unix "$JAVACMD"` 135 | 136 | # We build the pattern for arguments to be converted via cygpath 137 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` 138 | SEP="" 139 | for dir in $ROOTDIRSRAW ; do 140 | ROOTDIRS="$ROOTDIRS$SEP$dir" 141 | SEP="|" 142 | done 143 | OURCYGPATTERN="(^($ROOTDIRS))" 144 | # Add a user-defined pattern to the cygpath arguments 145 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then 146 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" 147 | fi 148 | # Now convert the arguments - kludge to limit ourselves to /bin/sh 149 | i=0 150 | for arg in "$@" ; do 151 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` 152 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option 153 | 154 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition 155 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` 156 | else 157 | eval `echo args$i`="\"$arg\"" 158 | fi 159 | i=`expr $i + 1` 160 | done 161 | case $i in 162 | 0) set -- ;; 163 | 1) set -- "$args0" ;; 164 | 2) set -- "$args0" "$args1" ;; 165 | 3) set -- "$args0" "$args1" "$args2" ;; 166 | 4) set -- "$args0" "$args1" "$args2" "$args3" ;; 167 | 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; 168 | 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; 169 | 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; 170 | 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; 171 | 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; 172 | esac 173 | fi 174 | 175 | # Escape application args 176 | save () { 177 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done 178 | echo " " 179 | } 180 | APP_ARGS=`save "$@"` 181 | 182 | # Collect all arguments for the java command, following the shell quoting and substitution rules 183 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" 184 | 185 | exec "$JAVACMD" "$@" 186 | -------------------------------------------------------------------------------- /src/index.ts: -------------------------------------------------------------------------------- 1 | import fastify from 'fastify' 2 | import { CronJob } from 'cron' 3 | import cors from '@fastify/cors' 4 | import fs from 'fs' 5 | import fetch from 'node-fetch' 6 | 7 | import { PollingState, pollingStateRegistry } from './polling.js' 8 | import { 9 | Activity, 10 | expiration, 11 | getList, 12 | isInZoomMeeting, 13 | isSleepingHours, 14 | pingDiff, 15 | prisma, 16 | requirePassword, 17 | sleepExpiration, 18 | sleepExpirationDelay 19 | } from './util.js' 20 | import { LastfmState } from './modules/lastfm.js' 21 | import env from './env.js' 22 | 23 | const server = fastify({ 24 | logger: { 25 | prettyPrint: 26 | process.env.NODE_ENV === 'production' 27 | ? false 28 | : { 29 | translateTime: 'sys:HH:MM:ss', 30 | ignore: 'pid,hostname' 31 | }, 32 | level: process.env.LOG_LEVEL ?? 'info' 33 | } 34 | }) 35 | server.register(cors) 36 | 37 | const runDecisionTree = async (pollingState: PollingState): Promise => { 38 | const manualActivity = await prisma.manualActivity.findFirst({ where: {} }) 39 | const [desktopDiff, mobileDiff] = await Promise.all([pingDiff('desktop'), pingDiff('mobile')]) 40 | const minDiff = Math.min(desktopDiff, mobileDiff) 41 | 42 | if (manualActivity) { 43 | // Requirements to disable sleep mode: 44 | // - Activity set to sleeping 45 | // - Last event within 1 hour 46 | // - Sleep button pressed more than 6 minutes before last event 47 | const sleepEnded = 48 | manualActivity && 49 | manualActivity.emoji === '💤' && 50 | minDiff < sleepExpiration && 51 | Date.now() - manualActivity.time.getTime() - minDiff > sleepExpirationDelay 52 | if (sleepEnded) { 53 | await prisma.manualActivity.deleteMany({ where: {} }) 54 | } else { 55 | return { emoji: manualActivity.emoji, label: manualActivity.label } 56 | } 57 | } 58 | 59 | // FSP reservation: 60 | if (pollingState.fsp.inReservation) return { emoji: '🛩️', label: 'flying a plane' } 61 | 62 | // Music apps: 63 | const apps = await getList('apps') 64 | if (['ableton', 'musescore', 'max'].some((app) => apps.includes(app))) return { emoji: '🎵', label: 'making music' } 65 | 66 | // Music calendar search: 67 | if (['cello', 'rehearsal', 'recital'].some((key) => pollingState.calendar.eventName?.toLowerCase()?.includes(key))) 68 | return { emoji: '🎵', label: 'making music' } 69 | 70 | // Toggl tracking: 71 | if (pollingState.toggl.tracking) return { emoji: '💼', label: 'working' } 72 | 73 | // Class calendar search: 74 | if (['lesson', 'class'].some((key) => pollingState.calendar.eventName?.toLowerCase()?.includes(key))) 75 | return { emoji: '📚', label: 'in class' } 76 | 77 | // Call websites: 78 | const domains = desktopDiff < expiration ? await getList('domains') : [] 79 | if (['meet.google.com', 'voice.google.com'].some((domain) => domains.includes(domain))) 80 | return { emoji: '📞', label: 'on a call' } 81 | 82 | // Call calendar regex: 83 | if (pollingState.calendar.eventName) { 84 | const callRegex = 85 | /^(?:[A-Z][a-z]+ )*[A-Z][a-z]+ (\+|<>|\/) (?:[A-Z][a-z]+ )*[A-Z][a-z]+(?: \1 (?:[A-Z][a-z]+ )*[A-Z][a-z]+)*$/ 86 | if (callRegex.test(pollingState.calendar.eventName.trim())) return { emoji: '📞', label: 'on a call' } 87 | } 88 | 89 | // Call calendar check: 90 | if (pollingState.calendar.isVideoMeeting) return { emoji: '📞', label: 'on a call' } 91 | 92 | // Zoom call: 93 | if (apps.includes('zoom') && (await isInZoomMeeting())) return { emoji: '📞', label: 'on a call' } 94 | 95 | // Programming apps: 96 | if (['vscode', 'terminal', 'android-studio'].some((app) => apps.includes(app))) 97 | return { emoji: '👩‍💻', label: 'programming' } 98 | 99 | // Programming websites: 100 | if (['github.com', 'replit.com', 'github.dev', 'vscode.dev'].some((domain) => domains.includes(domain))) 101 | return { emoji: '👩‍💻', label: 'programming' } 102 | 103 | // Figma: 104 | if (apps.includes('figma') || domains.includes('figma.com')) return { emoji: '🎨', label: 'doing visual design' } 105 | 106 | // Writing websites: 107 | if (['docs.google.com', 'app.grammarly.com'].some((domain) => domains.includes(domain))) 108 | return { emoji: '📝', label: 'writing something' } 109 | 110 | // Gaming: 111 | if (['minecraft', 'rimworld', 'celeste', 'factorio'].some((app) => apps.includes(app))) 112 | return { emoji: '🎮', label: 'gaming' } 113 | 114 | // Listening to music: 115 | if (pollingState.lastfm.nowPlaying) return { emoji: '🎧', label: 'listening to music' } 116 | 117 | // Meal calendar search: 118 | if (['dinner', 'lunch', 'breakfast'].some((key) => pollingState.calendar.eventName?.toLowerCase()?.includes(key))) 119 | return { emoji: '✨', label: 'doing something irl' } 120 | 121 | // Sleeping heuristic for when I forget to manually update: 122 | if (minDiff > sleepExpiration && isSleepingHours(new Date())) { 123 | const created = await prisma.manualActivity.create({ 124 | data: { emoji: '💤', label: 'sleeping', time: new Date() } 125 | }) 126 | return { emoji: created.emoji, label: created.label } 127 | } 128 | 129 | // Fallback: 130 | if (mobileDiff < expiration) return { emoji: '📱', label: 'doing stuff on my phone' } 131 | if (desktopDiff < expiration) return { emoji: '💻', label: 'doing stuff on my computer' } 132 | return { emoji: '✨', label: 'doing something irl' } 133 | } 134 | 135 | const updateDecisionTree = async (pollingState: PollingState): Promise => { 136 | const activity = await runDecisionTree(pollingState) 137 | const curActivity = await prisma.activity.findFirst({ orderBy: { time: 'desc' } }) 138 | if (activity.emoji !== curActivity?.emoji) { 139 | await prisma.activity.create({ data: activity }) 140 | await fetch('https://slack.com/api/users.profile.set', { 141 | method: 'POST', 142 | headers: { 143 | 'Content-Type': 'application/json', 144 | Authorization: `Bearer ${env.slackToken}` 145 | }, 146 | body: JSON.stringify({ 147 | profile: { 148 | status_text: activity.label[0].toUpperCase() + activity.label.slice(1), 149 | status_emoji: activity.emoji, 150 | status_expiration: 0 151 | } 152 | }) 153 | }) 154 | } 155 | return activity 156 | } 157 | 158 | const start = async () => { 159 | const pollingState: PollingState = {} as PollingState 160 | 161 | server.log.info('Fetching initial polling state...') 162 | await Promise.all( 163 | Object.keys(pollingStateRegistry) 164 | .map((key) => key as keyof typeof pollingStateRegistry) 165 | .map(async (key) => { 166 | // I hate it, I hate it, I hate it! 167 | pollingState[key] = (await pollingStateRegistry[key].fetcher()) as any 168 | new CronJob( 169 | pollingStateRegistry[key].cron, 170 | async () => { 171 | server.log.debug(`Polling '${key}' and running decision tree`) 172 | try { 173 | pollingState[key] = (await pollingStateRegistry[key].fetcher()) as any 174 | await updateDecisionTree(pollingState) 175 | server.log.debug(pollingState) 176 | } catch (err) { 177 | server.log.error((err as Error).message) 178 | } 179 | }, 180 | null, 181 | true, 182 | process.env.TZ ?? 'America/New_York' 183 | ) 184 | }) 185 | ) 186 | server.log.debug(pollingState) 187 | 188 | server.get<{ 189 | Reply: { activity: Activity; lastfm: LastfmState } 190 | }>('/', async () => { 191 | const activity = (await prisma.activity.findFirst({ orderBy: { time: 'desc' } }))! 192 | return { activity, lastfm: pollingState.lastfm } 193 | }) 194 | 195 | server.get('/dash', async (req, reply) => { 196 | reply.type('html') 197 | return fs.createReadStream('dash.html') 198 | }) 199 | 200 | server.get<{ 201 | Reply: { history: Activity[] } 202 | }>('/history', async () => { 203 | const history = await prisma.activity.findMany({ orderBy: { time: 'desc' } }) 204 | return { history } 205 | }) 206 | 207 | server.get('/check-auth', { preHandler: [requirePassword] }, async () => ({})) 208 | 209 | server.post<{ 210 | Body: { emoji: string; label: string } 211 | }>( 212 | '/manual', 213 | { 214 | preHandler: [requirePassword], 215 | schema: { 216 | body: { 217 | type: 'object', 218 | properties: { 219 | emoji: { type: 'string' }, 220 | label: { type: 'string' } 221 | }, 222 | required: ['emoji', 'label'] 223 | } 224 | } 225 | }, 226 | async (req) => { 227 | const { emoji, label } = req.body 228 | const current = await prisma.manualActivity.findFirst({ where: {} }) 229 | const data = { emoji, label, time: new Date() } 230 | 231 | if (current) { 232 | await prisma.manualActivity.update({ where: { id: current.id }, data }) 233 | } else { 234 | await prisma.manualActivity.create({ data }) 235 | } 236 | 237 | await updateDecisionTree(pollingState) 238 | return {} 239 | } 240 | ) 241 | 242 | server.post('/manual/clear', { preHandler: [requirePassword] }, async () => { 243 | await prisma.manualActivity.deleteMany({ where: {} }) 244 | await updateDecisionTree(pollingState) 245 | return {} 246 | }) 247 | 248 | server.post<{ 249 | Params: { key: string } 250 | }>('/ping/:key', { preHandler: [requirePassword] }, async (req, reply) => { 251 | if (['desktop', 'mobile'].includes(req.params.key)) { 252 | await prisma.ping.upsert({ 253 | where: { key: req.params.key }, 254 | update: { time: new Date() }, 255 | create: { key: req.params.key, time: new Date() } 256 | }) 257 | await updateDecisionTree(pollingState) 258 | return {} 259 | } else { 260 | reply.status(400) 261 | throw new Error('Invalid ping key') 262 | } 263 | }) 264 | 265 | server.post<{ 266 | Params: { key: string; sourceDevice: string } 267 | Body: { list: string[] } 268 | }>( 269 | '/list/:key/:sourceDevice', 270 | { 271 | preHandler: [requirePassword], 272 | schema: { 273 | body: { 274 | type: 'object', 275 | properties: { 276 | list: { type: 'array', items: { type: 'string' } } 277 | }, 278 | required: ['list'] 279 | } 280 | } 281 | }, 282 | async (req, reply) => { 283 | if (['apps', 'domains'].includes(req.params.key)) { 284 | await prisma.list.upsert({ 285 | where: { 286 | key_sourceDevice: { 287 | key: req.params.key, 288 | sourceDevice: req.params.sourceDevice 289 | } 290 | }, 291 | update: { list: req.body.list, time: new Date() }, 292 | create: { key: req.params.key, sourceDevice: req.params.sourceDevice, list: req.body.list, time: new Date() } 293 | }) 294 | await updateDecisionTree(pollingState) 295 | return {} 296 | } else { 297 | reply.status(400) 298 | throw new Error('Invalid list key') 299 | } 300 | } 301 | ) 302 | 303 | server.post<{ 304 | Body: { 305 | event: string 306 | event_ts: string 307 | payload: { 308 | account_id: string 309 | object: object 310 | } 311 | } 312 | }>( 313 | '/zoom', 314 | { 315 | schema: { 316 | body: { 317 | type: 'object', 318 | properties: { 319 | event: { type: 'string' }, 320 | event_ts: { type: 'string' }, 321 | payload: { 322 | type: 'object', 323 | properties: { 324 | account_id: { type: 'string' }, 325 | object: { type: 'object' } 326 | }, 327 | required: ['account_id', 'object'] 328 | } 329 | }, 330 | required: ['event', 'event_ts', 'payload'] 331 | } 332 | } 333 | }, 334 | async (req, reply) => { 335 | if (req.headers.authorization !== env.zoomVerificationToken) { 336 | reply.status(401) 337 | throw new Error('Invalid verification token') 338 | } 339 | if (req.body.event === 'user.presence_status_updated') { 340 | const { id, presence_status } = req.body.payload.object as { id: string; presence_status: string } 341 | req.log.info(`zoom presence: ${presence_status}`) 342 | const inMeeting = ['In_Meeting', 'Presenting', 'On_Phone_Call'].includes(presence_status) 343 | await prisma.zoomUser.upsert({ 344 | where: { id }, 345 | update: { inMeeting, lastUpdate: new Date() }, 346 | create: { id, inMeeting, lastUpdate: new Date() } 347 | }) 348 | await updateDecisionTree(pollingState) 349 | } 350 | return {} 351 | } 352 | ) 353 | 354 | await updateDecisionTree(pollingState) 355 | await server.listen(3000, '0.0.0.0') 356 | } 357 | 358 | start().catch(async (error) => { 359 | await prisma.$disconnect() 360 | server.log.error(error.message) 361 | process.exit(1) 362 | }) 363 | -------------------------------------------------------------------------------- /stalkerd/Cargo.lock: -------------------------------------------------------------------------------- 1 | # This file is automatically @generated by Cargo. 2 | # It is not intended for manual editing. 3 | version = 3 4 | 5 | [[package]] 6 | name = "autocfg" 7 | version = "1.1.0" 8 | source = "registry+https://github.com/rust-lang/crates.io-index" 9 | checksum = "d468802bab17cbc0cc575e9b053f41e72aa36bfa6b7f55e3529ffa43161b97fa" 10 | 11 | [[package]] 12 | name = "base64" 13 | version = "0.13.0" 14 | source = "registry+https://github.com/rust-lang/crates.io-index" 15 | checksum = "904dfeac50f3cdaba28fc6f57fdcddb75f49ed61346676a78c4ffe55877802fd" 16 | 17 | [[package]] 18 | name = "bitflags" 19 | version = "1.3.2" 20 | source = "registry+https://github.com/rust-lang/crates.io-index" 21 | checksum = "bef38d45163c2f1dde094a7dfd33ccf595c92905c8f8f4fdc18d06fb1037718a" 22 | 23 | [[package]] 24 | name = "boxfnonce" 25 | version = "0.1.1" 26 | source = "registry+https://github.com/rust-lang/crates.io-index" 27 | checksum = "5988cb1d626264ac94100be357308f29ff7cbdd3b36bda27f450a4ee3f713426" 28 | 29 | [[package]] 30 | name = "bumpalo" 31 | version = "3.9.1" 32 | source = "registry+https://github.com/rust-lang/crates.io-index" 33 | checksum = "a4a45a46ab1f2412e53d3a0ade76ffad2025804294569aae387231a0cd6e0899" 34 | 35 | [[package]] 36 | name = "bytes" 37 | version = "1.1.0" 38 | source = "registry+https://github.com/rust-lang/crates.io-index" 39 | checksum = "c4872d67bab6358e59559027aa3b9157c53d9358c51423c17554809a8858e0f8" 40 | 41 | [[package]] 42 | name = "cc" 43 | version = "1.0.73" 44 | source = "registry+https://github.com/rust-lang/crates.io-index" 45 | checksum = "2fff2a6927b3bb87f9595d67196a70493f627687a71d87a0d692242c33f58c11" 46 | 47 | [[package]] 48 | name = "cfg-if" 49 | version = "1.0.0" 50 | source = "registry+https://github.com/rust-lang/crates.io-index" 51 | checksum = "baf1de4339761588bc0619e3cbc0120ee582ebb74b53b4efbf79117bd2da40fd" 52 | 53 | [[package]] 54 | name = "core-foundation" 55 | version = "0.9.3" 56 | source = "registry+https://github.com/rust-lang/crates.io-index" 57 | checksum = "194a7a9e6de53fa55116934067c844d9d749312f75c6f6d0980e8c252f8c2146" 58 | dependencies = [ 59 | "core-foundation-sys", 60 | "libc", 61 | ] 62 | 63 | [[package]] 64 | name = "core-foundation-sys" 65 | version = "0.8.3" 66 | source = "registry+https://github.com/rust-lang/crates.io-index" 67 | checksum = "5827cebf4670468b8772dd191856768aedcb1b0278a04f989f7766351917b9dc" 68 | 69 | [[package]] 70 | name = "crossbeam-channel" 71 | version = "0.5.4" 72 | source = "registry+https://github.com/rust-lang/crates.io-index" 73 | checksum = "5aaa7bd5fb665c6864b5f963dd9097905c54125909c7aa94c9e18507cdbe6c53" 74 | dependencies = [ 75 | "cfg-if", 76 | "crossbeam-utils", 77 | ] 78 | 79 | [[package]] 80 | name = "crossbeam-deque" 81 | version = "0.8.1" 82 | source = "registry+https://github.com/rust-lang/crates.io-index" 83 | checksum = "6455c0ca19f0d2fbf751b908d5c55c1f5cbc65e03c4225427254b46890bdde1e" 84 | dependencies = [ 85 | "cfg-if", 86 | "crossbeam-epoch", 87 | "crossbeam-utils", 88 | ] 89 | 90 | [[package]] 91 | name = "crossbeam-epoch" 92 | version = "0.9.8" 93 | source = "registry+https://github.com/rust-lang/crates.io-index" 94 | checksum = "1145cf131a2c6ba0615079ab6a638f7e1973ac9c2634fcbeaaad6114246efe8c" 95 | dependencies = [ 96 | "autocfg", 97 | "cfg-if", 98 | "crossbeam-utils", 99 | "lazy_static", 100 | "memoffset", 101 | "scopeguard", 102 | ] 103 | 104 | [[package]] 105 | name = "crossbeam-utils" 106 | version = "0.8.8" 107 | source = "registry+https://github.com/rust-lang/crates.io-index" 108 | checksum = "0bf124c720b7686e3c2663cf54062ab0f68a88af2fb6a030e87e30bf721fcb38" 109 | dependencies = [ 110 | "cfg-if", 111 | "lazy_static", 112 | ] 113 | 114 | [[package]] 115 | name = "daemonize" 116 | version = "0.4.1" 117 | source = "registry+https://github.com/rust-lang/crates.io-index" 118 | checksum = "70c24513e34f53b640819f0ac9f705b673fcf4006d7aab8778bee72ebfc89815" 119 | dependencies = [ 120 | "boxfnonce", 121 | "libc", 122 | ] 123 | 124 | [[package]] 125 | name = "either" 126 | version = "1.6.1" 127 | source = "registry+https://github.com/rust-lang/crates.io-index" 128 | checksum = "e78d4f1cc4ae33bbfc157ed5d5a5ef3bc29227303d595861deb238fcec4e9457" 129 | 130 | [[package]] 131 | name = "encoding_rs" 132 | version = "0.8.31" 133 | source = "registry+https://github.com/rust-lang/crates.io-index" 134 | checksum = "9852635589dc9f9ea1b6fe9f05b50ef208c85c834a562f0c6abb1c475736ec2b" 135 | dependencies = [ 136 | "cfg-if", 137 | ] 138 | 139 | [[package]] 140 | name = "fastrand" 141 | version = "1.7.0" 142 | source = "registry+https://github.com/rust-lang/crates.io-index" 143 | checksum = "c3fcf0cee53519c866c09b5de1f6c56ff9d647101f81c1964fa632e148896cdf" 144 | dependencies = [ 145 | "instant", 146 | ] 147 | 148 | [[package]] 149 | name = "fnv" 150 | version = "1.0.7" 151 | source = "registry+https://github.com/rust-lang/crates.io-index" 152 | checksum = "3f9eec918d3f24069decb9af1554cad7c880e2da24a9afd88aca000531ab82c1" 153 | 154 | [[package]] 155 | name = "foreign-types" 156 | version = "0.3.2" 157 | source = "registry+https://github.com/rust-lang/crates.io-index" 158 | checksum = "f6f339eb8adc052cd2ca78910fda869aefa38d22d5cb648e6485e4d3fc06f3b1" 159 | dependencies = [ 160 | "foreign-types-shared", 161 | ] 162 | 163 | [[package]] 164 | name = "foreign-types-shared" 165 | version = "0.1.1" 166 | source = "registry+https://github.com/rust-lang/crates.io-index" 167 | checksum = "00b0228411908ca8685dba7fc2cdd70ec9990a6e753e89b6ac91a84c40fbaf4b" 168 | 169 | [[package]] 170 | name = "form_urlencoded" 171 | version = "1.0.1" 172 | source = "registry+https://github.com/rust-lang/crates.io-index" 173 | checksum = "5fc25a87fa4fd2094bffb06925852034d90a17f0d1e05197d4956d3555752191" 174 | dependencies = [ 175 | "matches", 176 | "percent-encoding", 177 | ] 178 | 179 | [[package]] 180 | name = "futures-channel" 181 | version = "0.3.21" 182 | source = "registry+https://github.com/rust-lang/crates.io-index" 183 | checksum = "c3083ce4b914124575708913bca19bfe887522d6e2e6d0952943f5eac4a74010" 184 | dependencies = [ 185 | "futures-core", 186 | ] 187 | 188 | [[package]] 189 | name = "futures-core" 190 | version = "0.3.21" 191 | source = "registry+https://github.com/rust-lang/crates.io-index" 192 | checksum = "0c09fd04b7e4073ac7156a9539b57a484a8ea920f79c7c675d05d289ab6110d3" 193 | 194 | [[package]] 195 | name = "futures-io" 196 | version = "0.3.21" 197 | source = "registry+https://github.com/rust-lang/crates.io-index" 198 | checksum = "fc4045962a5a5e935ee2fdedaa4e08284547402885ab326734432bed5d12966b" 199 | 200 | [[package]] 201 | name = "futures-sink" 202 | version = "0.3.21" 203 | source = "registry+https://github.com/rust-lang/crates.io-index" 204 | checksum = "21163e139fa306126e6eedaf49ecdb4588f939600f0b1e770f4205ee4b7fa868" 205 | 206 | [[package]] 207 | name = "futures-task" 208 | version = "0.3.21" 209 | source = "registry+https://github.com/rust-lang/crates.io-index" 210 | checksum = "57c66a976bf5909d801bbef33416c41372779507e7a6b3a5e25e4749c58f776a" 211 | 212 | [[package]] 213 | name = "futures-util" 214 | version = "0.3.21" 215 | source = "registry+https://github.com/rust-lang/crates.io-index" 216 | checksum = "d8b7abd5d659d9b90c8cba917f6ec750a74e2dc23902ef9cd4cc8c8b22e6036a" 217 | dependencies = [ 218 | "futures-core", 219 | "futures-io", 220 | "futures-task", 221 | "memchr", 222 | "pin-project-lite", 223 | "pin-utils", 224 | "slab", 225 | ] 226 | 227 | [[package]] 228 | name = "h2" 229 | version = "0.3.13" 230 | source = "registry+https://github.com/rust-lang/crates.io-index" 231 | checksum = "37a82c6d637fc9515a4694bbf1cb2457b79d81ce52b3108bdeea58b07dd34a57" 232 | dependencies = [ 233 | "bytes", 234 | "fnv", 235 | "futures-core", 236 | "futures-sink", 237 | "futures-util", 238 | "http", 239 | "indexmap", 240 | "slab", 241 | "tokio", 242 | "tokio-util", 243 | "tracing", 244 | ] 245 | 246 | [[package]] 247 | name = "hashbrown" 248 | version = "0.11.2" 249 | source = "registry+https://github.com/rust-lang/crates.io-index" 250 | checksum = "ab5ef0d4909ef3724cc8cce6ccc8572c5c817592e9285f5464f8e86f8bd3726e" 251 | 252 | [[package]] 253 | name = "hermit-abi" 254 | version = "0.1.19" 255 | source = "registry+https://github.com/rust-lang/crates.io-index" 256 | checksum = "62b467343b94ba476dcb2500d242dadbb39557df889310ac77c5d99100aaac33" 257 | dependencies = [ 258 | "libc", 259 | ] 260 | 261 | [[package]] 262 | name = "home" 263 | version = "0.5.3" 264 | source = "registry+https://github.com/rust-lang/crates.io-index" 265 | checksum = "2456aef2e6b6a9784192ae780c0f15bc57df0e918585282325e8c8ac27737654" 266 | dependencies = [ 267 | "winapi", 268 | ] 269 | 270 | [[package]] 271 | name = "http" 272 | version = "0.2.6" 273 | source = "registry+https://github.com/rust-lang/crates.io-index" 274 | checksum = "31f4c6746584866f0feabcc69893c5b51beef3831656a968ed7ae254cdc4fd03" 275 | dependencies = [ 276 | "bytes", 277 | "fnv", 278 | "itoa", 279 | ] 280 | 281 | [[package]] 282 | name = "http-body" 283 | version = "0.4.4" 284 | source = "registry+https://github.com/rust-lang/crates.io-index" 285 | checksum = "1ff4f84919677303da5f147645dbea6b1881f368d03ac84e1dc09031ebd7b2c6" 286 | dependencies = [ 287 | "bytes", 288 | "http", 289 | "pin-project-lite", 290 | ] 291 | 292 | [[package]] 293 | name = "httparse" 294 | version = "1.7.0" 295 | source = "registry+https://github.com/rust-lang/crates.io-index" 296 | checksum = "6330e8a36bd8c859f3fa6d9382911fbb7147ec39807f63b923933a247240b9ba" 297 | 298 | [[package]] 299 | name = "httpdate" 300 | version = "1.0.2" 301 | source = "registry+https://github.com/rust-lang/crates.io-index" 302 | checksum = "c4a1e36c821dbe04574f602848a19f742f4fb3c98d40449f11bcad18d6b17421" 303 | 304 | [[package]] 305 | name = "hyper" 306 | version = "0.14.18" 307 | source = "registry+https://github.com/rust-lang/crates.io-index" 308 | checksum = "b26ae0a80afebe130861d90abf98e3814a4f28a4c6ffeb5ab8ebb2be311e0ef2" 309 | dependencies = [ 310 | "bytes", 311 | "futures-channel", 312 | "futures-core", 313 | "futures-util", 314 | "h2", 315 | "http", 316 | "http-body", 317 | "httparse", 318 | "httpdate", 319 | "itoa", 320 | "pin-project-lite", 321 | "socket2", 322 | "tokio", 323 | "tower-service", 324 | "tracing", 325 | "want", 326 | ] 327 | 328 | [[package]] 329 | name = "hyper-tls" 330 | version = "0.5.0" 331 | source = "registry+https://github.com/rust-lang/crates.io-index" 332 | checksum = "d6183ddfa99b85da61a140bea0efc93fdf56ceaa041b37d553518030827f9905" 333 | dependencies = [ 334 | "bytes", 335 | "hyper", 336 | "native-tls", 337 | "tokio", 338 | "tokio-native-tls", 339 | ] 340 | 341 | [[package]] 342 | name = "idna" 343 | version = "0.2.3" 344 | source = "registry+https://github.com/rust-lang/crates.io-index" 345 | checksum = "418a0a6fab821475f634efe3ccc45c013f742efe03d853e8d3355d5cb850ecf8" 346 | dependencies = [ 347 | "matches", 348 | "unicode-bidi", 349 | "unicode-normalization", 350 | ] 351 | 352 | [[package]] 353 | name = "indexmap" 354 | version = "1.8.1" 355 | source = "registry+https://github.com/rust-lang/crates.io-index" 356 | checksum = "0f647032dfaa1f8b6dc29bd3edb7bbef4861b8b8007ebb118d6db284fd59f6ee" 357 | dependencies = [ 358 | "autocfg", 359 | "hashbrown", 360 | ] 361 | 362 | [[package]] 363 | name = "instant" 364 | version = "0.1.12" 365 | source = "registry+https://github.com/rust-lang/crates.io-index" 366 | checksum = "7a5bbe824c507c5da5956355e86a746d82e0e1464f65d862cc5e71da70e94b2c" 367 | dependencies = [ 368 | "cfg-if", 369 | ] 370 | 371 | [[package]] 372 | name = "ipnet" 373 | version = "2.5.0" 374 | source = "registry+https://github.com/rust-lang/crates.io-index" 375 | checksum = "879d54834c8c76457ef4293a689b2a8c59b076067ad77b15efafbb05f92a592b" 376 | 377 | [[package]] 378 | name = "itoa" 379 | version = "1.0.1" 380 | source = "registry+https://github.com/rust-lang/crates.io-index" 381 | checksum = "1aab8fc367588b89dcee83ab0fd66b72b50b72fa1904d7095045ace2b0c81c35" 382 | 383 | [[package]] 384 | name = "js-sys" 385 | version = "0.3.57" 386 | source = "registry+https://github.com/rust-lang/crates.io-index" 387 | checksum = "671a26f820db17c2a2750743f1dd03bafd15b98c9f30c7c2628c024c05d73397" 388 | dependencies = [ 389 | "wasm-bindgen", 390 | ] 391 | 392 | [[package]] 393 | name = "lazy_static" 394 | version = "1.4.0" 395 | source = "registry+https://github.com/rust-lang/crates.io-index" 396 | checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646" 397 | 398 | [[package]] 399 | name = "libc" 400 | version = "0.2.124" 401 | source = "registry+https://github.com/rust-lang/crates.io-index" 402 | checksum = "21a41fed9d98f27ab1c6d161da622a4fa35e8a54a8adc24bbf3ddd0ef70b0e50" 403 | 404 | [[package]] 405 | name = "log" 406 | version = "0.4.16" 407 | source = "registry+https://github.com/rust-lang/crates.io-index" 408 | checksum = "6389c490849ff5bc16be905ae24bc913a9c8892e19b2341dbc175e14c341c2b8" 409 | dependencies = [ 410 | "cfg-if", 411 | ] 412 | 413 | [[package]] 414 | name = "matches" 415 | version = "0.1.9" 416 | source = "registry+https://github.com/rust-lang/crates.io-index" 417 | checksum = "a3e378b66a060d48947b590737b30a1be76706c8dd7b8ba0f2fe3989c68a853f" 418 | 419 | [[package]] 420 | name = "memchr" 421 | version = "2.4.1" 422 | source = "registry+https://github.com/rust-lang/crates.io-index" 423 | checksum = "308cc39be01b73d0d18f82a0e7b2a3df85245f84af96fdddc5d202d27e47b86a" 424 | 425 | [[package]] 426 | name = "memoffset" 427 | version = "0.6.5" 428 | source = "registry+https://github.com/rust-lang/crates.io-index" 429 | checksum = "5aa361d4faea93603064a027415f07bd8e1d5c88c9fbf68bf56a285428fd79ce" 430 | dependencies = [ 431 | "autocfg", 432 | ] 433 | 434 | [[package]] 435 | name = "mime" 436 | version = "0.3.16" 437 | source = "registry+https://github.com/rust-lang/crates.io-index" 438 | checksum = "2a60c7ce501c71e03a9c9c0d35b861413ae925bd979cc7a4e30d060069aaac8d" 439 | 440 | [[package]] 441 | name = "mio" 442 | version = "0.8.2" 443 | source = "registry+https://github.com/rust-lang/crates.io-index" 444 | checksum = "52da4364ffb0e4fe33a9841a98a3f3014fb964045ce4f7a45a398243c8d6b0c9" 445 | dependencies = [ 446 | "libc", 447 | "log", 448 | "miow", 449 | "ntapi", 450 | "wasi", 451 | "winapi", 452 | ] 453 | 454 | [[package]] 455 | name = "miow" 456 | version = "0.3.7" 457 | source = "registry+https://github.com/rust-lang/crates.io-index" 458 | checksum = "b9f1c5b025cda876f66ef43a113f91ebc9f4ccef34843000e0adf6ebbab84e21" 459 | dependencies = [ 460 | "winapi", 461 | ] 462 | 463 | [[package]] 464 | name = "native-tls" 465 | version = "0.2.10" 466 | source = "registry+https://github.com/rust-lang/crates.io-index" 467 | checksum = "fd7e2f3618557f980e0b17e8856252eee3c97fa12c54dff0ca290fb6266ca4a9" 468 | dependencies = [ 469 | "lazy_static", 470 | "libc", 471 | "log", 472 | "openssl", 473 | "openssl-probe", 474 | "openssl-sys", 475 | "schannel", 476 | "security-framework", 477 | "security-framework-sys", 478 | "tempfile", 479 | ] 480 | 481 | [[package]] 482 | name = "ntapi" 483 | version = "0.3.7" 484 | source = "registry+https://github.com/rust-lang/crates.io-index" 485 | checksum = "c28774a7fd2fbb4f0babd8237ce554b73af68021b5f695a3cebd6c59bac0980f" 486 | dependencies = [ 487 | "winapi", 488 | ] 489 | 490 | [[package]] 491 | name = "num_cpus" 492 | version = "1.13.1" 493 | source = "registry+https://github.com/rust-lang/crates.io-index" 494 | checksum = "19e64526ebdee182341572e50e9ad03965aa510cd94427a4549448f285e957a1" 495 | dependencies = [ 496 | "hermit-abi", 497 | "libc", 498 | ] 499 | 500 | [[package]] 501 | name = "once_cell" 502 | version = "1.10.0" 503 | source = "registry+https://github.com/rust-lang/crates.io-index" 504 | checksum = "87f3e037eac156d1775da914196f0f37741a274155e34a0b7e427c35d2a2ecb9" 505 | 506 | [[package]] 507 | name = "openssl" 508 | version = "0.10.38" 509 | source = "registry+https://github.com/rust-lang/crates.io-index" 510 | checksum = "0c7ae222234c30df141154f159066c5093ff73b63204dcda7121eb082fc56a95" 511 | dependencies = [ 512 | "bitflags", 513 | "cfg-if", 514 | "foreign-types", 515 | "libc", 516 | "once_cell", 517 | "openssl-sys", 518 | ] 519 | 520 | [[package]] 521 | name = "openssl-probe" 522 | version = "0.1.5" 523 | source = "registry+https://github.com/rust-lang/crates.io-index" 524 | checksum = "ff011a302c396a5197692431fc1948019154afc178baf7d8e37367442a4601cf" 525 | 526 | [[package]] 527 | name = "openssl-sys" 528 | version = "0.9.72" 529 | source = "registry+https://github.com/rust-lang/crates.io-index" 530 | checksum = "7e46109c383602735fa0a2e48dd2b7c892b048e1bf69e5c3b1d804b7d9c203cb" 531 | dependencies = [ 532 | "autocfg", 533 | "cc", 534 | "libc", 535 | "pkg-config", 536 | "vcpkg", 537 | ] 538 | 539 | [[package]] 540 | name = "percent-encoding" 541 | version = "2.1.0" 542 | source = "registry+https://github.com/rust-lang/crates.io-index" 543 | checksum = "d4fd5641d01c8f18a23da7b6fe29298ff4b55afcccdf78973b24cf3175fee32e" 544 | 545 | [[package]] 546 | name = "pin-project-lite" 547 | version = "0.2.9" 548 | source = "registry+https://github.com/rust-lang/crates.io-index" 549 | checksum = "e0a7ae3ac2f1173085d398531c705756c94a4c56843785df85a60c1a0afac116" 550 | 551 | [[package]] 552 | name = "pin-utils" 553 | version = "0.1.0" 554 | source = "registry+https://github.com/rust-lang/crates.io-index" 555 | checksum = "8b870d8c151b6f2fb93e84a13146138f05d02ed11c7e7c54f8826aaaf7c9f184" 556 | 557 | [[package]] 558 | name = "pkg-config" 559 | version = "0.3.25" 560 | source = "registry+https://github.com/rust-lang/crates.io-index" 561 | checksum = "1df8c4ec4b0627e53bdf214615ad287367e482558cf84b109250b37464dc03ae" 562 | 563 | [[package]] 564 | name = "proc-macro2" 565 | version = "1.0.37" 566 | source = "registry+https://github.com/rust-lang/crates.io-index" 567 | checksum = "ec757218438d5fda206afc041538b2f6d889286160d649a86a24d37e1235afd1" 568 | dependencies = [ 569 | "unicode-xid", 570 | ] 571 | 572 | [[package]] 573 | name = "quote" 574 | version = "1.0.18" 575 | source = "registry+https://github.com/rust-lang/crates.io-index" 576 | checksum = "a1feb54ed693b93a84e14094943b84b7c4eae204c512b7ccb95ab0c66d278ad1" 577 | dependencies = [ 578 | "proc-macro2", 579 | ] 580 | 581 | [[package]] 582 | name = "rayon" 583 | version = "1.5.2" 584 | source = "registry+https://github.com/rust-lang/crates.io-index" 585 | checksum = "fd249e82c21598a9a426a4e00dd7adc1d640b22445ec8545feef801d1a74c221" 586 | dependencies = [ 587 | "autocfg", 588 | "crossbeam-deque", 589 | "either", 590 | "rayon-core", 591 | ] 592 | 593 | [[package]] 594 | name = "rayon-core" 595 | version = "1.9.2" 596 | source = "registry+https://github.com/rust-lang/crates.io-index" 597 | checksum = "9f51245e1e62e1f1629cbfec37b5793bbabcaeb90f30e94d2ba03564687353e4" 598 | dependencies = [ 599 | "crossbeam-channel", 600 | "crossbeam-deque", 601 | "crossbeam-utils", 602 | "num_cpus", 603 | ] 604 | 605 | [[package]] 606 | name = "redox_syscall" 607 | version = "0.2.13" 608 | source = "registry+https://github.com/rust-lang/crates.io-index" 609 | checksum = "62f25bc4c7e55e0b0b7a1d43fb893f4fa1361d0abe38b9ce4f323c2adfe6ef42" 610 | dependencies = [ 611 | "bitflags", 612 | ] 613 | 614 | [[package]] 615 | name = "remove_dir_all" 616 | version = "0.5.3" 617 | source = "registry+https://github.com/rust-lang/crates.io-index" 618 | checksum = "3acd125665422973a33ac9d3dd2df85edad0f4ae9b00dafb1a05e43a9f5ef8e7" 619 | dependencies = [ 620 | "winapi", 621 | ] 622 | 623 | [[package]] 624 | name = "reqwest" 625 | version = "0.11.10" 626 | source = "registry+https://github.com/rust-lang/crates.io-index" 627 | checksum = "46a1f7aa4f35e5e8b4160449f51afc758f0ce6454315a9fa7d0d113e958c41eb" 628 | dependencies = [ 629 | "base64", 630 | "bytes", 631 | "encoding_rs", 632 | "futures-core", 633 | "futures-util", 634 | "h2", 635 | "http", 636 | "http-body", 637 | "hyper", 638 | "hyper-tls", 639 | "ipnet", 640 | "js-sys", 641 | "lazy_static", 642 | "log", 643 | "mime", 644 | "native-tls", 645 | "percent-encoding", 646 | "pin-project-lite", 647 | "serde", 648 | "serde_json", 649 | "serde_urlencoded", 650 | "tokio", 651 | "tokio-native-tls", 652 | "url", 653 | "wasm-bindgen", 654 | "wasm-bindgen-futures", 655 | "web-sys", 656 | "winreg", 657 | ] 658 | 659 | [[package]] 660 | name = "ryu" 661 | version = "1.0.9" 662 | source = "registry+https://github.com/rust-lang/crates.io-index" 663 | checksum = "73b4b750c782965c211b42f022f59af1fbceabdd026623714f104152f1ec149f" 664 | 665 | [[package]] 666 | name = "schannel" 667 | version = "0.1.19" 668 | source = "registry+https://github.com/rust-lang/crates.io-index" 669 | checksum = "8f05ba609c234e60bee0d547fe94a4c7e9da733d1c962cf6e59efa4cd9c8bc75" 670 | dependencies = [ 671 | "lazy_static", 672 | "winapi", 673 | ] 674 | 675 | [[package]] 676 | name = "scopeguard" 677 | version = "1.1.0" 678 | source = "registry+https://github.com/rust-lang/crates.io-index" 679 | checksum = "d29ab0c6d3fc0ee92fe66e2d99f700eab17a8d57d1c1d3b748380fb20baa78cd" 680 | 681 | [[package]] 682 | name = "security-framework" 683 | version = "2.6.1" 684 | source = "registry+https://github.com/rust-lang/crates.io-index" 685 | checksum = "2dc14f172faf8a0194a3aded622712b0de276821addc574fa54fc0a1167e10dc" 686 | dependencies = [ 687 | "bitflags", 688 | "core-foundation", 689 | "core-foundation-sys", 690 | "libc", 691 | "security-framework-sys", 692 | ] 693 | 694 | [[package]] 695 | name = "security-framework-sys" 696 | version = "2.6.1" 697 | source = "registry+https://github.com/rust-lang/crates.io-index" 698 | checksum = "0160a13a177a45bfb43ce71c01580998474f556ad854dcbca936dd2841a5c556" 699 | dependencies = [ 700 | "core-foundation-sys", 701 | "libc", 702 | ] 703 | 704 | [[package]] 705 | name = "serde" 706 | version = "1.0.136" 707 | source = "registry+https://github.com/rust-lang/crates.io-index" 708 | checksum = "ce31e24b01e1e524df96f1c2fdd054405f8d7376249a5110886fb4b658484789" 709 | dependencies = [ 710 | "serde_derive", 711 | ] 712 | 713 | [[package]] 714 | name = "serde_derive" 715 | version = "1.0.136" 716 | source = "registry+https://github.com/rust-lang/crates.io-index" 717 | checksum = "08597e7152fcd306f41838ed3e37be9eaeed2b61c42e2117266a554fab4662f9" 718 | dependencies = [ 719 | "proc-macro2", 720 | "quote", 721 | "syn", 722 | ] 723 | 724 | [[package]] 725 | name = "serde_json" 726 | version = "1.0.79" 727 | source = "registry+https://github.com/rust-lang/crates.io-index" 728 | checksum = "8e8d9fa5c3b304765ce1fd9c4c8a3de2c8db365a5b91be52f186efc675681d95" 729 | dependencies = [ 730 | "itoa", 731 | "ryu", 732 | "serde", 733 | ] 734 | 735 | [[package]] 736 | name = "serde_urlencoded" 737 | version = "0.7.1" 738 | source = "registry+https://github.com/rust-lang/crates.io-index" 739 | checksum = "d3491c14715ca2294c4d6a88f15e84739788c1d030eed8c110436aafdaa2f3fd" 740 | dependencies = [ 741 | "form_urlencoded", 742 | "itoa", 743 | "ryu", 744 | "serde", 745 | ] 746 | 747 | [[package]] 748 | name = "slab" 749 | version = "0.4.6" 750 | source = "registry+https://github.com/rust-lang/crates.io-index" 751 | checksum = "eb703cfe953bccee95685111adeedb76fabe4e97549a58d16f03ea7b9367bb32" 752 | 753 | [[package]] 754 | name = "socket2" 755 | version = "0.4.4" 756 | source = "registry+https://github.com/rust-lang/crates.io-index" 757 | checksum = "66d72b759436ae32898a2af0a14218dbf55efde3feeb170eb623637db85ee1e0" 758 | dependencies = [ 759 | "libc", 760 | "winapi", 761 | ] 762 | 763 | [[package]] 764 | name = "stalkerd" 765 | version = "0.1.0" 766 | dependencies = [ 767 | "daemonize", 768 | "home", 769 | "reqwest", 770 | "serde", 771 | "sysinfo", 772 | "urlencoding", 773 | ] 774 | 775 | [[package]] 776 | name = "syn" 777 | version = "1.0.91" 778 | source = "registry+https://github.com/rust-lang/crates.io-index" 779 | checksum = "b683b2b825c8eef438b77c36a06dc262294da3d5a5813fac20da149241dcd44d" 780 | dependencies = [ 781 | "proc-macro2", 782 | "quote", 783 | "unicode-xid", 784 | ] 785 | 786 | [[package]] 787 | name = "sysinfo" 788 | version = "0.23.10" 789 | source = "registry+https://github.com/rust-lang/crates.io-index" 790 | checksum = "4eea2ed6847da2e0c7289f72cb4f285f0bd704694ca067d32be811b2a45ea858" 791 | dependencies = [ 792 | "cfg-if", 793 | "core-foundation-sys", 794 | "libc", 795 | "ntapi", 796 | "once_cell", 797 | "rayon", 798 | "winapi", 799 | ] 800 | 801 | [[package]] 802 | name = "tempfile" 803 | version = "3.3.0" 804 | source = "registry+https://github.com/rust-lang/crates.io-index" 805 | checksum = "5cdb1ef4eaeeaddc8fbd371e5017057064af0911902ef36b39801f67cc6d79e4" 806 | dependencies = [ 807 | "cfg-if", 808 | "fastrand", 809 | "libc", 810 | "redox_syscall", 811 | "remove_dir_all", 812 | "winapi", 813 | ] 814 | 815 | [[package]] 816 | name = "tinyvec" 817 | version = "1.6.0" 818 | source = "registry+https://github.com/rust-lang/crates.io-index" 819 | checksum = "87cc5ceb3875bb20c2890005a4e226a4651264a5c75edb2421b52861a0a0cb50" 820 | dependencies = [ 821 | "tinyvec_macros", 822 | ] 823 | 824 | [[package]] 825 | name = "tinyvec_macros" 826 | version = "0.1.0" 827 | source = "registry+https://github.com/rust-lang/crates.io-index" 828 | checksum = "cda74da7e1a664f795bb1f8a87ec406fb89a02522cf6e50620d016add6dbbf5c" 829 | 830 | [[package]] 831 | name = "tokio" 832 | version = "1.17.0" 833 | source = "registry+https://github.com/rust-lang/crates.io-index" 834 | checksum = "2af73ac49756f3f7c01172e34a23e5d0216f6c32333757c2c61feb2bbff5a5ee" 835 | dependencies = [ 836 | "bytes", 837 | "libc", 838 | "memchr", 839 | "mio", 840 | "num_cpus", 841 | "pin-project-lite", 842 | "socket2", 843 | "winapi", 844 | ] 845 | 846 | [[package]] 847 | name = "tokio-native-tls" 848 | version = "0.3.0" 849 | source = "registry+https://github.com/rust-lang/crates.io-index" 850 | checksum = "f7d995660bd2b7f8c1568414c1126076c13fbb725c40112dc0120b78eb9b717b" 851 | dependencies = [ 852 | "native-tls", 853 | "tokio", 854 | ] 855 | 856 | [[package]] 857 | name = "tokio-util" 858 | version = "0.7.1" 859 | source = "registry+https://github.com/rust-lang/crates.io-index" 860 | checksum = "0edfdeb067411dba2044da6d1cb2df793dd35add7888d73c16e3381ded401764" 861 | dependencies = [ 862 | "bytes", 863 | "futures-core", 864 | "futures-sink", 865 | "pin-project-lite", 866 | "tokio", 867 | "tracing", 868 | ] 869 | 870 | [[package]] 871 | name = "tower-service" 872 | version = "0.3.1" 873 | source = "registry+https://github.com/rust-lang/crates.io-index" 874 | checksum = "360dfd1d6d30e05fda32ace2c8c70e9c0a9da713275777f5a4dbb8a1893930c6" 875 | 876 | [[package]] 877 | name = "tracing" 878 | version = "0.1.34" 879 | source = "registry+https://github.com/rust-lang/crates.io-index" 880 | checksum = "5d0ecdcb44a79f0fe9844f0c4f33a342cbcbb5117de8001e6ba0dc2351327d09" 881 | dependencies = [ 882 | "cfg-if", 883 | "pin-project-lite", 884 | "tracing-attributes", 885 | "tracing-core", 886 | ] 887 | 888 | [[package]] 889 | name = "tracing-attributes" 890 | version = "0.1.20" 891 | source = "registry+https://github.com/rust-lang/crates.io-index" 892 | checksum = "2e65ce065b4b5c53e73bb28912318cb8c9e9ad3921f1d669eb0e68b4c8143a2b" 893 | dependencies = [ 894 | "proc-macro2", 895 | "quote", 896 | "syn", 897 | ] 898 | 899 | [[package]] 900 | name = "tracing-core" 901 | version = "0.1.26" 902 | source = "registry+https://github.com/rust-lang/crates.io-index" 903 | checksum = "f54c8ca710e81886d498c2fd3331b56c93aa248d49de2222ad2742247c60072f" 904 | dependencies = [ 905 | "lazy_static", 906 | ] 907 | 908 | [[package]] 909 | name = "try-lock" 910 | version = "0.2.3" 911 | source = "registry+https://github.com/rust-lang/crates.io-index" 912 | checksum = "59547bce71d9c38b83d9c0e92b6066c4253371f15005def0c30d9657f50c7642" 913 | 914 | [[package]] 915 | name = "unicode-bidi" 916 | version = "0.3.8" 917 | source = "registry+https://github.com/rust-lang/crates.io-index" 918 | checksum = "099b7128301d285f79ddd55b9a83d5e6b9e97c92e0ea0daebee7263e932de992" 919 | 920 | [[package]] 921 | name = "unicode-normalization" 922 | version = "0.1.19" 923 | source = "registry+https://github.com/rust-lang/crates.io-index" 924 | checksum = "d54590932941a9e9266f0832deed84ebe1bf2e4c9e4a3554d393d18f5e854bf9" 925 | dependencies = [ 926 | "tinyvec", 927 | ] 928 | 929 | [[package]] 930 | name = "unicode-xid" 931 | version = "0.2.2" 932 | source = "registry+https://github.com/rust-lang/crates.io-index" 933 | checksum = "8ccb82d61f80a663efe1f787a51b16b5a51e3314d6ac365b08639f52387b33f3" 934 | 935 | [[package]] 936 | name = "url" 937 | version = "2.2.2" 938 | source = "registry+https://github.com/rust-lang/crates.io-index" 939 | checksum = "a507c383b2d33b5fc35d1861e77e6b383d158b2da5e14fe51b83dfedf6fd578c" 940 | dependencies = [ 941 | "form_urlencoded", 942 | "idna", 943 | "matches", 944 | "percent-encoding", 945 | ] 946 | 947 | [[package]] 948 | name = "urlencoding" 949 | version = "2.1.0" 950 | source = "registry+https://github.com/rust-lang/crates.io-index" 951 | checksum = "68b90931029ab9b034b300b797048cf23723400aa757e8a2bfb9d748102f9821" 952 | 953 | [[package]] 954 | name = "vcpkg" 955 | version = "0.2.15" 956 | source = "registry+https://github.com/rust-lang/crates.io-index" 957 | checksum = "accd4ea62f7bb7a82fe23066fb0957d48ef677f6eeb8215f372f52e48bb32426" 958 | 959 | [[package]] 960 | name = "want" 961 | version = "0.3.0" 962 | source = "registry+https://github.com/rust-lang/crates.io-index" 963 | checksum = "1ce8a968cb1cd110d136ff8b819a556d6fb6d919363c61534f6860c7eb172ba0" 964 | dependencies = [ 965 | "log", 966 | "try-lock", 967 | ] 968 | 969 | [[package]] 970 | name = "wasi" 971 | version = "0.11.0+wasi-snapshot-preview1" 972 | source = "registry+https://github.com/rust-lang/crates.io-index" 973 | checksum = "9c8d87e72b64a3b4db28d11ce29237c246188f4f51057d65a7eab63b7987e423" 974 | 975 | [[package]] 976 | name = "wasm-bindgen" 977 | version = "0.2.80" 978 | source = "registry+https://github.com/rust-lang/crates.io-index" 979 | checksum = "27370197c907c55e3f1a9fbe26f44e937fe6451368324e009cba39e139dc08ad" 980 | dependencies = [ 981 | "cfg-if", 982 | "wasm-bindgen-macro", 983 | ] 984 | 985 | [[package]] 986 | name = "wasm-bindgen-backend" 987 | version = "0.2.80" 988 | source = "registry+https://github.com/rust-lang/crates.io-index" 989 | checksum = "53e04185bfa3a779273da532f5025e33398409573f348985af9a1cbf3774d3f4" 990 | dependencies = [ 991 | "bumpalo", 992 | "lazy_static", 993 | "log", 994 | "proc-macro2", 995 | "quote", 996 | "syn", 997 | "wasm-bindgen-shared", 998 | ] 999 | 1000 | [[package]] 1001 | name = "wasm-bindgen-futures" 1002 | version = "0.4.30" 1003 | source = "registry+https://github.com/rust-lang/crates.io-index" 1004 | checksum = "6f741de44b75e14c35df886aff5f1eb73aa114fa5d4d00dcd37b5e01259bf3b2" 1005 | dependencies = [ 1006 | "cfg-if", 1007 | "js-sys", 1008 | "wasm-bindgen", 1009 | "web-sys", 1010 | ] 1011 | 1012 | [[package]] 1013 | name = "wasm-bindgen-macro" 1014 | version = "0.2.80" 1015 | source = "registry+https://github.com/rust-lang/crates.io-index" 1016 | checksum = "17cae7ff784d7e83a2fe7611cfe766ecf034111b49deb850a3dc7699c08251f5" 1017 | dependencies = [ 1018 | "quote", 1019 | "wasm-bindgen-macro-support", 1020 | ] 1021 | 1022 | [[package]] 1023 | name = "wasm-bindgen-macro-support" 1024 | version = "0.2.80" 1025 | source = "registry+https://github.com/rust-lang/crates.io-index" 1026 | checksum = "99ec0dc7a4756fffc231aab1b9f2f578d23cd391390ab27f952ae0c9b3ece20b" 1027 | dependencies = [ 1028 | "proc-macro2", 1029 | "quote", 1030 | "syn", 1031 | "wasm-bindgen-backend", 1032 | "wasm-bindgen-shared", 1033 | ] 1034 | 1035 | [[package]] 1036 | name = "wasm-bindgen-shared" 1037 | version = "0.2.80" 1038 | source = "registry+https://github.com/rust-lang/crates.io-index" 1039 | checksum = "d554b7f530dee5964d9a9468d95c1f8b8acae4f282807e7d27d4b03099a46744" 1040 | 1041 | [[package]] 1042 | name = "web-sys" 1043 | version = "0.3.57" 1044 | source = "registry+https://github.com/rust-lang/crates.io-index" 1045 | checksum = "7b17e741662c70c8bd24ac5c5b18de314a2c26c32bf8346ee1e6f53de919c283" 1046 | dependencies = [ 1047 | "js-sys", 1048 | "wasm-bindgen", 1049 | ] 1050 | 1051 | [[package]] 1052 | name = "winapi" 1053 | version = "0.3.9" 1054 | source = "registry+https://github.com/rust-lang/crates.io-index" 1055 | checksum = "5c839a674fcd7a98952e593242ea400abe93992746761e38641405d28b00f419" 1056 | dependencies = [ 1057 | "winapi-i686-pc-windows-gnu", 1058 | "winapi-x86_64-pc-windows-gnu", 1059 | ] 1060 | 1061 | [[package]] 1062 | name = "winapi-i686-pc-windows-gnu" 1063 | version = "0.4.0" 1064 | source = "registry+https://github.com/rust-lang/crates.io-index" 1065 | checksum = "ac3b87c63620426dd9b991e5ce0329eff545bccbbb34f3be09ff6fb6ab51b7b6" 1066 | 1067 | [[package]] 1068 | name = "winapi-x86_64-pc-windows-gnu" 1069 | version = "0.4.0" 1070 | source = "registry+https://github.com/rust-lang/crates.io-index" 1071 | checksum = "712e227841d057c1ee1cd2fb22fa7e5a5461ae8e48fa2ca79ec42cfc1931183f" 1072 | 1073 | [[package]] 1074 | name = "winreg" 1075 | version = "0.10.1" 1076 | source = "registry+https://github.com/rust-lang/crates.io-index" 1077 | checksum = "80d0f4e272c85def139476380b12f9ac60926689dd2e01d4923222f40580869d" 1078 | dependencies = [ 1079 | "winapi", 1080 | ] 1081 | -------------------------------------------------------------------------------- /yarn.lock: -------------------------------------------------------------------------------- 1 | # THIS IS AN AUTOGENERATED FILE. DO NOT EDIT THIS FILE DIRECTLY. 2 | # yarn lockfile v1 3 | 4 | 5 | "@cspotcode/source-map-consumer@0.8.0": 6 | version "0.8.0" 7 | resolved "https://registry.yarnpkg.com/@cspotcode/source-map-consumer/-/source-map-consumer-0.8.0.tgz#33bf4b7b39c178821606f669bbc447a6a629786b" 8 | integrity sha512-41qniHzTU8yAGbCp04ohlmSrZf8bkf/iJsl3V0dRGsQN/5GFfx+LbCSsCpp2gqrqjTVg/K6O8ycoV35JIwAzAg== 9 | 10 | "@cspotcode/source-map-support@0.7.0": 11 | version "0.7.0" 12 | resolved "https://registry.yarnpkg.com/@cspotcode/source-map-support/-/source-map-support-0.7.0.tgz#4789840aa859e46d2f3173727ab707c66bf344f5" 13 | integrity sha512-X4xqRHqN8ACt2aHVe51OxeA2HjbcL4MqFqXkrmQszJ1NOUuUu5u6Vqx/0lZSVNku7velL5FC/s5uEAj1lsBMhA== 14 | dependencies: 15 | "@cspotcode/source-map-consumer" "0.8.0" 16 | 17 | "@fastify/ajv-compiler@^1.0.0": 18 | version "1.1.0" 19 | resolved "https://registry.yarnpkg.com/@fastify/ajv-compiler/-/ajv-compiler-1.1.0.tgz#5ce80b1fc8bebffc8c5ba428d5e392d0f9ed10a1" 20 | integrity sha512-gvCOUNpXsWrIQ3A4aXCLIdblL0tDq42BG/2Xw7oxbil9h11uow10ztS2GuFazNBfjbrsZ5nl+nPl5jDSjj5TSg== 21 | dependencies: 22 | ajv "^6.12.6" 23 | 24 | "@fastify/cors@^7.0.0": 25 | version "7.0.0" 26 | resolved "https://registry.yarnpkg.com/@fastify/cors/-/cors-7.0.0.tgz#c67c5a5909498b696bb19578e903f36037ac6f32" 27 | integrity sha512-nlo6ScwagBNJacAZD3KX90xjWLIoV0vN9QqoX1wUE9ZeZMdvkVkMZCGlxEtr00NshV0X5wDge4w5rwox7rRzSg== 28 | dependencies: 29 | fastify-plugin "^3.0.0" 30 | vary "^1.1.2" 31 | 32 | "@prisma/client@^3.13.0": 33 | version "3.13.0" 34 | resolved "https://registry.yarnpkg.com/@prisma/client/-/client-3.13.0.tgz#84511ebdf6ba75f77ca08495b9f73f22c4255654" 35 | integrity sha512-lnEA2tTyVbO5mS1ehmHJQKBDiKB8shaR6s3azwj3Azfi5XHIfnqmkolLCvUeFYnkDCNVzGXJpUgKwQt/UOOYVQ== 36 | dependencies: 37 | "@prisma/engines-version" "3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b" 38 | 39 | "@prisma/engines-version@3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b": 40 | version "3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b" 41 | resolved "https://registry.yarnpkg.com/@prisma/engines-version/-/engines-version-3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b.tgz#676aca309d66d9be2aad8911ca31f1ee5561041c" 42 | integrity sha512-TGp9rvgJIKo8NgvAHSwOosbut9mTA7VC6/rpQI9gh+ySSRjdQFhbGyNUiOcQrlI9Ob2DWeO7y4HEnhdKxYiECg== 43 | 44 | "@prisma/engines@3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b": 45 | version "3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b" 46 | resolved "https://registry.yarnpkg.com/@prisma/engines/-/engines-3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b.tgz#d3a457cec4ef7a3b3412c45b1f2eac68c974474b" 47 | integrity sha512-Ip9CcCeUocH61eXu4BUGpvl5KleQyhcUVLpWCv+0ZmDv44bFaDpREqjGHHdRupvPN/ugB6gTlD9b9ewdj02yVA== 48 | 49 | "@sindresorhus/is@^0.14.0": 50 | version "0.14.0" 51 | resolved "https://registry.yarnpkg.com/@sindresorhus/is/-/is-0.14.0.tgz#9fb3a3cf3132328151f353de4632e01e52102bea" 52 | integrity sha512-9NET910DNaIPngYnLLPeg+Ogzqsi9uM4mSboU5y6p8S5DzMTVEsJZrawi+BoDNUVBa2DhJqQYUFvMDfgU062LQ== 53 | 54 | "@szmarczak/http-timer@^1.1.2": 55 | version "1.1.2" 56 | resolved "https://registry.yarnpkg.com/@szmarczak/http-timer/-/http-timer-1.1.2.tgz#b1665e2c461a2cd92f4c1bbf50d5454de0d4b421" 57 | integrity sha512-XIB2XbzHTN6ieIjfIMV9hlVcfPU26s2vafYWQcZHWXHOxiaRZYEDKEwdl129Zyg50+foYV2jCgtrqSA6qNuNSA== 58 | dependencies: 59 | defer-to-connect "^1.0.1" 60 | 61 | "@tsconfig/node10@^1.0.7": 62 | version "1.0.8" 63 | resolved "https://registry.yarnpkg.com/@tsconfig/node10/-/node10-1.0.8.tgz#c1e4e80d6f964fbecb3359c43bd48b40f7cadad9" 64 | integrity sha512-6XFfSQmMgq0CFLY1MslA/CPUfhIL919M1rMsa5lP2P097N2Wd1sSX0tx1u4olM16fLNhtHZpRhedZJphNJqmZg== 65 | 66 | "@tsconfig/node12@^1.0.7": 67 | version "1.0.9" 68 | resolved "https://registry.yarnpkg.com/@tsconfig/node12/-/node12-1.0.9.tgz#62c1f6dee2ebd9aead80dc3afa56810e58e1a04c" 69 | integrity sha512-/yBMcem+fbvhSREH+s14YJi18sp7J9jpuhYByADT2rypfajMZZN4WQ6zBGgBKp53NKmqI36wFYDb3yaMPurITw== 70 | 71 | "@tsconfig/node14@^1.0.0": 72 | version "1.0.1" 73 | resolved "https://registry.yarnpkg.com/@tsconfig/node14/-/node14-1.0.1.tgz#95f2d167ffb9b8d2068b0b235302fafd4df711f2" 74 | integrity sha512-509r2+yARFfHHE7T6Puu2jjkoycftovhXRqW328PDXTVGKihlb1P8Z9mMZH04ebyajfRY7dedfGynlrFHJUQCg== 75 | 76 | "@tsconfig/node16@^1.0.2": 77 | version "1.0.2" 78 | resolved "https://registry.yarnpkg.com/@tsconfig/node16/-/node16-1.0.2.tgz#423c77877d0569db20e1fc80885ac4118314010e" 79 | integrity sha512-eZxlbI8GZscaGS7kkc/trHTT5xgrjH3/1n2JDwusC9iahPKWMRvRjJSAN5mCXviuTGQ/lHnhvv8Q1YTpnfz9gA== 80 | 81 | "@types/cookie@^0.5.1": 82 | version "0.5.1" 83 | resolved "https://registry.yarnpkg.com/@types/cookie/-/cookie-0.5.1.tgz#b29aa1f91a59f35e29ff8f7cb24faf1a3a750554" 84 | integrity sha512-COUnqfB2+ckwXXSFInsFdOAWQzCCx+a5hq2ruyj+Vjund94RJQd4LG2u9hnvJrTgunKAaax7ancBYlDrNYxA0g== 85 | 86 | "@types/cron@^1.7.3": 87 | version "1.7.3" 88 | resolved "https://registry.yarnpkg.com/@types/cron/-/cron-1.7.3.tgz#993db7d54646f61128c851607b64ba4495deae93" 89 | integrity sha512-iPmUXyIJG1Js+ldPYhOQcYU3kCAQ2FWrSkm1FJPoii2eYSn6wEW6onPukNTT0bfiflexNSRPl6KWmAIqS+36YA== 90 | dependencies: 91 | "@types/node" "*" 92 | moment ">=2.14.0" 93 | 94 | "@types/node@*": 95 | version "17.0.25" 96 | resolved "https://registry.yarnpkg.com/@types/node/-/node-17.0.25.tgz#527051f3c2f77aa52e5dc74e45a3da5fb2301448" 97 | integrity sha512-wANk6fBrUwdpY4isjWrKTufkrXdu1D2YHCot2fD/DfWxF5sMrVSA+KN7ydckvaTCh0HiqX9IVl0L5/ZoXg5M7w== 98 | 99 | "@types/node@^17.0.29": 100 | version "17.0.29" 101 | resolved "https://registry.yarnpkg.com/@types/node/-/node-17.0.29.tgz#7f2e1159231d4a077bb660edab0fde373e375a3d" 102 | integrity sha512-tx5jMmMFwx7wBwq/V7OohKDVb/JwJU5qCVkeLMh1//xycAJ/ESuw9aJ9SEtlCZDYi2pBfe4JkisSoAtbOsBNAA== 103 | 104 | abbrev@1: 105 | version "1.1.1" 106 | resolved "https://registry.yarnpkg.com/abbrev/-/abbrev-1.1.1.tgz#f8f2c887ad10bf67f634f005b6987fed3179aac8" 107 | integrity sha512-nne9/IiQ/hzIhY6pdDnbBtz7DjPTKrY00P/zvPSm5pOFkl6xuGrGnXn/VtTNNfNtAfZ9/1RtehkszU9qcTii0Q== 108 | 109 | abstract-logging@^2.0.0: 110 | version "2.0.1" 111 | resolved "https://registry.yarnpkg.com/abstract-logging/-/abstract-logging-2.0.1.tgz#6b0c371df212db7129b57d2e7fcf282b8bf1c839" 112 | integrity sha512-2BjRTZxTPvheOvGbBslFSYOUkr+SjPtOnrLP33f+VIWLzezQpZcqVg7ja3L4dBXmzzgwT+a029jRx5PCi3JuiA== 113 | 114 | acorn-walk@^8.1.1: 115 | version "8.2.0" 116 | resolved "https://registry.yarnpkg.com/acorn-walk/-/acorn-walk-8.2.0.tgz#741210f2e2426454508853a2f44d0ab83b7f69c1" 117 | integrity sha512-k+iyHEuPgSw6SbuDpGQM+06HQUa04DZ3o+F6CSzXMvvI5KMvnaEqXe+YVe555R9nn6GPt404fos4wcgpw12SDA== 118 | 119 | acorn@^8.4.1: 120 | version "8.7.0" 121 | resolved "https://registry.yarnpkg.com/acorn/-/acorn-8.7.0.tgz#90951fde0f8f09df93549481e5fc141445b791cf" 122 | integrity sha512-V/LGr1APy+PXIwKebEWrkZPwoeoF+w1jiOBUmuxuiUIaOHtob8Qc9BTrYo7VuI5fR8tqsy+buA2WFooR5olqvQ== 123 | 124 | ajv@^6.11.0, ajv@^6.12.6: 125 | version "6.12.6" 126 | resolved "https://registry.yarnpkg.com/ajv/-/ajv-6.12.6.tgz#baf5a62e802b07d977034586f8c3baf5adf26df4" 127 | integrity sha512-j3fVLgvTo527anyYyJOGTYJbG+vnnQYvE0m5mmkc1TK+nxAppkCLMIL0aZ4dblVCNoGShhm+kzE4ZUykBoMg4g== 128 | dependencies: 129 | fast-deep-equal "^3.1.1" 130 | fast-json-stable-stringify "^2.0.0" 131 | json-schema-traverse "^0.4.1" 132 | uri-js "^4.2.2" 133 | 134 | ajv@^8.1.0: 135 | version "8.11.0" 136 | resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.11.0.tgz#977e91dd96ca669f54a11e23e378e33b884a565f" 137 | integrity sha512-wGgprdCvMalC0BztXvitD2hC04YffAvtsUn93JbGXYLAtCUO4xd17mCCZQxUOItiBwZvJScWo8NIvQMQ71rdpg== 138 | dependencies: 139 | fast-deep-equal "^3.1.1" 140 | json-schema-traverse "^1.0.0" 141 | require-from-string "^2.0.2" 142 | uri-js "^4.2.2" 143 | 144 | ansi-align@^3.0.0: 145 | version "3.0.1" 146 | resolved "https://registry.yarnpkg.com/ansi-align/-/ansi-align-3.0.1.tgz#0cdf12e111ace773a86e9a1fad1225c43cb19a59" 147 | integrity sha512-IOfwwBF5iczOjp/WeY4YxyjqAFMQoZufdQWDd19SEExbVLNXqvpzSJ/M7Za4/sCPmQ0+GRquoA7bGcINcxew6w== 148 | dependencies: 149 | string-width "^4.1.0" 150 | 151 | ansi-regex@^5.0.1: 152 | version "5.0.1" 153 | resolved "https://registry.yarnpkg.com/ansi-regex/-/ansi-regex-5.0.1.tgz#082cb2c89c9fe8659a311a53bd6a4dc5301db304" 154 | integrity sha512-quJQXlTSUGL2LH9SUXo8VwsY4soanhgo6LNSm84E1LBcE8s3O0wpdiRzyR9z/ZZJMlMWv37qOOb9pdJlMUEKFQ== 155 | 156 | ansi-styles@^3.2.1: 157 | version "3.2.1" 158 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-3.2.1.tgz#41fbb20243e50b12be0f04b8dedbf07520ce841d" 159 | integrity sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA== 160 | dependencies: 161 | color-convert "^1.9.0" 162 | 163 | ansi-styles@^4.0.0, ansi-styles@^4.1.0: 164 | version "4.3.0" 165 | resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" 166 | integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== 167 | dependencies: 168 | color-convert "^2.0.1" 169 | 170 | anymatch@~3.1.2: 171 | version "3.1.2" 172 | resolved "https://registry.yarnpkg.com/anymatch/-/anymatch-3.1.2.tgz#c0557c096af32f106198f4f4e2a383537e378716" 173 | integrity sha512-P43ePfOAIupkguHUycrc4qJ9kz8ZiuOUijaETwX7THt0Y/GNK7v0aa8rY816xWjZ7rJdA5XdMcpVFTKMq+RvWg== 174 | dependencies: 175 | normalize-path "^3.0.0" 176 | picomatch "^2.0.4" 177 | 178 | archy@^1.0.0: 179 | version "1.0.0" 180 | resolved "https://registry.yarnpkg.com/archy/-/archy-1.0.0.tgz#f9c8c13757cc1dd7bc379ac77b2c62a5c2868c40" 181 | integrity sha1-+cjBN1fMHde8N5rHeyxipcKGjEA= 182 | 183 | arg@^4.1.0: 184 | version "4.1.3" 185 | resolved "https://registry.yarnpkg.com/arg/-/arg-4.1.3.tgz#269fc7ad5b8e42cb63c896d5666017261c144089" 186 | integrity sha512-58S9QDqG0Xx27YwPSt9fJxivjYl432YCwfDMfZ+71RAqUrZef7LrKQZ3LHLOwCS4FLNBplP533Zx895SeOCHvA== 187 | 188 | args@^5.0.1: 189 | version "5.0.1" 190 | resolved "https://registry.yarnpkg.com/args/-/args-5.0.1.tgz#4bf298df90a4799a09521362c579278cc2fdd761" 191 | integrity sha512-1kqmFCFsPffavQFGt8OxJdIcETti99kySRUPMpOhaGjL6mRJn8HFU1OxKY5bMqfZKUwTQc1mZkAjmGYaVOHFtQ== 192 | dependencies: 193 | camelcase "5.0.0" 194 | chalk "2.4.2" 195 | leven "2.1.0" 196 | mri "1.1.4" 197 | 198 | atomic-sleep@^1.0.0: 199 | version "1.0.0" 200 | resolved "https://registry.yarnpkg.com/atomic-sleep/-/atomic-sleep-1.0.0.tgz#eb85b77a601fc932cfe432c5acd364a9e2c9075b" 201 | integrity sha512-kNOjDqAh7px0XWNI+4QbzoiR/nTkHAWNud2uvnJquD1/x5a7EQZMJT0AczqK0Qn67oY/TTQ1LbUKajZpp3I9tQ== 202 | 203 | avvio@^7.1.2: 204 | version "7.2.5" 205 | resolved "https://registry.yarnpkg.com/avvio/-/avvio-7.2.5.tgz#65ba255f10b0bea7ac6eded71a5344cd88f5de19" 206 | integrity sha512-AOhBxyLVdpOad3TujtC9kL/9r3HnTkxwQ5ggOsYrvvZP1cCFvzHWJd5XxZDFuTn+IN8vkKSG5SEJrd27vCSbeA== 207 | dependencies: 208 | archy "^1.0.0" 209 | debug "^4.0.0" 210 | fastq "^1.6.1" 211 | queue-microtask "^1.1.2" 212 | 213 | balanced-match@^1.0.0: 214 | version "1.0.2" 215 | resolved "https://registry.yarnpkg.com/balanced-match/-/balanced-match-1.0.2.tgz#e83e3a7e3f300b34cb9d87f615fa0cbf357690ee" 216 | integrity sha512-3oSeUO0TMV67hN1AmbXsK4yaqU7tjiHlbxRDZOpH0KW9+CeX4bRAaX0Anxt0tx2MrpRpWwQaPwIlISEJhYU5Pw== 217 | 218 | binary-extensions@^2.0.0: 219 | version "2.2.0" 220 | resolved "https://registry.yarnpkg.com/binary-extensions/-/binary-extensions-2.2.0.tgz#75f502eeaf9ffde42fc98829645be4ea76bd9e2d" 221 | integrity sha512-jDctJ/IVQbZoJykoeHbhXpOlNBqGNcwXJKJog42E5HDPUwQTSdjCHdihjj0DlnheQ7blbT6dHOafNAiS8ooQKA== 222 | 223 | boxen@^5.0.0: 224 | version "5.1.2" 225 | resolved "https://registry.yarnpkg.com/boxen/-/boxen-5.1.2.tgz#788cb686fc83c1f486dfa8a40c68fc2b831d2b50" 226 | integrity sha512-9gYgQKXx+1nP8mP7CzFyaUARhg7D3n1dF/FnErWmu9l6JvGpNUN278h0aSb+QjoiKSWG+iZ3uHrcqk0qrY9RQQ== 227 | dependencies: 228 | ansi-align "^3.0.0" 229 | camelcase "^6.2.0" 230 | chalk "^4.1.0" 231 | cli-boxes "^2.2.1" 232 | string-width "^4.2.2" 233 | type-fest "^0.20.2" 234 | widest-line "^3.1.0" 235 | wrap-ansi "^7.0.0" 236 | 237 | brace-expansion@^1.1.7: 238 | version "1.1.11" 239 | resolved "https://registry.yarnpkg.com/brace-expansion/-/brace-expansion-1.1.11.tgz#3c7fcbf529d87226f3d2f52b966ff5271eb441dd" 240 | integrity sha512-iCuPHDFgrHX7H2vEI/5xpz07zSHB00TpugqhmYtVmMO6518mCuRMoOYFldEBl0g187ufozdaHgWKcYFb61qGiA== 241 | dependencies: 242 | balanced-match "^1.0.0" 243 | concat-map "0.0.1" 244 | 245 | braces@~3.0.2: 246 | version "3.0.2" 247 | resolved "https://registry.yarnpkg.com/braces/-/braces-3.0.2.tgz#3454e1a462ee8d599e236df336cd9ea4f8afe107" 248 | integrity sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A== 249 | dependencies: 250 | fill-range "^7.0.1" 251 | 252 | cacheable-request@^6.0.0: 253 | version "6.1.0" 254 | resolved "https://registry.yarnpkg.com/cacheable-request/-/cacheable-request-6.1.0.tgz#20ffb8bd162ba4be11e9567d823db651052ca912" 255 | integrity sha512-Oj3cAGPCqOZX7Rz64Uny2GYAZNliQSqfbePrgAQ1wKAihYmCUnraBtJtKcGR4xz7wF+LoJC+ssFZvv5BgF9Igg== 256 | dependencies: 257 | clone-response "^1.0.2" 258 | get-stream "^5.1.0" 259 | http-cache-semantics "^4.0.0" 260 | keyv "^3.0.0" 261 | lowercase-keys "^2.0.0" 262 | normalize-url "^4.1.0" 263 | responselike "^1.0.2" 264 | 265 | camelcase@5.0.0: 266 | version "5.0.0" 267 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-5.0.0.tgz#03295527d58bd3cd4aa75363f35b2e8d97be2f42" 268 | integrity sha512-faqwZqnWxbxn+F1d399ygeamQNy3lPp/H9H6rNrqYh4FSVCtcY+3cub1MxA8o9mDd55mM8Aghuu/kuyYA6VTsA== 269 | 270 | camelcase@^6.2.0: 271 | version "6.3.0" 272 | resolved "https://registry.yarnpkg.com/camelcase/-/camelcase-6.3.0.tgz#5685b95eb209ac9c0c177467778c9c84df58ba9a" 273 | integrity sha512-Gmy6FhYlCY7uOElZUSbxo2UCDH8owEk996gkbrpsgGtrJLM3J7jGxl9Ic7Qwwj4ivOE5AWZWRMecDdF7hqGjFA== 274 | 275 | chalk@2.4.2: 276 | version "2.4.2" 277 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-2.4.2.tgz#cd42541677a54333cf541a49108c1432b44c9424" 278 | integrity sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ== 279 | dependencies: 280 | ansi-styles "^3.2.1" 281 | escape-string-regexp "^1.0.5" 282 | supports-color "^5.3.0" 283 | 284 | chalk@^4.1.0: 285 | version "4.1.2" 286 | resolved "https://registry.yarnpkg.com/chalk/-/chalk-4.1.2.tgz#aac4e2b7734a740867aeb16bf02aad556a1e7a01" 287 | integrity sha512-oKnbhFyRIXpUuez8iBMmyEa4nbj4IOQyuhc/wy9kY7/WVPcwIO9VA668Pu8RkO7+0G76SLROeyw9CpQ061i4mA== 288 | dependencies: 289 | ansi-styles "^4.1.0" 290 | supports-color "^7.1.0" 291 | 292 | chokidar@^3.5.2: 293 | version "3.5.3" 294 | resolved "https://registry.yarnpkg.com/chokidar/-/chokidar-3.5.3.tgz#1cf37c8707b932bd1af1ae22c0432e2acd1903bd" 295 | integrity sha512-Dr3sfKRP6oTcjf2JmUmFJfeVMvXBdegxB0iVQ5eb2V10uFJUCAS8OByZdVAyVb8xXNz3GjjTgj9kLWsZTqE6kw== 296 | dependencies: 297 | anymatch "~3.1.2" 298 | braces "~3.0.2" 299 | glob-parent "~5.1.2" 300 | is-binary-path "~2.1.0" 301 | is-glob "~4.0.1" 302 | normalize-path "~3.0.0" 303 | readdirp "~3.6.0" 304 | optionalDependencies: 305 | fsevents "~2.3.2" 306 | 307 | ci-info@^2.0.0: 308 | version "2.0.0" 309 | resolved "https://registry.yarnpkg.com/ci-info/-/ci-info-2.0.0.tgz#67a9e964be31a51e15e5010d58e6f12834002f46" 310 | integrity sha512-5tK7EtrZ0N+OLFMthtqOj4fI2Jeb88C4CAZPu25LDVUgXJ0A3Js4PMGqrn0JU1W0Mh1/Z8wZzYPxqUrXeBboCQ== 311 | 312 | cli-boxes@^2.2.1: 313 | version "2.2.1" 314 | resolved "https://registry.yarnpkg.com/cli-boxes/-/cli-boxes-2.2.1.tgz#ddd5035d25094fce220e9cab40a45840a440318f" 315 | integrity sha512-y4coMcylgSCdVinjiDBuR8PCC2bLjyGTwEmPb9NHR/QaNU6EUOXcTY/s6VjGMD6ENSEaeQYHCY0GNGS5jfMwPw== 316 | 317 | clone-response@^1.0.2: 318 | version "1.0.2" 319 | resolved "https://registry.yarnpkg.com/clone-response/-/clone-response-1.0.2.tgz#d1dc973920314df67fbeb94223b4ee350239e96b" 320 | integrity sha1-0dyXOSAxTfZ/vrlCI7TuNQI56Ws= 321 | dependencies: 322 | mimic-response "^1.0.0" 323 | 324 | color-convert@^1.9.0: 325 | version "1.9.3" 326 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-1.9.3.tgz#bb71850690e1f136567de629d2d5471deda4c1e8" 327 | integrity sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg== 328 | dependencies: 329 | color-name "1.1.3" 330 | 331 | color-convert@^2.0.1: 332 | version "2.0.1" 333 | resolved "https://registry.yarnpkg.com/color-convert/-/color-convert-2.0.1.tgz#72d3a68d598c9bdb3af2ad1e84f21d896abd4de3" 334 | integrity sha512-RRECPsj7iu/xb5oKYcsFHSppFNnsj/52OVTRKb4zP5onXwVF3zVmmToNcOfGC+CRDpfK/U584fMg38ZHCaElKQ== 335 | dependencies: 336 | color-name "~1.1.4" 337 | 338 | color-name@1.1.3: 339 | version "1.1.3" 340 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" 341 | integrity sha1-p9BVi9icQveV3UIyj3QIMcpTvCU= 342 | 343 | color-name@~1.1.4: 344 | version "1.1.4" 345 | resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" 346 | integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== 347 | 348 | colorette@^2.0.7: 349 | version "2.0.16" 350 | resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.16.tgz#713b9af84fdb000139f04546bd4a93f62a5085da" 351 | integrity sha512-hUewv7oMjCp+wkBv5Rm0v87eJhq4woh5rSR+42YSQJKecCqgIqNkZ6lAlQms/BwHPJA5NKMRlpxPRv0n8HQW6g== 352 | 353 | concat-map@0.0.1: 354 | version "0.0.1" 355 | resolved "https://registry.yarnpkg.com/concat-map/-/concat-map-0.0.1.tgz#d8a96bd77fd68df7793a73036a3ba0d5405d477b" 356 | integrity sha1-2Klr13/Wjfd5OnMDajug1UBdR3s= 357 | 358 | configstore@^5.0.1: 359 | version "5.0.1" 360 | resolved "https://registry.yarnpkg.com/configstore/-/configstore-5.0.1.tgz#d365021b5df4b98cdd187d6a3b0e3f6a7cc5ed96" 361 | integrity sha512-aMKprgk5YhBNyH25hj8wGt2+D52Sw1DRRIzqBwLp2Ya9mFmY8KPvvtvmna8SxVR9JMZ4kzMD68N22vlaRpkeFA== 362 | dependencies: 363 | dot-prop "^5.2.0" 364 | graceful-fs "^4.1.2" 365 | make-dir "^3.0.0" 366 | unique-string "^2.0.0" 367 | write-file-atomic "^3.0.0" 368 | xdg-basedir "^4.0.0" 369 | 370 | cookie@^0.4.0: 371 | version "0.4.2" 372 | resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.4.2.tgz#0e41f24de5ecf317947c82fc789e06a884824432" 373 | integrity sha512-aSWTXFzaKWkvHO1Ny/s+ePFpvKsPnjc551iI41v3ny/ow6tBG5Vd+FuqGNhh1LxOmVzOlGUriIlOaokOvhaStA== 374 | 375 | cookie@^0.5.0: 376 | version "0.5.0" 377 | resolved "https://registry.yarnpkg.com/cookie/-/cookie-0.5.0.tgz#d1f5d71adec6558c58f389987c366aa47e994f8b" 378 | integrity sha512-YZ3GUyn/o8gfKJlnlX7g7xq4gyO6OSuhGPKaaGssGB2qgDUS0gPgtTvoyZLTt9Ab6dC4hfc9dV5arkvc/OCmrw== 379 | 380 | create-require@^1.1.0: 381 | version "1.1.1" 382 | resolved "https://registry.yarnpkg.com/create-require/-/create-require-1.1.1.tgz#c1d7e8f1e5f6cfc9ff65f9cd352d37348756c333" 383 | integrity sha512-dcKFX3jn0MpIaXjisoRvexIJVEKzaq7z2rZKxf+MSr9TkdmHmsU4m2lcLojrj/FHl8mk5VxMmYA+ftRkP/3oKQ== 384 | 385 | cron@^1.8.2: 386 | version "1.8.2" 387 | resolved "https://registry.yarnpkg.com/cron/-/cron-1.8.2.tgz#4ac5e3c55ba8c163d84f3407bde94632da8370ce" 388 | integrity sha512-Gk2c4y6xKEO8FSAUTklqtfSr7oTq0CiPQeLBG5Fl0qoXpZyMcj1SG59YL+hqq04bu6/IuEA7lMkYDAplQNKkyg== 389 | dependencies: 390 | moment-timezone "^0.5.x" 391 | 392 | crypto-random-string@^2.0.0: 393 | version "2.0.0" 394 | resolved "https://registry.yarnpkg.com/crypto-random-string/-/crypto-random-string-2.0.0.tgz#ef2a7a966ec11083388369baa02ebead229b30d5" 395 | integrity sha512-v1plID3y9r/lPhviJ1wrXpLeyUIGAZ2SHNYTEapm7/8A9nLPoyvVp3RK/EPFqn5kEznyWgYZNsRtYYIWbuG8KA== 396 | 397 | data-uri-to-buffer@^4.0.0: 398 | version "4.0.0" 399 | resolved "https://registry.yarnpkg.com/data-uri-to-buffer/-/data-uri-to-buffer-4.0.0.tgz#b5db46aea50f6176428ac05b73be39a57701a64b" 400 | integrity sha512-Vr3mLBA8qWmcuschSLAOogKgQ/Jwxulv3RNE4FXnYWRGujzrRWQI4m12fQqRkwX06C0KanhLr4hK+GydchZsaA== 401 | 402 | dateformat@^4.6.3: 403 | version "4.6.3" 404 | resolved "https://registry.yarnpkg.com/dateformat/-/dateformat-4.6.3.tgz#556fa6497e5217fedb78821424f8a1c22fa3f4b5" 405 | integrity sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA== 406 | 407 | debug@^3.2.7: 408 | version "3.2.7" 409 | resolved "https://registry.yarnpkg.com/debug/-/debug-3.2.7.tgz#72580b7e9145fb39b6676f9c5e5fb100b934179a" 410 | integrity sha512-CFjzYYAi4ThfiQvizrFQevTTXHtnCqWfe7x1AhgEscTz6ZbLbfoLRLPugTQyBth6f8ZERVUSyWHFD/7Wu4t1XQ== 411 | dependencies: 412 | ms "^2.1.1" 413 | 414 | debug@^4.0.0: 415 | version "4.3.4" 416 | resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" 417 | integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== 418 | dependencies: 419 | ms "2.1.2" 420 | 421 | decompress-response@^3.3.0: 422 | version "3.3.0" 423 | resolved "https://registry.yarnpkg.com/decompress-response/-/decompress-response-3.3.0.tgz#80a4dd323748384bfa248083622aedec982adff3" 424 | integrity sha1-gKTdMjdIOEv6JICDYirt7Jgq3/M= 425 | dependencies: 426 | mimic-response "^1.0.0" 427 | 428 | deep-extend@^0.6.0: 429 | version "0.6.0" 430 | resolved "https://registry.yarnpkg.com/deep-extend/-/deep-extend-0.6.0.tgz#c4fa7c95404a17a9c3e8ca7e1537312b736330ac" 431 | integrity sha512-LOHxIOaPYdHlJRtCQfDIVZtfw/ufM8+rVj649RIHzcm/vGwQRXFt6OPqIFWsm2XEMrNIEtWR64sY1LEKD2vAOA== 432 | 433 | deepmerge@^4.2.2: 434 | version "4.2.2" 435 | resolved "https://registry.yarnpkg.com/deepmerge/-/deepmerge-4.2.2.tgz#44d2ea3679b8f4d4ffba33f03d865fc1e7bf4955" 436 | integrity sha512-FJ3UgI4gIl+PHZm53knsuSFpE+nESMr7M4v9QcgB7S63Kj/6WqMiFQJpBBYz1Pt+66bZpP3Q7Lye0Oo9MPKEdg== 437 | 438 | defer-to-connect@^1.0.1: 439 | version "1.1.3" 440 | resolved "https://registry.yarnpkg.com/defer-to-connect/-/defer-to-connect-1.1.3.tgz#331ae050c08dcf789f8c83a7b81f0ed94f4ac591" 441 | integrity sha512-0ISdNousHvZT2EiFlZeZAHBUvSxmKswVCEf8hW7KWgG4a8MVEu/3Vb6uWYozkjylyCxe0JBIiRB1jV45S70WVQ== 442 | 443 | diff@^4.0.1: 444 | version "4.0.2" 445 | resolved "https://registry.yarnpkg.com/diff/-/diff-4.0.2.tgz#60f3aecb89d5fae520c11aa19efc2bb982aade7d" 446 | integrity sha512-58lmxKSA4BNyLz+HHMUzlOEpg09FV+ev6ZMe3vJihgdxzgcwZ8VoEEPmALCZG9LmqfVoNMMKpttIYTVG6uDY7A== 447 | 448 | dot-prop@^5.2.0: 449 | version "5.3.0" 450 | resolved "https://registry.yarnpkg.com/dot-prop/-/dot-prop-5.3.0.tgz#90ccce708cd9cd82cc4dc8c3ddd9abdd55b20e88" 451 | integrity sha512-QM8q3zDe58hqUqjraQOmzZ1LIH9SWQJTlEKCH4kJ2oQvLZk7RbQXvtDM2XEq3fwkV9CCvvH4LA0AV+ogFsBM2Q== 452 | dependencies: 453 | is-obj "^2.0.0" 454 | 455 | dotenv@^16.0.0: 456 | version "16.0.0" 457 | resolved "https://registry.yarnpkg.com/dotenv/-/dotenv-16.0.0.tgz#c619001253be89ebb638d027b609c75c26e47411" 458 | integrity sha512-qD9WU0MPM4SWLPJy/r2Be+2WgQj8plChsyrCNQzW/0WjvcJQiKQJ9mH3ZgB3fxbUUxgc/11ZJ0Fi5KiimWGz2Q== 459 | 460 | duplexer3@^0.1.4: 461 | version "0.1.4" 462 | resolved "https://registry.yarnpkg.com/duplexer3/-/duplexer3-0.1.4.tgz#ee01dd1cac0ed3cbc7fdbea37dc0a8f1ce002ce2" 463 | integrity sha1-7gHdHKwO08vH/b6jfcCo8c4ALOI= 464 | 465 | duplexify@^4.1.2: 466 | version "4.1.2" 467 | resolved "https://registry.yarnpkg.com/duplexify/-/duplexify-4.1.2.tgz#18b4f8d28289132fa0b9573c898d9f903f81c7b0" 468 | integrity sha512-fz3OjcNCHmRP12MJoZMPglx8m4rrFP8rovnk4vT8Fs+aonZoCwGg10dSsQsfP/E62eZcPTMSMP6686fu9Qlqtw== 469 | dependencies: 470 | end-of-stream "^1.4.1" 471 | inherits "^2.0.3" 472 | readable-stream "^3.1.1" 473 | stream-shift "^1.0.0" 474 | 475 | emoji-regex@^8.0.0: 476 | version "8.0.0" 477 | resolved "https://registry.yarnpkg.com/emoji-regex/-/emoji-regex-8.0.0.tgz#e818fd69ce5ccfcb404594f842963bf53164cc37" 478 | integrity sha512-MSjYzcWNOA0ewAHpz0MxpYFvwg6yjy1NG3xteoqz644VCo/RPgnr1/GGt+ic3iJTzQ8Eu3TdM14SawnVUmGE6A== 479 | 480 | end-of-stream@^1.1.0, end-of-stream@^1.4.1: 481 | version "1.4.4" 482 | resolved "https://registry.yarnpkg.com/end-of-stream/-/end-of-stream-1.4.4.tgz#5ae64a5f45057baf3626ec14da0ca5e4b2431eb0" 483 | integrity sha512-+uw1inIHVPQoaVuHzRyXd21icM+cnt4CzD5rW+NC1wjOUSTOs+Te7FOv7AhN7vS9x/oIyhLP5PR1H+phQAHu5Q== 484 | dependencies: 485 | once "^1.4.0" 486 | 487 | escape-goat@^2.0.0: 488 | version "2.1.1" 489 | resolved "https://registry.yarnpkg.com/escape-goat/-/escape-goat-2.1.1.tgz#1b2dc77003676c457ec760b2dc68edb648188675" 490 | integrity sha512-8/uIhbG12Csjy2JEW7D9pHbreaVaS/OpN3ycnyvElTdwM5n6GY6W6e2IPemfvGZeUMqZ9A/3GqIZMgKnBhAw/Q== 491 | 492 | escape-string-regexp@^1.0.5: 493 | version "1.0.5" 494 | resolved "https://registry.yarnpkg.com/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz#1b61c0562190a8dff6ae3bb2cf0200ca130b86d4" 495 | integrity sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ= 496 | 497 | fast-decode-uri-component@^1.0.1: 498 | version "1.0.1" 499 | resolved "https://registry.yarnpkg.com/fast-decode-uri-component/-/fast-decode-uri-component-1.0.1.tgz#46f8b6c22b30ff7a81357d4f59abfae938202543" 500 | integrity sha512-WKgKWg5eUxvRZGwW8FvfbaH7AXSh2cL+3j5fMGzUMCxWBJ3dV3a7Wz8y2f/uQ0e3B6WmodD3oS54jTQ9HVTIIg== 501 | 502 | fast-deep-equal@^3.1.1, fast-deep-equal@^3.1.3: 503 | version "3.1.3" 504 | resolved "https://registry.yarnpkg.com/fast-deep-equal/-/fast-deep-equal-3.1.3.tgz#3a7d56b559d6cbc3eb512325244e619a65c6c525" 505 | integrity sha512-f3qQ9oQy9j2AhBe/H9VC91wLmKBCCU/gDOnKNAYG5hswO7BLKj09Hc5HYNz9cGI++xlpDCIgDaitVs03ATR84Q== 506 | 507 | fast-json-stable-stringify@^2.0.0: 508 | version "2.1.0" 509 | resolved "https://registry.yarnpkg.com/fast-json-stable-stringify/-/fast-json-stable-stringify-2.1.0.tgz#874bf69c6f404c2b5d99c481341399fd55892633" 510 | integrity sha512-lhd/wF+Lk98HZoTCtlVraHtfh5XYijIjalXck7saUtuanSDyLMxnHhSXEDJqHxD7msR8D0uCmqlkwjCV8xvwHw== 511 | 512 | fast-json-stringify@^2.5.2: 513 | version "2.7.13" 514 | resolved "https://registry.yarnpkg.com/fast-json-stringify/-/fast-json-stringify-2.7.13.tgz#277aa86c2acba4d9851bd6108ed657aa327ed8c0" 515 | integrity sha512-ar+hQ4+OIurUGjSJD1anvYSDcUflywhKjfxnsW4TBTD7+u0tJufv6DKRWoQk3vI6YBOWMoz0TQtfbe7dxbQmvA== 516 | dependencies: 517 | ajv "^6.11.0" 518 | deepmerge "^4.2.2" 519 | rfdc "^1.2.0" 520 | string-similarity "^4.0.1" 521 | 522 | fast-redact@^3.0.0: 523 | version "3.1.1" 524 | resolved "https://registry.yarnpkg.com/fast-redact/-/fast-redact-3.1.1.tgz#790fcff8f808c2e12fabbfb2be5cb2deda448fa0" 525 | integrity sha512-odVmjC8x8jNeMZ3C+rPMESzXVSEU8tSWSHv9HFxP2mm89G/1WwqhrerJDQm9Zus8X6aoRgQDThKqptdNA6bt+A== 526 | 527 | fast-safe-stringify@^2.0.7, fast-safe-stringify@^2.0.8: 528 | version "2.1.1" 529 | resolved "https://registry.yarnpkg.com/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz#c406a83b6e70d9e35ce3b30a81141df30aeba884" 530 | integrity sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA== 531 | 532 | fastify-error@^0.3.0: 533 | version "0.3.1" 534 | resolved "https://registry.yarnpkg.com/fastify-error/-/fastify-error-0.3.1.tgz#8eb993e15e3cf57f0357fc452af9290f1c1278d2" 535 | integrity sha512-oCfpcsDndgnDVgiI7bwFKAun2dO+4h84vBlkWsWnz/OUK9Reff5UFoFl241xTiLeHWX/vU9zkDVXqYUxjOwHcQ== 536 | 537 | fastify-plugin@^3.0.0: 538 | version "3.0.1" 539 | resolved "https://registry.yarnpkg.com/fastify-plugin/-/fastify-plugin-3.0.1.tgz#79e84c29f401020f38b524f59f2402103fd21ed2" 540 | integrity sha512-qKcDXmuZadJqdTm6vlCqioEbyewF60b/0LOFCcYN1B6BIZGlYJumWWOYs70SFYLDAH4YqdE1cxH/RKMG7rFxgA== 541 | 542 | fastify@^3.28.0: 543 | version "3.28.0" 544 | resolved "https://registry.yarnpkg.com/fastify/-/fastify-3.28.0.tgz#14d939a2f176b82af1094de7abcb0b2d83bcff8f" 545 | integrity sha512-LAQtGllpkRe8L6Tpf3zdbvXzXFOrgaWV3Tbvp3xMv9ngcr9zht9U2/mo5zq9qp9kplSiBJ0w43aVAMqv6PBMbw== 546 | dependencies: 547 | "@fastify/ajv-compiler" "^1.0.0" 548 | abstract-logging "^2.0.0" 549 | avvio "^7.1.2" 550 | fast-json-stringify "^2.5.2" 551 | fastify-error "^0.3.0" 552 | find-my-way "^4.5.0" 553 | flatstr "^1.0.12" 554 | light-my-request "^4.2.0" 555 | pino "^6.13.0" 556 | process-warning "^1.0.0" 557 | proxy-addr "^2.0.7" 558 | rfdc "^1.1.4" 559 | secure-json-parse "^2.0.0" 560 | semver "^7.3.2" 561 | tiny-lru "^8.0.1" 562 | 563 | fastq@^1.6.1: 564 | version "1.13.0" 565 | resolved "https://registry.yarnpkg.com/fastq/-/fastq-1.13.0.tgz#616760f88a7526bdfc596b7cab8c18938c36b98c" 566 | integrity sha512-YpkpUnK8od0o1hmeSc7UUs/eB/vIPWJYjKck2QKIzAf71Vm1AAQ3EbuZB3g2JIy+pg+ERD0vqI79KyZiB2e2Nw== 567 | dependencies: 568 | reusify "^1.0.4" 569 | 570 | fetch-blob@^3.1.2, fetch-blob@^3.1.4: 571 | version "3.1.5" 572 | resolved "https://registry.yarnpkg.com/fetch-blob/-/fetch-blob-3.1.5.tgz#0077bf5f3fcdbd9d75a0b5362f77dbb743489863" 573 | integrity sha512-N64ZpKqoLejlrwkIAnb9iLSA3Vx/kjgzpcDhygcqJ2KKjky8nCgUQ+dzXtbrLaWZGZNmNfQTsiQ0weZ1svglHg== 574 | dependencies: 575 | node-domexception "^1.0.0" 576 | web-streams-polyfill "^3.0.3" 577 | 578 | fill-range@^7.0.1: 579 | version "7.0.1" 580 | resolved "https://registry.yarnpkg.com/fill-range/-/fill-range-7.0.1.tgz#1919a6a7c75fe38b2c7c77e5198535da9acdda40" 581 | integrity sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ== 582 | dependencies: 583 | to-regex-range "^5.0.1" 584 | 585 | find-my-way@^4.5.0: 586 | version "4.5.1" 587 | resolved "https://registry.yarnpkg.com/find-my-way/-/find-my-way-4.5.1.tgz#758e959194b90aea0270db18fff75e2fceb2239f" 588 | integrity sha512-kE0u7sGoUFbMXcOG/xpkmz4sRLCklERnBcg7Ftuu1iAxsfEt2S46RLJ3Sq7vshsEy2wJT2hZxE58XZK27qa8kg== 589 | dependencies: 590 | fast-decode-uri-component "^1.0.1" 591 | fast-deep-equal "^3.1.3" 592 | safe-regex2 "^2.0.0" 593 | semver-store "^0.3.0" 594 | 595 | flatstr@^1.0.12: 596 | version "1.0.12" 597 | resolved "https://registry.yarnpkg.com/flatstr/-/flatstr-1.0.12.tgz#c2ba6a08173edbb6c9640e3055b95e287ceb5931" 598 | integrity sha512-4zPxDyhCyiN2wIAtSLI6gc82/EjqZc1onI4Mz/l0pWrAlsSfYH/2ZIcU+e3oA2wDwbzIWNKwa23F8rh6+DRWkw== 599 | 600 | formdata-polyfill@^4.0.10: 601 | version "4.0.10" 602 | resolved "https://registry.yarnpkg.com/formdata-polyfill/-/formdata-polyfill-4.0.10.tgz#24807c31c9d402e002ab3d8c720144ceb8848423" 603 | integrity sha512-buewHzMvYL29jdeQTVILecSaZKnt/RJWjoZCF5OW60Z67/GmSLBkOFM7qh1PI3zFNtJbaZL5eQu1vLfazOwj4g== 604 | dependencies: 605 | fetch-blob "^3.1.2" 606 | 607 | forwarded@0.2.0: 608 | version "0.2.0" 609 | resolved "https://registry.yarnpkg.com/forwarded/-/forwarded-0.2.0.tgz#2269936428aad4c15c7ebe9779a84bf0b2a81811" 610 | integrity sha512-buRG0fpBtRHSTCOASe6hD258tEubFoRLb4ZNA6NxMVHNw2gOcwHo9wyablzMzOA5z9xA9L1KNjk/Nt6MT9aYow== 611 | 612 | fsevents@~2.3.2: 613 | version "2.3.2" 614 | resolved "https://registry.yarnpkg.com/fsevents/-/fsevents-2.3.2.tgz#8a526f78b8fdf4623b709e0b975c52c24c02fd1a" 615 | integrity sha512-xiqMQR4xAeHTuB9uWm+fFRcIOgKBMiOBP+eXiyT7jsgVCq1bkVygt00oASowB7EdtpOHaaPgKt812P9ab+DDKA== 616 | 617 | get-stream@^4.1.0: 618 | version "4.1.0" 619 | resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-4.1.0.tgz#c1b255575f3dc21d59bfc79cd3d2b46b1c3a54b5" 620 | integrity sha512-GMat4EJ5161kIy2HevLlr4luNjBgvmj413KaQA7jt4V8B4RDsfpHk7WQ9GVqfYyyx8OS/L66Kox+rJRNklLK7w== 621 | dependencies: 622 | pump "^3.0.0" 623 | 624 | get-stream@^5.1.0: 625 | version "5.2.0" 626 | resolved "https://registry.yarnpkg.com/get-stream/-/get-stream-5.2.0.tgz#4966a1795ee5ace65e706c4b7beb71257d6e22d3" 627 | integrity sha512-nBF+F1rAZVCu/p7rjzgA+Yb4lfYXrpl7a6VmJrU8wF9I1CKvP/QwPNZHnOlwbTkY6dvtFIzFMSyQXbLoTQPRpA== 628 | dependencies: 629 | pump "^3.0.0" 630 | 631 | glob-parent@~5.1.2: 632 | version "5.1.2" 633 | resolved "https://registry.yarnpkg.com/glob-parent/-/glob-parent-5.1.2.tgz#869832c58034fe68a4093c17dc15e8340d8401c4" 634 | integrity sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow== 635 | dependencies: 636 | is-glob "^4.0.1" 637 | 638 | global-dirs@^3.0.0: 639 | version "3.0.0" 640 | resolved "https://registry.yarnpkg.com/global-dirs/-/global-dirs-3.0.0.tgz#70a76fe84ea315ab37b1f5576cbde7d48ef72686" 641 | integrity sha512-v8ho2DS5RiCjftj1nD9NmnfaOzTdud7RRnVd9kFNOjqZbISlx5DQ+OrTkywgd0dIt7oFCvKetZSHoHcP3sDdiA== 642 | dependencies: 643 | ini "2.0.0" 644 | 645 | got@^9.6.0: 646 | version "9.6.0" 647 | resolved "https://registry.yarnpkg.com/got/-/got-9.6.0.tgz#edf45e7d67f99545705de1f7bbeeeb121765ed85" 648 | integrity sha512-R7eWptXuGYxwijs0eV+v3o6+XH1IqVK8dJOEecQfTmkncw9AV4dcw/Dhxi8MdlqPthxxpZyizMzyg8RTmEsG+Q== 649 | dependencies: 650 | "@sindresorhus/is" "^0.14.0" 651 | "@szmarczak/http-timer" "^1.1.2" 652 | cacheable-request "^6.0.0" 653 | decompress-response "^3.3.0" 654 | duplexer3 "^0.1.4" 655 | get-stream "^4.1.0" 656 | lowercase-keys "^1.0.1" 657 | mimic-response "^1.0.1" 658 | p-cancelable "^1.0.0" 659 | to-readable-stream "^1.0.0" 660 | url-parse-lax "^3.0.0" 661 | 662 | graceful-fs@^4.1.2: 663 | version "4.2.10" 664 | resolved "https://registry.yarnpkg.com/graceful-fs/-/graceful-fs-4.2.10.tgz#147d3a006da4ca3ce14728c7aefc287c367d7a6c" 665 | integrity sha512-9ByhssR2fPVsNZj478qUUbKfmL0+t5BDVyjShtyZZLiK7ZDAArFFfopyOTj0M05wE2tJPisA4iTnnXl2YoPvOA== 666 | 667 | has-flag@^3.0.0: 668 | version "3.0.0" 669 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-3.0.0.tgz#b5d454dc2199ae225699f3467e5a07f3b955bafd" 670 | integrity sha1-tdRU3CGZriJWmfNGfloH87lVuv0= 671 | 672 | has-flag@^4.0.0: 673 | version "4.0.0" 674 | resolved "https://registry.yarnpkg.com/has-flag/-/has-flag-4.0.0.tgz#944771fd9c81c81265c4d6941860da06bb59479b" 675 | integrity sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ== 676 | 677 | has-yarn@^2.1.0: 678 | version "2.1.0" 679 | resolved "https://registry.yarnpkg.com/has-yarn/-/has-yarn-2.1.0.tgz#137e11354a7b5bf11aa5cb649cf0c6f3ff2b2e77" 680 | integrity sha512-UqBRqi4ju7T+TqGNdqAO0PaSVGsDGJUBQvk9eUWNGRY1CFGDzYhLWoM7JQEemnlvVcv/YEmc2wNW8BC24EnUsw== 681 | 682 | http-cache-semantics@^4.0.0: 683 | version "4.1.0" 684 | resolved "https://registry.yarnpkg.com/http-cache-semantics/-/http-cache-semantics-4.1.0.tgz#49e91c5cbf36c9b94bcfcd71c23d5249ec74e390" 685 | integrity sha512-carPklcUh7ROWRK7Cv27RPtdhYhUsela/ue5/jKzjegVvXDqM2ILE9Q2BGn9JZJh1g87cp56su/FgQSzcWS8cQ== 686 | 687 | ignore-by-default@^1.0.1: 688 | version "1.0.1" 689 | resolved "https://registry.yarnpkg.com/ignore-by-default/-/ignore-by-default-1.0.1.tgz#48ca6d72f6c6a3af00a9ad4ae6876be3889e2b09" 690 | integrity sha1-SMptcvbGo68Aqa1K5odr44ieKwk= 691 | 692 | import-lazy@^2.1.0: 693 | version "2.1.0" 694 | resolved "https://registry.yarnpkg.com/import-lazy/-/import-lazy-2.1.0.tgz#05698e3d45c88e8d7e9d92cb0584e77f096f3e43" 695 | integrity sha1-BWmOPUXIjo1+nZLLBYTnfwlvPkM= 696 | 697 | imurmurhash@^0.1.4: 698 | version "0.1.4" 699 | resolved "https://registry.yarnpkg.com/imurmurhash/-/imurmurhash-0.1.4.tgz#9218b9b2b928a238b13dc4fb6b6d576f231453ea" 700 | integrity sha1-khi5srkoojixPcT7a21XbyMUU+o= 701 | 702 | inherits@^2.0.3: 703 | version "2.0.4" 704 | resolved "https://registry.yarnpkg.com/inherits/-/inherits-2.0.4.tgz#0fa2c64f932917c3433a0ded55363aae37416b7c" 705 | integrity sha512-k/vGaX4/Yla3WzyMCvTQOXYeIHvqOKtnqBduzTHpzpQZzAskKMhZ2K+EnBiSM9zGSoIFeMpXKxa4dYeZIQqewQ== 706 | 707 | ini@2.0.0: 708 | version "2.0.0" 709 | resolved "https://registry.yarnpkg.com/ini/-/ini-2.0.0.tgz#e5fd556ecdd5726be978fa1001862eacb0a94bc5" 710 | integrity sha512-7PnF4oN3CvZF23ADhA5wRaYEQpJ8qygSkbtTXWBeXWXmEVRXK+1ITciHWwHhsjv1TmW0MgacIv6hEi5pX5NQdA== 711 | 712 | ini@~1.3.0: 713 | version "1.3.8" 714 | resolved "https://registry.yarnpkg.com/ini/-/ini-1.3.8.tgz#a29da425b48806f34767a4efce397269af28432c" 715 | integrity sha512-JV/yugV2uzW5iMRSiZAyDtQd+nxtUnjeLt0acNdw98kKLrvuRVyB80tsREOE7yvGVgalhZ6RNXCmEHkUKBKxew== 716 | 717 | ipaddr.js@1.9.1: 718 | version "1.9.1" 719 | resolved "https://registry.yarnpkg.com/ipaddr.js/-/ipaddr.js-1.9.1.tgz#bff38543eeb8984825079ff3a2a8e6cbd46781b3" 720 | integrity sha512-0KI/607xoxSToH7GjN1FfSbLoU0+btTicjsQSWQlh/hZykN8KpmMf7uYwPW3R+akZ6R/w18ZlXSHBYXiYUPO3g== 721 | 722 | is-binary-path@~2.1.0: 723 | version "2.1.0" 724 | resolved "https://registry.yarnpkg.com/is-binary-path/-/is-binary-path-2.1.0.tgz#ea1f7f3b80f064236e83470f86c09c254fb45b09" 725 | integrity sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw== 726 | dependencies: 727 | binary-extensions "^2.0.0" 728 | 729 | is-ci@^2.0.0: 730 | version "2.0.0" 731 | resolved "https://registry.yarnpkg.com/is-ci/-/is-ci-2.0.0.tgz#6bc6334181810e04b5c22b3d589fdca55026404c" 732 | integrity sha512-YfJT7rkpQB0updsdHLGWrvhBJfcfzNNawYDNIyQXJz0IViGf75O8EBPKSdvw2rF+LGCsX4FZ8tcr3b19LcZq4w== 733 | dependencies: 734 | ci-info "^2.0.0" 735 | 736 | is-extglob@^2.1.1: 737 | version "2.1.1" 738 | resolved "https://registry.yarnpkg.com/is-extglob/-/is-extglob-2.1.1.tgz#a88c02535791f02ed37c76a1b9ea9773c833f8c2" 739 | integrity sha1-qIwCU1eR8C7TfHahueqXc8gz+MI= 740 | 741 | is-fullwidth-code-point@^3.0.0: 742 | version "3.0.0" 743 | resolved "https://registry.yarnpkg.com/is-fullwidth-code-point/-/is-fullwidth-code-point-3.0.0.tgz#f116f8064fe90b3f7844a38997c0b75051269f1d" 744 | integrity sha512-zymm5+u+sCsSWyD9qNaejV3DFvhCKclKdizYaJUuHA83RLjb7nSuGnddCHGv0hk+KY7BMAlsWeK4Ueg6EV6XQg== 745 | 746 | is-glob@^4.0.1, is-glob@~4.0.1: 747 | version "4.0.3" 748 | resolved "https://registry.yarnpkg.com/is-glob/-/is-glob-4.0.3.tgz#64f61e42cbbb2eec2071a9dac0b28ba1e65d5084" 749 | integrity sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg== 750 | dependencies: 751 | is-extglob "^2.1.1" 752 | 753 | is-installed-globally@^0.4.0: 754 | version "0.4.0" 755 | resolved "https://registry.yarnpkg.com/is-installed-globally/-/is-installed-globally-0.4.0.tgz#9a0fd407949c30f86eb6959ef1b7994ed0b7b520" 756 | integrity sha512-iwGqO3J21aaSkC7jWnHP/difazwS7SFeIqxv6wEtLU8Y5KlzFTjyqcSIT0d8s4+dDhKytsk9PJZ2BkS5eZwQRQ== 757 | dependencies: 758 | global-dirs "^3.0.0" 759 | is-path-inside "^3.0.2" 760 | 761 | is-npm@^5.0.0: 762 | version "5.0.0" 763 | resolved "https://registry.yarnpkg.com/is-npm/-/is-npm-5.0.0.tgz#43e8d65cc56e1b67f8d47262cf667099193f45a8" 764 | integrity sha512-WW/rQLOazUq+ST/bCAVBp/2oMERWLsR7OrKyt052dNDk4DHcDE0/7QSXITlmi+VBcV13DfIbysG3tZJm5RfdBA== 765 | 766 | is-number@^7.0.0: 767 | version "7.0.0" 768 | resolved "https://registry.yarnpkg.com/is-number/-/is-number-7.0.0.tgz#7535345b896734d5f80c4d06c50955527a14f12b" 769 | integrity sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng== 770 | 771 | is-obj@^2.0.0: 772 | version "2.0.0" 773 | resolved "https://registry.yarnpkg.com/is-obj/-/is-obj-2.0.0.tgz#473fb05d973705e3fd9620545018ca8e22ef4982" 774 | integrity sha512-drqDG3cbczxxEJRoOXcOjtdp1J/lyp1mNn0xaznRs8+muBhgQcrnbspox5X5fOw0HnMnbfDzvnEMEtqDEJEo8w== 775 | 776 | is-path-inside@^3.0.2: 777 | version "3.0.3" 778 | resolved "https://registry.yarnpkg.com/is-path-inside/-/is-path-inside-3.0.3.tgz#d231362e53a07ff2b0e0ea7fed049161ffd16283" 779 | integrity sha512-Fd4gABb+ycGAmKou8eMftCupSir5lRxqf4aD/vd0cD2qc4HL07OjCeuHMr8Ro4CoMaeCKDB0/ECBOVWjTwUvPQ== 780 | 781 | is-typedarray@^1.0.0: 782 | version "1.0.0" 783 | resolved "https://registry.yarnpkg.com/is-typedarray/-/is-typedarray-1.0.0.tgz#e479c80858df0c1b11ddda6940f96011fcda4a9a" 784 | integrity sha1-5HnICFjfDBsR3dppQPlgEfzaSpo= 785 | 786 | is-yarn-global@^0.3.0: 787 | version "0.3.0" 788 | resolved "https://registry.yarnpkg.com/is-yarn-global/-/is-yarn-global-0.3.0.tgz#d502d3382590ea3004893746754c89139973e232" 789 | integrity sha512-VjSeb/lHmkoyd8ryPVIKvOCn4D1koMqY+vqyjjUfc3xyKtP4dYOxM44sZrnqQSzSds3xyOrUTLTC9LVCVgLngw== 790 | 791 | joycon@^3.1.1: 792 | version "3.1.1" 793 | resolved "https://registry.yarnpkg.com/joycon/-/joycon-3.1.1.tgz#bce8596d6ae808f8b68168f5fc69280996894f03" 794 | integrity sha512-34wB/Y7MW7bzjKRjUKTa46I2Z7eV62Rkhva+KkopW7Qvv/OSWBqvkSY7vusOPrNuZcUG3tApvdVgNB8POj3SPw== 795 | 796 | json-buffer@3.0.0: 797 | version "3.0.0" 798 | resolved "https://registry.yarnpkg.com/json-buffer/-/json-buffer-3.0.0.tgz#5b1f397afc75d677bde8bcfc0e47e1f9a3d9a898" 799 | integrity sha1-Wx85evx11ne96Lz8Dkfh+aPZqJg= 800 | 801 | json-schema-traverse@^0.4.1: 802 | version "0.4.1" 803 | resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-0.4.1.tgz#69f6a87d9513ab8bb8fe63bdb0979c448e684660" 804 | integrity sha512-xbbCH5dCYU5T8LcEhhuh7HJ88HXuW3qsI3Y0zOZFKfZEHcpWiHU/Jxzk629Brsab/mMiHQti9wMP+845RPe3Vg== 805 | 806 | json-schema-traverse@^1.0.0: 807 | version "1.0.0" 808 | resolved "https://registry.yarnpkg.com/json-schema-traverse/-/json-schema-traverse-1.0.0.tgz#ae7bcb3656ab77a73ba5c49bf654f38e6b6860e2" 809 | integrity sha512-NM8/P9n3XjXhIZn1lLhkFaACTOURQXjWhV4BA/RnOv8xvgqtqpAX9IO4mRQxSx1Rlo4tqzeqb0sOlruaOy3dug== 810 | 811 | keyv@^3.0.0: 812 | version "3.1.0" 813 | resolved "https://registry.yarnpkg.com/keyv/-/keyv-3.1.0.tgz#ecc228486f69991e49e9476485a5be1e8fc5c4d9" 814 | integrity sha512-9ykJ/46SN/9KPM/sichzQ7OvXyGDYKGTaDlKMGCAlg2UK8KRy4jb0d8sFc+0Tt0YYnThq8X2RZgCg74RPxgcVA== 815 | dependencies: 816 | json-buffer "3.0.0" 817 | 818 | latest-version@^5.1.0: 819 | version "5.1.0" 820 | resolved "https://registry.yarnpkg.com/latest-version/-/latest-version-5.1.0.tgz#119dfe908fe38d15dfa43ecd13fa12ec8832face" 821 | integrity sha512-weT+r0kTkRQdCdYCNtkMwWXQTMEswKrFBkm4ckQOMVhhqhIMI1UT2hMj+1iigIhgSZm5gTmrRXBNoGUgaTY1xA== 822 | dependencies: 823 | package-json "^6.3.0" 824 | 825 | leven@2.1.0: 826 | version "2.1.0" 827 | resolved "https://registry.yarnpkg.com/leven/-/leven-2.1.0.tgz#c2e7a9f772094dee9d34202ae8acce4687875580" 828 | integrity sha1-wuep93IJTe6dNCAq6KzORoeHVYA= 829 | 830 | light-my-request@^4.2.0: 831 | version "4.9.0" 832 | resolved "https://registry.yarnpkg.com/light-my-request/-/light-my-request-4.9.0.tgz#83559c7ce7e503466113e36f40a1d596a1886626" 833 | integrity sha512-b1U3z4OVPoO/KanT14NRkXMr9rRtXAiq0ORqNrqhDyb5bGkZjAdEc6GRN1GWCfgaLBG+aq73qkCLDNeB3c2sLw== 834 | dependencies: 835 | ajv "^8.1.0" 836 | cookie "^0.4.0" 837 | process-warning "^1.0.0" 838 | set-cookie-parser "^2.4.1" 839 | 840 | lowercase-keys@^1.0.0, lowercase-keys@^1.0.1: 841 | version "1.0.1" 842 | resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-1.0.1.tgz#6f9e30b47084d971a7c820ff15a6c5167b74c26f" 843 | integrity sha512-G2Lj61tXDnVFFOi8VZds+SoQjtQC3dgokKdDG2mTm1tx4m50NUHBOZSBwQQHyy0V12A0JTG4icfZQH+xPyh8VA== 844 | 845 | lowercase-keys@^2.0.0: 846 | version "2.0.0" 847 | resolved "https://registry.yarnpkg.com/lowercase-keys/-/lowercase-keys-2.0.0.tgz#2603e78b7b4b0006cbca2fbcc8a3202558ac9479" 848 | integrity sha512-tqNXrS78oMOE73NMxK4EMLQsQowWf8jKooH9g7xPavRT706R6bkQJ6DY2Te7QukaZsulxa30wQ7bk0pm4XiHmA== 849 | 850 | lru-cache@^6.0.0: 851 | version "6.0.0" 852 | resolved "https://registry.yarnpkg.com/lru-cache/-/lru-cache-6.0.0.tgz#6d6fe6570ebd96aaf90fcad1dafa3b2566db3a94" 853 | integrity sha512-Jo6dJ04CmSjuznwJSS3pUeWmd/H0ffTlkXXgwZi+eq1UCmqQwCh+eLsYOYCwY991i2Fah4h1BEMCx4qThGbsiA== 854 | dependencies: 855 | yallist "^4.0.0" 856 | 857 | make-dir@^3.0.0: 858 | version "3.1.0" 859 | resolved "https://registry.yarnpkg.com/make-dir/-/make-dir-3.1.0.tgz#415e967046b3a7f1d185277d84aa58203726a13f" 860 | integrity sha512-g3FeP20LNwhALb/6Cz6Dd4F2ngze0jz7tbzrD2wAV+o9FeNHe4rL+yK2md0J/fiSf1sa1ADhXqi5+oVwOM/eGw== 861 | dependencies: 862 | semver "^6.0.0" 863 | 864 | make-error@^1.1.1: 865 | version "1.3.6" 866 | resolved "https://registry.yarnpkg.com/make-error/-/make-error-1.3.6.tgz#2eb2e37ea9b67c4891f684a1394799af484cf7a2" 867 | integrity sha512-s8UhlNe7vPKomQhC1qFelMokr/Sc3AgNbso3n74mVPA5LTZwkB9NlXf4XPamLxJE8h0gh73rM94xvwRT2CVInw== 868 | 869 | mimic-response@^1.0.0, mimic-response@^1.0.1: 870 | version "1.0.1" 871 | resolved "https://registry.yarnpkg.com/mimic-response/-/mimic-response-1.0.1.tgz#4923538878eef42063cb8a3e3b0798781487ab1b" 872 | integrity sha512-j5EctnkH7amfV/q5Hgmoal1g2QHFJRraOtmx0JpIqkxhBhI/lJSl1nMpQ45hVarwNETOoWEimndZ4QK0RHxuxQ== 873 | 874 | minimatch@^3.0.4: 875 | version "3.1.2" 876 | resolved "https://registry.yarnpkg.com/minimatch/-/minimatch-3.1.2.tgz#19cd194bfd3e428f049a70817c038d89ab4be35b" 877 | integrity sha512-J7p63hRiAjw1NDEww1W7i37+ByIrOWO5XQQAzZ3VOcL0PNybwpfmV/N05zFAzwQ9USyEcX6t3UO+K5aqBQOIHw== 878 | dependencies: 879 | brace-expansion "^1.1.7" 880 | 881 | minimist@^1.2.0: 882 | version "1.2.6" 883 | resolved "https://registry.yarnpkg.com/minimist/-/minimist-1.2.6.tgz#8637a5b759ea0d6e98702cfb3a9283323c93af44" 884 | integrity sha512-Jsjnk4bw3YJqYzbdyBiNsPWHPfO++UGG749Cxs6peCu5Xg4nrena6OVxOYxrQTqww0Jmwt+Ref8rggumkTLz9Q== 885 | 886 | moment-timezone@^0.5.x: 887 | version "0.5.34" 888 | resolved "https://registry.yarnpkg.com/moment-timezone/-/moment-timezone-0.5.34.tgz#a75938f7476b88f155d3504a9343f7519d9a405c" 889 | integrity sha512-3zAEHh2hKUs3EXLESx/wsgw6IQdusOT8Bxm3D9UrHPQR7zlMmzwybC8zHEM1tQ4LJwP7fcxrWr8tuBg05fFCbg== 890 | dependencies: 891 | moment ">= 2.9.0" 892 | 893 | "moment@>= 2.9.0", moment@>=2.14.0: 894 | version "2.29.3" 895 | resolved "https://registry.yarnpkg.com/moment/-/moment-2.29.3.tgz#edd47411c322413999f7a5940d526de183c031f3" 896 | integrity sha512-c6YRvhEo//6T2Jz/vVtYzqBzwvPT95JBQ+smCytzf7c50oMZRsR/a4w88aD34I+/QVSfnoAnSBFPJHItlOMJVw== 897 | 898 | mri@1.1.4: 899 | version "1.1.4" 900 | resolved "https://registry.yarnpkg.com/mri/-/mri-1.1.4.tgz#7cb1dd1b9b40905f1fac053abe25b6720f44744a" 901 | integrity sha512-6y7IjGPm8AzlvoUrwAaw1tLnUBudaS3752vcd8JtrpGGQn+rXIe63LFVHm/YMwtqAuh+LJPCFdlLYPWM1nYn6w== 902 | 903 | ms@2.1.2: 904 | version "2.1.2" 905 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.2.tgz#d09d1f357b443f493382a8eb3ccd183872ae6009" 906 | integrity sha512-sGkPx+VjMtmA6MX27oA4FBFELFCZZ4S4XqeGOXCv68tT+jb3vk/RyaKWP0PTKyWtmLSM0b+adUTEvbs1PEaH2w== 907 | 908 | ms@^2.1.1: 909 | version "2.1.3" 910 | resolved "https://registry.yarnpkg.com/ms/-/ms-2.1.3.tgz#574c8138ce1d2b5861f0b44579dbadd60c6615b2" 911 | integrity sha512-6FlzubTLZG3J2a/NVCAleEhjzq5oxgHyaCU9yYXvcLsvoVaHJq/s5xXI6/XXP6tz7R9xAOtHnSO/tXtF3WRTlA== 912 | 913 | node-domexception@^1.0.0: 914 | version "1.0.0" 915 | resolved "https://registry.yarnpkg.com/node-domexception/-/node-domexception-1.0.0.tgz#6888db46a1f71c0b76b3f7555016b63fe64766e5" 916 | integrity sha512-/jKZoMpw0F8GRwl4/eLROPA3cfcXtLApP0QzLmUT/HuPCZWyB7IY9ZrMeKw2O/nFIqPQB3PVM9aYm0F312AXDQ== 917 | 918 | node-fetch@^3.2.3: 919 | version "3.2.3" 920 | resolved "https://registry.yarnpkg.com/node-fetch/-/node-fetch-3.2.3.tgz#a03c9cc2044d21d1a021566bd52f080f333719a6" 921 | integrity sha512-AXP18u4pidSZ1xYXRDPY/8jdv3RAozIt/WLNR/MBGZAz+xjtlr90RvCnsvHQRiXyWliZF/CpytExp32UU67/SA== 922 | dependencies: 923 | data-uri-to-buffer "^4.0.0" 924 | fetch-blob "^3.1.4" 925 | formdata-polyfill "^4.0.10" 926 | 927 | nodemon@^2.0.15: 928 | version "2.0.15" 929 | resolved "https://registry.yarnpkg.com/nodemon/-/nodemon-2.0.15.tgz#504516ce3b43d9dc9a955ccd9ec57550a31a8d4e" 930 | integrity sha512-gdHMNx47Gw7b3kWxJV64NI+Q5nfl0y5DgDbiVtShiwa7Z0IZ07Ll4RLFo6AjrhzMtoEZn5PDE3/c2AbVsiCkpA== 931 | dependencies: 932 | chokidar "^3.5.2" 933 | debug "^3.2.7" 934 | ignore-by-default "^1.0.1" 935 | minimatch "^3.0.4" 936 | pstree.remy "^1.1.8" 937 | semver "^5.7.1" 938 | supports-color "^5.5.0" 939 | touch "^3.1.0" 940 | undefsafe "^2.0.5" 941 | update-notifier "^5.1.0" 942 | 943 | nopt@~1.0.10: 944 | version "1.0.10" 945 | resolved "https://registry.yarnpkg.com/nopt/-/nopt-1.0.10.tgz#6ddd21bd2a31417b92727dd585f8a6f37608ebee" 946 | integrity sha1-bd0hvSoxQXuScn3Vhfim83YI6+4= 947 | dependencies: 948 | abbrev "1" 949 | 950 | normalize-path@^3.0.0, normalize-path@~3.0.0: 951 | version "3.0.0" 952 | resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65" 953 | integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA== 954 | 955 | normalize-url@^4.1.0: 956 | version "4.5.1" 957 | resolved "https://registry.yarnpkg.com/normalize-url/-/normalize-url-4.5.1.tgz#0dd90cf1288ee1d1313b87081c9a5932ee48518a" 958 | integrity sha512-9UZCFRHQdNrfTpGg8+1INIg93B6zE0aXMVFkw1WFwvO4SlZywU6aLg5Of0Ap/PgcbSw4LNxvMWXMeugwMCX0AA== 959 | 960 | on-exit-leak-free@^0.2.0: 961 | version "0.2.0" 962 | resolved "https://registry.yarnpkg.com/on-exit-leak-free/-/on-exit-leak-free-0.2.0.tgz#b39c9e3bf7690d890f4861558b0d7b90a442d209" 963 | integrity sha512-dqaz3u44QbRXQooZLTUKU41ZrzYrcvLISVgbrzbyCMxpmSLJvZ3ZamIJIZ29P6OhZIkNIQKosdeM6t1LYbA9hg== 964 | 965 | once@^1.3.1, once@^1.4.0: 966 | version "1.4.0" 967 | resolved "https://registry.yarnpkg.com/once/-/once-1.4.0.tgz#583b1aa775961d4b113ac17d9c50baef9dd76bd1" 968 | integrity sha1-WDsap3WWHUsROsF9nFC6753Xa9E= 969 | dependencies: 970 | wrappy "1" 971 | 972 | p-cancelable@^1.0.0: 973 | version "1.1.0" 974 | resolved "https://registry.yarnpkg.com/p-cancelable/-/p-cancelable-1.1.0.tgz#d078d15a3af409220c886f1d9a0ca2e441ab26cc" 975 | integrity sha512-s73XxOZ4zpt1edZYZzvhqFa6uvQc1vwUa0K0BdtIZgQMAJj9IbebH+JkgKZc9h+B05PKHLOTl4ajG1BmNrVZlw== 976 | 977 | package-json@^6.3.0: 978 | version "6.5.0" 979 | resolved "https://registry.yarnpkg.com/package-json/-/package-json-6.5.0.tgz#6feedaca35e75725876d0b0e64974697fed145b0" 980 | integrity sha512-k3bdm2n25tkyxcjSKzB5x8kfVxlMdgsbPr0GkZcwHsLpba6cBjqCt1KlcChKEvxHIcTB1FVMuwoijZ26xex5MQ== 981 | dependencies: 982 | got "^9.6.0" 983 | registry-auth-token "^4.0.0" 984 | registry-url "^5.0.0" 985 | semver "^6.2.0" 986 | 987 | picomatch@^2.0.4, picomatch@^2.2.1: 988 | version "2.3.1" 989 | resolved "https://registry.yarnpkg.com/picomatch/-/picomatch-2.3.1.tgz#3ba3833733646d9d3e4995946c1365a67fb07a42" 990 | integrity sha512-JU3teHTNjmE2VCGFzuY8EXzCDVwEqB2a8fsIvwaStHhAWJEeVd1o1QD80CU6+ZdEXXSLbSsuLwJjkCBWqRQUVA== 991 | 992 | pino-abstract-transport@^0.5.0: 993 | version "0.5.0" 994 | resolved "https://registry.yarnpkg.com/pino-abstract-transport/-/pino-abstract-transport-0.5.0.tgz#4b54348d8f73713bfd14e3dc44228739aa13d9c0" 995 | integrity sha512-+KAgmVeqXYbTtU2FScx1XS3kNyfZ5TrXY07V96QnUSFqo2gAqlvmaxH67Lj7SWazqsMabf+58ctdTcBgnOLUOQ== 996 | dependencies: 997 | duplexify "^4.1.2" 998 | split2 "^4.0.0" 999 | 1000 | pino-pretty@^7.6.1: 1001 | version "7.6.1" 1002 | resolved "https://registry.yarnpkg.com/pino-pretty/-/pino-pretty-7.6.1.tgz#42d20611050ad80d619edaf132c6d81d40f81d98" 1003 | integrity sha512-H7N6ZYkiyrfwBGW9CSjx0uyO9Q2Lyt73881+OTYk8v3TiTdgN92QHrWlEq/LeWw5XtDP64jeSk3mnc6T+xX9/w== 1004 | dependencies: 1005 | args "^5.0.1" 1006 | colorette "^2.0.7" 1007 | dateformat "^4.6.3" 1008 | fast-safe-stringify "^2.0.7" 1009 | joycon "^3.1.1" 1010 | on-exit-leak-free "^0.2.0" 1011 | pino-abstract-transport "^0.5.0" 1012 | pump "^3.0.0" 1013 | readable-stream "^3.6.0" 1014 | rfdc "^1.3.0" 1015 | secure-json-parse "^2.4.0" 1016 | sonic-boom "^2.2.0" 1017 | strip-json-comments "^3.1.1" 1018 | 1019 | pino-std-serializers@^3.1.0: 1020 | version "3.2.0" 1021 | resolved "https://registry.yarnpkg.com/pino-std-serializers/-/pino-std-serializers-3.2.0.tgz#b56487c402d882eb96cd67c257868016b61ad671" 1022 | integrity sha512-EqX4pwDPrt3MuOAAUBMU0Tk5kR/YcCM5fNPEzgCO2zJ5HfX0vbiH9HbJglnyeQsN96Kznae6MWD47pZB5avTrg== 1023 | 1024 | pino@^6.13.0: 1025 | version "6.14.0" 1026 | resolved "https://registry.yarnpkg.com/pino/-/pino-6.14.0.tgz#b745ea87a99a6c4c9b374e4f29ca7910d4c69f78" 1027 | integrity sha512-iuhEDel3Z3hF9Jfe44DPXR8l07bhjuFY3GMHIXbjnY9XcafbyDDwl2sN2vw2GjMPf5Nkoe+OFao7ffn9SXaKDg== 1028 | dependencies: 1029 | fast-redact "^3.0.0" 1030 | fast-safe-stringify "^2.0.8" 1031 | flatstr "^1.0.12" 1032 | pino-std-serializers "^3.1.0" 1033 | process-warning "^1.0.0" 1034 | quick-format-unescaped "^4.0.3" 1035 | sonic-boom "^1.0.2" 1036 | 1037 | prepend-http@^2.0.0: 1038 | version "2.0.0" 1039 | resolved "https://registry.yarnpkg.com/prepend-http/-/prepend-http-2.0.0.tgz#e92434bfa5ea8c19f41cdfd401d741a3c819d897" 1040 | integrity sha1-6SQ0v6XqjBn0HN/UAddBo8gZ2Jc= 1041 | 1042 | prettier@^2.6.2: 1043 | version "2.6.2" 1044 | resolved "https://registry.yarnpkg.com/prettier/-/prettier-2.6.2.tgz#e26d71a18a74c3d0f0597f55f01fb6c06c206032" 1045 | integrity sha512-PkUpF+qoXTqhOeWL9fu7As8LXsIUZ1WYaJiY/a7McAQzxjk82OF0tibkFXVCDImZtWxbvojFjerkiLb0/q8mew== 1046 | 1047 | prisma@^3.13.0: 1048 | version "3.13.0" 1049 | resolved "https://registry.yarnpkg.com/prisma/-/prisma-3.13.0.tgz#b11edd5631222ff1bf1d5324732d47801386aa8c" 1050 | integrity sha512-oO1auBnBtieGdiN+57IgsA9Vr7Sy4HkILi1KSaUG4mpKfEbnkTGnLOxAqjLed+K2nsG/GtE1tJBtB7JxN1a78Q== 1051 | dependencies: 1052 | "@prisma/engines" "3.13.0-17.efdf9b1183dddfd4258cd181a72125755215ab7b" 1053 | ts-pattern "^4.0.1" 1054 | 1055 | process-warning@^1.0.0: 1056 | version "1.0.0" 1057 | resolved "https://registry.yarnpkg.com/process-warning/-/process-warning-1.0.0.tgz#980a0b25dc38cd6034181be4b7726d89066b4616" 1058 | integrity sha512-du4wfLyj4yCZq1VupnVSZmRsPJsNuxoDQFdCFHLaYiEbFBD7QE0a+I4D7hOxrVnh78QE/YipFAj9lXHiXocV+Q== 1059 | 1060 | proxy-addr@^2.0.7: 1061 | version "2.0.7" 1062 | resolved "https://registry.yarnpkg.com/proxy-addr/-/proxy-addr-2.0.7.tgz#f19fe69ceab311eeb94b42e70e8c2070f9ba1025" 1063 | integrity sha512-llQsMLSUDUPT44jdrU/O37qlnifitDP+ZwrmmZcoSKyLKvtZxpyV0n2/bD/N4tBAAZ/gJEdZU7KMraoK1+XYAg== 1064 | dependencies: 1065 | forwarded "0.2.0" 1066 | ipaddr.js "1.9.1" 1067 | 1068 | pstree.remy@^1.1.8: 1069 | version "1.1.8" 1070 | resolved "https://registry.yarnpkg.com/pstree.remy/-/pstree.remy-1.1.8.tgz#c242224f4a67c21f686839bbdb4ac282b8373d3a" 1071 | integrity sha512-77DZwxQmxKnu3aR542U+X8FypNzbfJ+C5XQDk3uWjWxn6151aIMGthWYRXTqT1E5oJvg+ljaa2OJi+VfvCOQ8w== 1072 | 1073 | pump@^3.0.0: 1074 | version "3.0.0" 1075 | resolved "https://registry.yarnpkg.com/pump/-/pump-3.0.0.tgz#b4a2116815bde2f4e1ea602354e8c75565107a64" 1076 | integrity sha512-LwZy+p3SFs1Pytd/jYct4wpv49HiYCqd9Rlc5ZVdk0V+8Yzv6jR5Blk3TRmPL1ft69TxP0IMZGJ+WPFU2BFhww== 1077 | dependencies: 1078 | end-of-stream "^1.1.0" 1079 | once "^1.3.1" 1080 | 1081 | punycode@^2.1.0: 1082 | version "2.1.1" 1083 | resolved "https://registry.yarnpkg.com/punycode/-/punycode-2.1.1.tgz#b58b010ac40c22c5657616c8d2c2c02c7bf479ec" 1084 | integrity sha512-XRsRjdf+j5ml+y/6GKHPZbrF/8p2Yga0JPtdqTIY2Xe5ohJPD9saDJJLPvp9+NSBprVvevdXZybnj2cv8OEd0A== 1085 | 1086 | pupa@^2.1.1: 1087 | version "2.1.1" 1088 | resolved "https://registry.yarnpkg.com/pupa/-/pupa-2.1.1.tgz#f5e8fd4afc2c5d97828faa523549ed8744a20d62" 1089 | integrity sha512-l1jNAspIBSFqbT+y+5FosojNpVpF94nlI+wDUpqP9enwOTfHx9f0gh5nB96vl+6yTpsJsypeNrwfzPrKuHB41A== 1090 | dependencies: 1091 | escape-goat "^2.0.0" 1092 | 1093 | queue-microtask@^1.1.2: 1094 | version "1.2.3" 1095 | resolved "https://registry.yarnpkg.com/queue-microtask/-/queue-microtask-1.2.3.tgz#4929228bbc724dfac43e0efb058caf7b6cfb6243" 1096 | integrity sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A== 1097 | 1098 | quick-format-unescaped@^4.0.3: 1099 | version "4.0.4" 1100 | resolved "https://registry.yarnpkg.com/quick-format-unescaped/-/quick-format-unescaped-4.0.4.tgz#93ef6dd8d3453cbc7970dd614fad4c5954d6b5a7" 1101 | integrity sha512-tYC1Q1hgyRuHgloV/YXs2w15unPVh8qfu/qCTfhTYamaw7fyhumKa2yGpdSo87vY32rIclj+4fWYQXUMs9EHvg== 1102 | 1103 | rc@^1.2.8: 1104 | version "1.2.8" 1105 | resolved "https://registry.yarnpkg.com/rc/-/rc-1.2.8.tgz#cd924bf5200a075b83c188cd6b9e211b7fc0d3ed" 1106 | integrity sha512-y3bGgqKj3QBdxLbLkomlohkvsA8gdAiUQlSBJnBhfn+BPxg4bc62d8TcBW15wavDfgexCgccckhcZvywyQYPOw== 1107 | dependencies: 1108 | deep-extend "^0.6.0" 1109 | ini "~1.3.0" 1110 | minimist "^1.2.0" 1111 | strip-json-comments "~2.0.1" 1112 | 1113 | readable-stream@^3.1.1, readable-stream@^3.6.0: 1114 | version "3.6.0" 1115 | resolved "https://registry.yarnpkg.com/readable-stream/-/readable-stream-3.6.0.tgz#337bbda3adc0706bd3e024426a286d4b4b2c9198" 1116 | integrity sha512-BViHy7LKeTz4oNnkcLJ+lVSL6vpiFeX6/d3oSH8zCW7UxP2onchk+vTGB143xuFjHS3deTgkKoXXymXqymiIdA== 1117 | dependencies: 1118 | inherits "^2.0.3" 1119 | string_decoder "^1.1.1" 1120 | util-deprecate "^1.0.1" 1121 | 1122 | readdirp@~3.6.0: 1123 | version "3.6.0" 1124 | resolved "https://registry.yarnpkg.com/readdirp/-/readdirp-3.6.0.tgz#74a370bd857116e245b29cc97340cd431a02a6c7" 1125 | integrity sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA== 1126 | dependencies: 1127 | picomatch "^2.2.1" 1128 | 1129 | registry-auth-token@^4.0.0: 1130 | version "4.2.1" 1131 | resolved "https://registry.yarnpkg.com/registry-auth-token/-/registry-auth-token-4.2.1.tgz#6d7b4006441918972ccd5fedcd41dc322c79b250" 1132 | integrity sha512-6gkSb4U6aWJB4SF2ZvLb76yCBjcvufXBqvvEx1HbmKPkutswjW1xNVRY0+daljIYRbogN7O0etYSlbiaEQyMyw== 1133 | dependencies: 1134 | rc "^1.2.8" 1135 | 1136 | registry-url@^5.0.0: 1137 | version "5.1.0" 1138 | resolved "https://registry.yarnpkg.com/registry-url/-/registry-url-5.1.0.tgz#e98334b50d5434b81136b44ec638d9c2009c5009" 1139 | integrity sha512-8acYXXTI0AkQv6RAOjE3vOaIXZkT9wo4LOFbBKYQEEnnMNBpKqdUrI6S4NT0KPIo/WVvJ5tE/X5LF/TQUf0ekw== 1140 | dependencies: 1141 | rc "^1.2.8" 1142 | 1143 | require-from-string@^2.0.2: 1144 | version "2.0.2" 1145 | resolved "https://registry.yarnpkg.com/require-from-string/-/require-from-string-2.0.2.tgz#89a7fdd938261267318eafe14f9c32e598c36909" 1146 | integrity sha512-Xf0nWe6RseziFMu+Ap9biiUbmplq6S9/p+7w7YXP/JBHhrUDDUhwa+vANyubuqfZWTveU//DYVGsDG7RKL/vEw== 1147 | 1148 | responselike@^1.0.2: 1149 | version "1.0.2" 1150 | resolved "https://registry.yarnpkg.com/responselike/-/responselike-1.0.2.tgz#918720ef3b631c5642be068f15ade5a46f4ba1e7" 1151 | integrity sha1-kYcg7ztjHFZCvgaPFa3lpG9Loec= 1152 | dependencies: 1153 | lowercase-keys "^1.0.0" 1154 | 1155 | ret@~0.2.0: 1156 | version "0.2.2" 1157 | resolved "https://registry.yarnpkg.com/ret/-/ret-0.2.2.tgz#b6861782a1f4762dce43402a71eb7a283f44573c" 1158 | integrity sha512-M0b3YWQs7R3Z917WRQy1HHA7Ba7D8hvZg6UE5mLykJxQVE2ju0IXbGlaHPPlkY+WN7wFP+wUMXmBFA0aV6vYGQ== 1159 | 1160 | reusify@^1.0.4: 1161 | version "1.0.4" 1162 | resolved "https://registry.yarnpkg.com/reusify/-/reusify-1.0.4.tgz#90da382b1e126efc02146e90845a88db12925d76" 1163 | integrity sha512-U9nH88a3fc/ekCF1l0/UP1IosiuIjyTh7hBvXVMHYgVcfGvt897Xguj2UOLDeI5BG2m7/uwyaLVT6fbtCwTyzw== 1164 | 1165 | rfdc@^1.1.4, rfdc@^1.2.0, rfdc@^1.3.0: 1166 | version "1.3.0" 1167 | resolved "https://registry.yarnpkg.com/rfdc/-/rfdc-1.3.0.tgz#d0b7c441ab2720d05dc4cf26e01c89631d9da08b" 1168 | integrity sha512-V2hovdzFbOi77/WajaSMXk2OLm+xNIeQdMMuB7icj7bk6zi2F8GGAxigcnDFpJHbNyNcgyJDiP+8nOrY5cZGrA== 1169 | 1170 | safe-buffer@~5.2.0: 1171 | version "5.2.1" 1172 | resolved "https://registry.yarnpkg.com/safe-buffer/-/safe-buffer-5.2.1.tgz#1eaf9fa9bdb1fdd4ec75f58f9cdb4e6b7827eec6" 1173 | integrity sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ== 1174 | 1175 | safe-regex2@^2.0.0: 1176 | version "2.0.0" 1177 | resolved "https://registry.yarnpkg.com/safe-regex2/-/safe-regex2-2.0.0.tgz#b287524c397c7a2994470367e0185e1916b1f5b9" 1178 | integrity sha512-PaUSFsUaNNuKwkBijoAPHAK6/eM6VirvyPWlZ7BAQy4D+hCvh4B6lIG+nPdhbFfIbP+gTGBcrdsOaUs0F+ZBOQ== 1179 | dependencies: 1180 | ret "~0.2.0" 1181 | 1182 | secure-json-parse@^2.0.0, secure-json-parse@^2.4.0: 1183 | version "2.4.0" 1184 | resolved "https://registry.yarnpkg.com/secure-json-parse/-/secure-json-parse-2.4.0.tgz#5aaeaaef85c7a417f76271a4f5b0cc3315ddca85" 1185 | integrity sha512-Q5Z/97nbON5t/L/sH6mY2EacfjVGwrCcSi5D3btRO2GZ8pf1K1UN7Z9H5J57hjVU2Qzxr1xO+FmBhOvEkzCMmg== 1186 | 1187 | semver-diff@^3.1.1: 1188 | version "3.1.1" 1189 | resolved "https://registry.yarnpkg.com/semver-diff/-/semver-diff-3.1.1.tgz#05f77ce59f325e00e2706afd67bb506ddb1ca32b" 1190 | integrity sha512-GX0Ix/CJcHyB8c4ykpHGIAvLyOwOobtM/8d+TQkAd81/bEjgPHrfba41Vpesr7jX/t8Uh+R3EX9eAS5be+jQYg== 1191 | dependencies: 1192 | semver "^6.3.0" 1193 | 1194 | semver-store@^0.3.0: 1195 | version "0.3.0" 1196 | resolved "https://registry.yarnpkg.com/semver-store/-/semver-store-0.3.0.tgz#ce602ff07df37080ec9f4fb40b29576547befbe9" 1197 | integrity sha512-TcZvGMMy9vodEFSse30lWinkj+JgOBvPn8wRItpQRSayhc+4ssDs335uklkfvQQJgL/WvmHLVj4Ycv2s7QCQMg== 1198 | 1199 | semver@^5.7.1: 1200 | version "5.7.1" 1201 | resolved "https://registry.yarnpkg.com/semver/-/semver-5.7.1.tgz#a954f931aeba508d307bbf069eff0c01c96116f7" 1202 | integrity sha512-sauaDf/PZdVgrLTNYHRtpXa1iRiKcaebiKQ1BJdpQlWH2lCvexQdX55snPFyK7QzpudqbCI0qXFfOasHdyNDGQ== 1203 | 1204 | semver@^6.0.0, semver@^6.2.0, semver@^6.3.0: 1205 | version "6.3.0" 1206 | resolved "https://registry.yarnpkg.com/semver/-/semver-6.3.0.tgz#ee0a64c8af5e8ceea67687b133761e1becbd1d3d" 1207 | integrity sha512-b39TBaTSfV6yBrapU89p5fKekE2m/NwnDocOVruQFS1/veMgdzuPcnOM34M6CwxW8jH/lxEa5rBoDeUwu5HHTw== 1208 | 1209 | semver@^7.3.2, semver@^7.3.4: 1210 | version "7.3.7" 1211 | resolved "https://registry.yarnpkg.com/semver/-/semver-7.3.7.tgz#12c5b649afdbf9049707796e22a4028814ce523f" 1212 | integrity sha512-QlYTucUYOews+WeEujDoEGziz4K6c47V/Bd+LjSSYcA94p+DmINdf7ncaUinThfvZyu13lN9OY1XDxt8C0Tw0g== 1213 | dependencies: 1214 | lru-cache "^6.0.0" 1215 | 1216 | set-cookie-parser@^2.4.1: 1217 | version "2.4.8" 1218 | resolved "https://registry.yarnpkg.com/set-cookie-parser/-/set-cookie-parser-2.4.8.tgz#d0da0ed388bc8f24e706a391f9c9e252a13c58b2" 1219 | integrity sha512-edRH8mBKEWNVIVMKejNnuJxleqYE/ZSdcT8/Nem9/mmosx12pctd80s2Oy00KNZzrogMZS5mauK2/ymL1bvlvg== 1220 | 1221 | signal-exit@^3.0.2: 1222 | version "3.0.7" 1223 | resolved "https://registry.yarnpkg.com/signal-exit/-/signal-exit-3.0.7.tgz#a9a1767f8af84155114eaabd73f99273c8f59ad9" 1224 | integrity sha512-wnD2ZE+l+SPC/uoS0vXeE9L1+0wuaMqKlfz9AMUo38JsyLSBWSFcHR1Rri62LZc12vLr1gb3jl7iwQhgwpAbGQ== 1225 | 1226 | sonic-boom@^1.0.2: 1227 | version "1.4.1" 1228 | resolved "https://registry.yarnpkg.com/sonic-boom/-/sonic-boom-1.4.1.tgz#d35d6a74076624f12e6f917ade7b9d75e918f53e" 1229 | integrity sha512-LRHh/A8tpW7ru89lrlkU4AszXt1dbwSjVWguGrmlxE7tawVmDBlI1PILMkXAxJTwqhgsEeTHzj36D5CmHgQmNg== 1230 | dependencies: 1231 | atomic-sleep "^1.0.0" 1232 | flatstr "^1.0.12" 1233 | 1234 | sonic-boom@^2.2.0: 1235 | version "2.7.0" 1236 | resolved "https://registry.yarnpkg.com/sonic-boom/-/sonic-boom-2.7.0.tgz#55c9390cc123ae4e42c044e822eb7e7246b01c9c" 1237 | integrity sha512-Ynxp0OGQG91wvDjCbFlRMHbSUmDq7dE/EgDeUJ/j+Q9x1FVkFry20cjLykxRSmlm3QS0B4JYAKE8239XKN4SHQ== 1238 | dependencies: 1239 | atomic-sleep "^1.0.0" 1240 | 1241 | split2@^4.0.0: 1242 | version "4.1.0" 1243 | resolved "https://registry.yarnpkg.com/split2/-/split2-4.1.0.tgz#101907a24370f85bb782f08adaabe4e281ecf809" 1244 | integrity sha512-VBiJxFkxiXRlUIeyMQi8s4hgvKCSjtknJv/LVYbrgALPwf5zSKmEwV9Lst25AkvMDnvxODugjdl6KZgwKM1WYQ== 1245 | 1246 | stream-shift@^1.0.0: 1247 | version "1.0.1" 1248 | resolved "https://registry.yarnpkg.com/stream-shift/-/stream-shift-1.0.1.tgz#d7088281559ab2778424279b0877da3c392d5a3d" 1249 | integrity sha512-AiisoFqQ0vbGcZgQPY1cdP2I76glaVA/RauYR4G4thNFgkTqr90yXTo4LYX60Jl+sIlPNHHdGSwo01AvbKUSVQ== 1250 | 1251 | string-similarity@^4.0.1: 1252 | version "4.0.4" 1253 | resolved "https://registry.yarnpkg.com/string-similarity/-/string-similarity-4.0.4.tgz#42d01ab0b34660ea8a018da8f56a3309bb8b2a5b" 1254 | integrity sha512-/q/8Q4Bl4ZKAPjj8WerIBJWALKkaPRfrvhfF8k/B23i4nzrlRj2/go1m90In7nG/3XDSbOo0+pu6RvCTM9RGMQ== 1255 | 1256 | string-width@^4.0.0, string-width@^4.1.0, string-width@^4.2.2: 1257 | version "4.2.3" 1258 | resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" 1259 | integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== 1260 | dependencies: 1261 | emoji-regex "^8.0.0" 1262 | is-fullwidth-code-point "^3.0.0" 1263 | strip-ansi "^6.0.1" 1264 | 1265 | string_decoder@^1.1.1: 1266 | version "1.3.0" 1267 | resolved "https://registry.yarnpkg.com/string_decoder/-/string_decoder-1.3.0.tgz#42f114594a46cf1a8e30b0a84f56c78c3edac21e" 1268 | integrity sha512-hkRX8U1WjJFd8LsDJ2yQ/wWWxaopEsABU1XfkM8A+j0+85JAGppt16cr1Whg6KIbb4okU6Mql6BOj+uup/wKeA== 1269 | dependencies: 1270 | safe-buffer "~5.2.0" 1271 | 1272 | strip-ansi@^6.0.0, strip-ansi@^6.0.1: 1273 | version "6.0.1" 1274 | resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" 1275 | integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== 1276 | dependencies: 1277 | ansi-regex "^5.0.1" 1278 | 1279 | strip-json-comments@^3.1.1: 1280 | version "3.1.1" 1281 | resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-3.1.1.tgz#31f1281b3832630434831c310c01cccda8cbe006" 1282 | integrity sha512-6fPc+R4ihwqP6N/aIv2f1gMH8lOVtWQHoqC4yK6oSDVVocumAsfCqjkXnqiYMhmMwS/mEHLp7Vehlt3ql6lEig== 1283 | 1284 | strip-json-comments@~2.0.1: 1285 | version "2.0.1" 1286 | resolved "https://registry.yarnpkg.com/strip-json-comments/-/strip-json-comments-2.0.1.tgz#3c531942e908c2697c0ec344858c286c7ca0a60a" 1287 | integrity sha1-PFMZQukIwml8DsNEhYwobHygpgo= 1288 | 1289 | supports-color@^5.3.0, supports-color@^5.5.0: 1290 | version "5.5.0" 1291 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-5.5.0.tgz#e2e69a44ac8772f78a1ec0b35b689df6530efc8f" 1292 | integrity sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow== 1293 | dependencies: 1294 | has-flag "^3.0.0" 1295 | 1296 | supports-color@^7.1.0: 1297 | version "7.2.0" 1298 | resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" 1299 | integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== 1300 | dependencies: 1301 | has-flag "^4.0.0" 1302 | 1303 | tiny-lru@^8.0.1: 1304 | version "8.0.2" 1305 | resolved "https://registry.yarnpkg.com/tiny-lru/-/tiny-lru-8.0.2.tgz#812fccbe6e622ded552e3ff8a4c3b5ff34a85e4c" 1306 | integrity sha512-ApGvZ6vVvTNdsmt676grvCkUCGwzG9IqXma5Z07xJgiC5L7akUMof5U8G2JTI9Rz/ovtVhJBlY6mNhEvtjzOIg== 1307 | 1308 | to-readable-stream@^1.0.0: 1309 | version "1.0.0" 1310 | resolved "https://registry.yarnpkg.com/to-readable-stream/-/to-readable-stream-1.0.0.tgz#ce0aa0c2f3df6adf852efb404a783e77c0475771" 1311 | integrity sha512-Iq25XBt6zD5npPhlLVXGFN3/gyR2/qODcKNNyTMd4vbm39HUaOiAM4PMq0eMVC/Tkxz+Zjdsc55g9yyz+Yq00Q== 1312 | 1313 | to-regex-range@^5.0.1: 1314 | version "5.0.1" 1315 | resolved "https://registry.yarnpkg.com/to-regex-range/-/to-regex-range-5.0.1.tgz#1648c44aae7c8d988a326018ed72f5b4dd0392e4" 1316 | integrity sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ== 1317 | dependencies: 1318 | is-number "^7.0.0" 1319 | 1320 | touch@^3.1.0: 1321 | version "3.1.0" 1322 | resolved "https://registry.yarnpkg.com/touch/-/touch-3.1.0.tgz#fe365f5f75ec9ed4e56825e0bb76d24ab74af83b" 1323 | integrity sha512-WBx8Uy5TLtOSRtIq+M03/sKDrXCLHxwDcquSP2c43Le03/9serjQBIztjRz6FkJez9D/hleyAXTBGLwwZUw9lA== 1324 | dependencies: 1325 | nopt "~1.0.10" 1326 | 1327 | ts-node@^10.7.0: 1328 | version "10.7.0" 1329 | resolved "https://registry.yarnpkg.com/ts-node/-/ts-node-10.7.0.tgz#35d503d0fab3e2baa672a0e94f4b40653c2463f5" 1330 | integrity sha512-TbIGS4xgJoX2i3do417KSaep1uRAW/Lu+WAL2doDHC0D6ummjirVOXU5/7aiZotbQ5p1Zp9tP7U6cYhA0O7M8A== 1331 | dependencies: 1332 | "@cspotcode/source-map-support" "0.7.0" 1333 | "@tsconfig/node10" "^1.0.7" 1334 | "@tsconfig/node12" "^1.0.7" 1335 | "@tsconfig/node14" "^1.0.0" 1336 | "@tsconfig/node16" "^1.0.2" 1337 | acorn "^8.4.1" 1338 | acorn-walk "^8.1.1" 1339 | arg "^4.1.0" 1340 | create-require "^1.1.0" 1341 | diff "^4.0.1" 1342 | make-error "^1.1.1" 1343 | v8-compile-cache-lib "^3.0.0" 1344 | yn "3.1.1" 1345 | 1346 | ts-pattern@^4.0.1: 1347 | version "4.0.2" 1348 | resolved "https://registry.yarnpkg.com/ts-pattern/-/ts-pattern-4.0.2.tgz#b36afdb2de1ec0224539dcb7cea3a57c41453b9f" 1349 | integrity sha512-eHqR/7A6fcw05vCOfnL6RwgGJbVi9G/YHTdYdjYmElhDdJ1SMn7pWs+6+YuxygaFwQS/g+cIDlu+UD8IVpur1A== 1350 | 1351 | type-fest@^0.20.2: 1352 | version "0.20.2" 1353 | resolved "https://registry.yarnpkg.com/type-fest/-/type-fest-0.20.2.tgz#1bf207f4b28f91583666cb5fbd327887301cd5f4" 1354 | integrity sha512-Ne+eE4r0/iWnpAxD852z3A+N0Bt5RN//NjJwRd2VFHEmrywxf5vsZlh4R6lixl6B+wz/8d+maTSAkN1FIkI3LQ== 1355 | 1356 | typedarray-to-buffer@^3.1.5: 1357 | version "3.1.5" 1358 | resolved "https://registry.yarnpkg.com/typedarray-to-buffer/-/typedarray-to-buffer-3.1.5.tgz#a97ee7a9ff42691b9f783ff1bc5112fe3fca9080" 1359 | integrity sha512-zdu8XMNEDepKKR+XYOXAVPtWui0ly0NtohUscw+UmaHiAWT8hrV1rr//H6V+0DvJ3OQ19S979M0laLfX8rm82Q== 1360 | dependencies: 1361 | is-typedarray "^1.0.0" 1362 | 1363 | typescript@^4.6.3: 1364 | version "4.6.3" 1365 | resolved "https://registry.yarnpkg.com/typescript/-/typescript-4.6.3.tgz#eefeafa6afdd31d725584c67a0eaba80f6fc6c6c" 1366 | integrity sha512-yNIatDa5iaofVozS/uQJEl3JRWLKKGJKh6Yaiv0GLGSuhpFJe7P3SbHZ8/yjAHRQwKRoA6YZqlfjXWmVzoVSMw== 1367 | 1368 | undefsafe@^2.0.5: 1369 | version "2.0.5" 1370 | resolved "https://registry.yarnpkg.com/undefsafe/-/undefsafe-2.0.5.tgz#38733b9327bdcd226db889fb723a6efd162e6e2c" 1371 | integrity sha512-WxONCrssBM8TSPRqN5EmsjVrsv4A8X12J4ArBiiayv3DyyG3ZlIg6yysuuSYdZsVz3TKcTg2fd//Ujd4CHV1iA== 1372 | 1373 | unique-string@^2.0.0: 1374 | version "2.0.0" 1375 | resolved "https://registry.yarnpkg.com/unique-string/-/unique-string-2.0.0.tgz#39c6451f81afb2749de2b233e3f7c5e8843bd89d" 1376 | integrity sha512-uNaeirEPvpZWSgzwsPGtU2zVSTrn/8L5q/IexZmH0eH6SA73CmAA5U4GwORTxQAZs95TAXLNqeLoPPNO5gZfWg== 1377 | dependencies: 1378 | crypto-random-string "^2.0.0" 1379 | 1380 | update-notifier@^5.1.0: 1381 | version "5.1.0" 1382 | resolved "https://registry.yarnpkg.com/update-notifier/-/update-notifier-5.1.0.tgz#4ab0d7c7f36a231dd7316cf7729313f0214d9ad9" 1383 | integrity sha512-ItnICHbeMh9GqUy31hFPrD1kcuZ3rpxDZbf4KUDavXwS0bW5m7SLbDQpGX3UYr072cbrF5hFUs3r5tUsPwjfHw== 1384 | dependencies: 1385 | boxen "^5.0.0" 1386 | chalk "^4.1.0" 1387 | configstore "^5.0.1" 1388 | has-yarn "^2.1.0" 1389 | import-lazy "^2.1.0" 1390 | is-ci "^2.0.0" 1391 | is-installed-globally "^0.4.0" 1392 | is-npm "^5.0.0" 1393 | is-yarn-global "^0.3.0" 1394 | latest-version "^5.1.0" 1395 | pupa "^2.1.1" 1396 | semver "^7.3.4" 1397 | semver-diff "^3.1.1" 1398 | xdg-basedir "^4.0.0" 1399 | 1400 | uri-js@^4.2.2: 1401 | version "4.4.1" 1402 | resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" 1403 | integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== 1404 | dependencies: 1405 | punycode "^2.1.0" 1406 | 1407 | url-parse-lax@^3.0.0: 1408 | version "3.0.0" 1409 | resolved "https://registry.yarnpkg.com/url-parse-lax/-/url-parse-lax-3.0.0.tgz#16b5cafc07dbe3676c1b1999177823d6503acb0c" 1410 | integrity sha1-FrXK/Afb42dsGxmZF3gj1lA6yww= 1411 | dependencies: 1412 | prepend-http "^2.0.0" 1413 | 1414 | util-deprecate@^1.0.1: 1415 | version "1.0.2" 1416 | resolved "https://registry.yarnpkg.com/util-deprecate/-/util-deprecate-1.0.2.tgz#450d4dc9fa70de732762fbd2d4a28981419a0ccf" 1417 | integrity sha1-RQ1Nyfpw3nMnYvvS1KKJgUGaDM8= 1418 | 1419 | v8-compile-cache-lib@^3.0.0: 1420 | version "3.0.1" 1421 | resolved "https://registry.yarnpkg.com/v8-compile-cache-lib/-/v8-compile-cache-lib-3.0.1.tgz#6336e8d71965cb3d35a1bbb7868445a7c05264bf" 1422 | integrity sha512-wa7YjyUGfNZngI/vtK0UHAN+lgDCxBPCylVXGp0zu59Fz5aiGtNXaq3DhIov063MorB+VfufLh3JlF2KdTK3xg== 1423 | 1424 | vary@^1.1.2: 1425 | version "1.1.2" 1426 | resolved "https://registry.yarnpkg.com/vary/-/vary-1.1.2.tgz#2299f02c6ded30d4a5961b0b9f74524a18f634fc" 1427 | integrity sha1-IpnwLG3tMNSllhsLn3RSShj2NPw= 1428 | 1429 | web-streams-polyfill@^3.0.3: 1430 | version "3.2.1" 1431 | resolved "https://registry.yarnpkg.com/web-streams-polyfill/-/web-streams-polyfill-3.2.1.tgz#71c2718c52b45fd49dbeee88634b3a60ceab42a6" 1432 | integrity sha512-e0MO3wdXWKrLbL0DgGnUV7WHVuw9OUvL4hjgnPkIeEvESk74gAITi5G606JtZPp39cd8HA9VQzCIvA49LpPN5Q== 1433 | 1434 | widest-line@^3.1.0: 1435 | version "3.1.0" 1436 | resolved "https://registry.yarnpkg.com/widest-line/-/widest-line-3.1.0.tgz#8292333bbf66cb45ff0de1603b136b7ae1496eca" 1437 | integrity sha512-NsmoXalsWVDMGupxZ5R08ka9flZjjiLvHVAWYOKtiKM8ujtZWr9cRffak+uSE48+Ob8ObalXpwyeUiyDD6QFgg== 1438 | dependencies: 1439 | string-width "^4.0.0" 1440 | 1441 | wrap-ansi@^7.0.0: 1442 | version "7.0.0" 1443 | resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" 1444 | integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== 1445 | dependencies: 1446 | ansi-styles "^4.0.0" 1447 | string-width "^4.1.0" 1448 | strip-ansi "^6.0.0" 1449 | 1450 | wrappy@1: 1451 | version "1.0.2" 1452 | resolved "https://registry.yarnpkg.com/wrappy/-/wrappy-1.0.2.tgz#b5243d8f3ec1aa35f1364605bc0d1036e30ab69f" 1453 | integrity sha1-tSQ9jz7BqjXxNkYFvA0QNuMKtp8= 1454 | 1455 | write-file-atomic@^3.0.0: 1456 | version "3.0.3" 1457 | resolved "https://registry.yarnpkg.com/write-file-atomic/-/write-file-atomic-3.0.3.tgz#56bd5c5a5c70481cd19c571bd39ab965a5de56e8" 1458 | integrity sha512-AvHcyZ5JnSfq3ioSyjrBkH9yW4m7Ayk8/9My/DD9onKeu/94fwrMocemO2QAJFAlnnDN+ZDS+ZjAR5ua1/PV/Q== 1459 | dependencies: 1460 | imurmurhash "^0.1.4" 1461 | is-typedarray "^1.0.0" 1462 | signal-exit "^3.0.2" 1463 | typedarray-to-buffer "^3.1.5" 1464 | 1465 | xdg-basedir@^4.0.0: 1466 | version "4.0.0" 1467 | resolved "https://registry.yarnpkg.com/xdg-basedir/-/xdg-basedir-4.0.0.tgz#4bc8d9984403696225ef83a1573cbbcb4e79db13" 1468 | integrity sha512-PSNhEJDejZYV7h50BohL09Er9VaIefr2LMAf3OEmpCkjOi34eYyQYAXUTjEQtZJTKcF0E2UKTh+osDLsgNim9Q== 1469 | 1470 | yallist@^4.0.0: 1471 | version "4.0.0" 1472 | resolved "https://registry.yarnpkg.com/yallist/-/yallist-4.0.0.tgz#9bb92790d9c0effec63be73519e11a35019a3a72" 1473 | integrity sha512-3wdGidZyq5PB084XLES5TpOSRA3wjXAlIWMhum2kRcv/41Sn2emQ0dycQW4uZXLejwKvg6EsvbdlVL+FYEct7A== 1474 | 1475 | yn@3.1.1: 1476 | version "3.1.1" 1477 | resolved "https://registry.yarnpkg.com/yn/-/yn-3.1.1.tgz#1e87401a09d767c1d5eab26a6e4c185182d2eb50" 1478 | integrity sha512-Ux4ygGWsu2c7isFWe8Yu1YluJmqVhxqK2cLXNQA5AcC3QfbGNpM7fu0Y8b/z16pXLnFxZYvWhd3fhBY9DLmC6Q== 1479 | --------------------------------------------------------------------------------