forked from cerc-io/snowballtools-base
Compare commits
6 Commits
main
...
sushan/T-4
Author | SHA1 | Date | |
---|---|---|---|
|
c57bf61b37 | ||
|
6b948d47e8 | ||
|
ae1e3455e4 | ||
|
951fd9a3f3 | ||
|
97fea83582 | ||
|
47cd8079e4 |
@ -1,29 +0,0 @@
|
|||||||
name: Lint
|
|
||||||
|
|
||||||
on:
|
|
||||||
pull_request:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- main
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
lint:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
node-version: [20.x]
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v2
|
|
||||||
- name: Download yarn
|
|
||||||
run: |
|
|
||||||
curl -fsSL -o /usr/local/bin/yarn https://github.com/yarnpkg/yarn/releases/download/v1.22.21/yarn-1.22.21.js
|
|
||||||
chmod +x /usr/local/bin/yarn
|
|
||||||
- name: Use Node.js ${{ matrix.node-version }}
|
|
||||||
uses: actions/setup-node@v3
|
|
||||||
with:
|
|
||||||
node-version: ${{ matrix.node-version }}
|
|
||||||
- run: yarn
|
|
||||||
- name: Build libs
|
|
||||||
run: yarn workspace gql-client run build
|
|
||||||
- name: Linter check
|
|
||||||
run: yarn lint
|
|
2
.github/workflows/lint.yaml
vendored
2
.github/workflows/lint.yaml
vendored
@ -19,7 +19,5 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
node-version: ${{ matrix.node-version }}
|
node-version: ${{ matrix.node-version }}
|
||||||
- run: yarn
|
- run: yarn
|
||||||
- name: Build libs
|
|
||||||
run: yarn workspace gql-client run build
|
|
||||||
- name: Linter check
|
- name: Linter check
|
||||||
run: yarn lint
|
run: yarn lint
|
||||||
|
39
.github/workflows/test-app-deployment.yaml
vendored
39
.github/workflows/test-app-deployment.yaml
vendored
@ -1,39 +0,0 @@
|
|||||||
name: Test webapp deployment
|
|
||||||
|
|
||||||
on:
|
|
||||||
schedule:
|
|
||||||
- cron: '0 3 * * *'
|
|
||||||
workflow_dispatch:
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
test_app_deployment:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
strategy:
|
|
||||||
matrix:
|
|
||||||
node-version: [20.x]
|
|
||||||
steps:
|
|
||||||
- uses: actions/checkout@v3
|
|
||||||
- name: Use Node.js ${{ matrix.node-version }}
|
|
||||||
uses: actions/setup-node@v3
|
|
||||||
with:
|
|
||||||
node-version: ${{ matrix.node-version }}
|
|
||||||
- name: Install dependencies
|
|
||||||
run: yarn
|
|
||||||
- name: Test webapp deployment
|
|
||||||
run: ./packages/deployer/test/test-webapp-deployment-undeployment.sh
|
|
||||||
- name: Notify Vulcanize Slack on CI failure
|
|
||||||
if: ${{ always() && github.ref_name == 'main' }}
|
|
||||||
uses: ravsamhq/notify-slack-action@v2
|
|
||||||
with:
|
|
||||||
status: ${{ job.status }}
|
|
||||||
notify_when: 'failure'
|
|
||||||
env:
|
|
||||||
SLACK_WEBHOOK_URL: ${{ secrets.VULCANIZE_SLACK_CI_ALERTS_WEBHOOK }}
|
|
||||||
- name: Notify DeepStack Slack on CI failure
|
|
||||||
if: ${{ always() && github.ref_name == 'main' }}
|
|
||||||
uses: ravsamhq/notify-slack-action@v2
|
|
||||||
with:
|
|
||||||
status: ${{ job.status }}
|
|
||||||
notify_when: 'failure'
|
|
||||||
env:
|
|
||||||
SLACK_WEBHOOK_URL: ${{ secrets.DEEPSTACK_SLACK_CI_ALERTS_WEBHOOK }}
|
|
5
.gitignore
vendored
5
.gitignore
vendored
@ -5,8 +5,3 @@ yarn-error.log
|
|||||||
.yarnrc
|
.yarnrc
|
||||||
|
|
||||||
packages/backend/environments/local.toml
|
packages/backend/environments/local.toml
|
||||||
packages/backend/dev/
|
|
||||||
packages/frontend/dist/
|
|
||||||
|
|
||||||
# ignore all .DS_Store files
|
|
||||||
**/.DS_Store
|
|
||||||
|
@ -1 +0,0 @@
|
|||||||
v20.12.1
|
|
3
.vscode/settings.json
vendored
3
.vscode/settings.json
vendored
@ -1,7 +1,6 @@
|
|||||||
{
|
{
|
||||||
// IntelliSense for taiwind variants
|
// IntelliSense for taiwind variants
|
||||||
"tailwindCSS.experimental.classRegex": [
|
"tailwindCSS.experimental.classRegex": [
|
||||||
"tv\\('([^)]*)\\')",
|
["tv\\((([^()]*|\\([^()]*\\))*)\\)", "[\"'`]([^\"'`]*).*?[\"'`]"]
|
||||||
"(?:'|\"|`)([^\"'`]*)(?:'|\"|`)"
|
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
272
README.md
272
README.md
@ -1,23 +1,269 @@
|
|||||||
# snowballtools-base
|
# snowballtools-base
|
||||||
|
|
||||||
This is a [yarn workspace](https://yarnpkg.com/features/workspaces) monorepo for the dashboard.
|
## Setup
|
||||||
|
|
||||||
## Getting Started
|
- Clone the `snowballtools-base` repo
|
||||||
|
|
||||||
### Install dependencies
|
```bash
|
||||||
|
git clone git@github.com:snowball-tools/snowballtools-base.git
|
||||||
|
```
|
||||||
|
|
||||||
In the root of the project, run:
|
- In root of the repo, install depedencies
|
||||||
|
|
||||||
```zsh
|
```bash
|
||||||
yarn
|
yarn
|
||||||
```
|
```
|
||||||
|
|
||||||
### Build backend
|
- Build packages
|
||||||
|
|
||||||
```zsh
|
```bash
|
||||||
yarn build --ignore frontend
|
yarn build --ignore frontend
|
||||||
```
|
```
|
||||||
|
|
||||||
### Environment variables, running the development server, and deployment
|
## Backend
|
||||||
|
|
||||||
Follow the instructions in the README.md files of the [backend](packages/backend/README.md) and [frontend](packages/frontend/README.md) packages.
|
- Change directory to `packages/backend`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd packages/backend
|
||||||
|
```
|
||||||
|
|
||||||
|
- Rename backend config file from [environments/local.toml.example](packages/backend/environments/local.toml.example) to `local.toml`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mv environments/local.toml.example environments/local.toml
|
||||||
|
```
|
||||||
|
|
||||||
|
- Set `gitHub.oAuth.clientId` and `gitHub.oAuth.clientSecret` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
- Client ID and secret will be available after [creating an OAuth app](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/creating-an-oauth-app)
|
||||||
|
- In "Homepage URL", type `http://localhost:3000`
|
||||||
|
- In "Authorization callback URL", type `http://localhost:3000/organization/projects/create`
|
||||||
|
- Generate a new client secret after app is created
|
||||||
|
|
||||||
|
### Backend Production
|
||||||
|
|
||||||
|
- Let us assume the following domains for backend and frontend
|
||||||
|
- Backend server: `api.snowballtools.com`
|
||||||
|
- Frontend app: `dashboard.snowballtools.com`
|
||||||
|
|
||||||
|
- Update the following in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
```toml
|
||||||
|
[server]
|
||||||
|
...
|
||||||
|
[server.session]
|
||||||
|
# Secret should be changed to a different random string
|
||||||
|
secret = "p4yfpkqnddkui2iw7t6hbhwq74lbqs7sidnc382"
|
||||||
|
# Set URL of the frontend app
|
||||||
|
appOriginUrl = "https://dashboard.snowballtools.com"
|
||||||
|
# Set to true for session cookies to work behind proxy
|
||||||
|
trustProxy = true
|
||||||
|
# Set empty domain when using secure connection
|
||||||
|
domain = ""
|
||||||
|
```
|
||||||
|
|
||||||
|
- Set `gitHub.oAuth.clientId` and `gitHub.oAuth.clientSecret` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
- Client ID and secret will be available after [creating an OAuth app](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/creating-an-oauth-app)
|
||||||
|
- In "Homepage URL", type `https://dashboard.snowballtools.com`
|
||||||
|
- In "Authorization callback URL", type `https://dashboard.snowballtools.com/organization/projects/create`
|
||||||
|
- Generate a new client secret after app is created
|
||||||
|
|
||||||
|
- Set `gitHub.webhookUrl` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
```toml
|
||||||
|
...
|
||||||
|
[gitHub]
|
||||||
|
webhookUrl = "https://api.snowballtools.com"
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Let us assume domain for Laconicd to be `api.laconic.com` and set the following in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
```toml
|
||||||
|
...
|
||||||
|
[registryConfig]
|
||||||
|
fetchDeploymentRecordDelay = 5000
|
||||||
|
# Use actual port for REST endpoint
|
||||||
|
restEndpoint = "http://api.laconic.com:1317"
|
||||||
|
# Use actual port for GQL endpoint
|
||||||
|
gqlEndpoint = "http://api.laconic.com:9473/api"
|
||||||
|
# Set private key of account to be used in Laconicd
|
||||||
|
privateKey = "0wtu92cd4f1y791ezpjwgzzazni4dmd3q3mzqc3t6i6r9v06ji784tey6hwmnn69"
|
||||||
|
# Set Bond ID to be used for publishing records
|
||||||
|
bondId = "8xk8c2pb61kajwixpm223zvptr2x2ncajq0vd998p6aqhvqqep2reu6pik245epf"
|
||||||
|
chainId = "laconic_9000-1"
|
||||||
|
# Set authority that is existing in the chain
|
||||||
|
authority = "laconic"
|
||||||
|
[registryConfig.fee]
|
||||||
|
amount = "200000"
|
||||||
|
denom = "aphoton"
|
||||||
|
gas = "750000"
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Start the server in `packages/backend`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn start
|
||||||
|
```
|
||||||
|
|
||||||
|
### Backend Development
|
||||||
|
|
||||||
|
- Set `gitHub.oAuth.clientId` and `gitHub.oAuth.clientSecret` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
- Client ID and secret will be available after [creating an OAuth app](https://docs.github.com/en/apps/oauth-apps/building-oauth-apps/creating-an-oauth-app)
|
||||||
|
- In "Homepage URL", type `http://localhost:3000`
|
||||||
|
- In "Authorization callback URL", type `http://localhost:3000/organization/projects/create`
|
||||||
|
- Generate a new client secret after app is created
|
||||||
|
|
||||||
|
- Setup Laconicd
|
||||||
|
- Run the laconicd stack following this [doc](https://git.vdb.to/cerc-io/stack-orchestrator/src/branch/main/docs/laconicd-with-console.md)
|
||||||
|
- Get the private key and set `registryConfig.privateKey` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
laconic-so --stack fixturenet-laconic-loaded deploy exec laconicd "laconicd keys export mykey --unarmored-hex --unsafe"
|
||||||
|
# WARNING: The private key will be exported as an unarmored hexadecimal string. USE AT YOUR OWN RISK. Continue? [y/N]: y
|
||||||
|
# 754cca7b4b729a99d156913aea95366411d072856666e95ba09ef6c664357d81
|
||||||
|
```
|
||||||
|
|
||||||
|
- Get the REST and GQL endpoint ports of Laconicd and replace the ports for `registryConfig.restEndpoint` and `registryConfig.gqlEndpoint` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# For registryConfig.restEndpoint
|
||||||
|
laconic-so --stack fixturenet-laconic-loaded deploy port laconicd 1317
|
||||||
|
# 0.0.0.0:32777
|
||||||
|
|
||||||
|
# For registryConfig.gqlEndpoint
|
||||||
|
laconic-so --stack fixturenet-laconic-loaded deploy port laconicd 9473
|
||||||
|
# 0.0.0.0:32771
|
||||||
|
```
|
||||||
|
|
||||||
|
- Set authority in `registryConfig.authority` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
- Run the script to create bond, reserve the authority and set authority bond
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn test:registry:init
|
||||||
|
# snowball:initialize-registry bondId: 6af0ab81973b93d3511ae79841756fb5da3fd2f70ea1279e81fae7c9b19af6c4 +0ms
|
||||||
|
```
|
||||||
|
|
||||||
|
- Get the bond id and set `registryConfig.bondId` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
- Setup ngrok for GitHub webhooks
|
||||||
|
- [ngrok getting started](https://ngrok.com/docs/getting-started/)
|
||||||
|
- Start ngrok and point to backend server endpoint
|
||||||
|
|
||||||
|
```bash
|
||||||
|
ngrok http http://localhost:8000
|
||||||
|
```
|
||||||
|
|
||||||
|
- Look for the forwarding URL in ngrok
|
||||||
|
|
||||||
|
```bash
|
||||||
|
...
|
||||||
|
Forwarding https://19c1-61-95-158-116.ngrok-free.app -> http://localhost:8000
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Set `gitHub.webhookUrl` in backend [config file](packages/backend/environments/local.toml)
|
||||||
|
|
||||||
|
```toml
|
||||||
|
...
|
||||||
|
[gitHub]
|
||||||
|
webhookUrl = "https://19c1-61-95-158-116.ngrok-free.app"
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Start the server in `packages/backend`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn start:dev
|
||||||
|
```
|
||||||
|
|
||||||
|
## Frontend
|
||||||
|
|
||||||
|
- Change directory to `packages/frontend` in a new terminal
|
||||||
|
|
||||||
|
```bash
|
||||||
|
cd packages/frontend
|
||||||
|
```
|
||||||
|
|
||||||
|
- Rename [.env.example](packages/frontend/.env.example) to `.env`
|
||||||
|
|
||||||
|
```bash
|
||||||
|
mv .env.example .env
|
||||||
|
```
|
||||||
|
|
||||||
|
- Copy the GitHub OAuth app client ID from previous steps and set it in frontend [.env](packages/frontend/.env) file
|
||||||
|
|
||||||
|
```env
|
||||||
|
REACT_APP_GITHUB_CLIENT_ID = <CLIENT_ID>
|
||||||
|
```
|
||||||
|
|
||||||
|
- Set `REACT_APP_GITHUB_PWA_TEMPLATE_REPO` and `REACT_APP_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO` in [.env](packages/frontend/.env) file
|
||||||
|
|
||||||
|
```env
|
||||||
|
# Set actual owner/name of the template repo that will be used for creating new repo
|
||||||
|
REACT_APP_GITHUB_PWA_TEMPLATE_REPO = cerc-io/test-progressive-web-app
|
||||||
|
REACT_APP_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO = cerc-io/image-upload-pwa-example
|
||||||
|
```
|
||||||
|
|
||||||
|
### Frontend Production
|
||||||
|
|
||||||
|
- Let us assume the following domains for backend and frontend
|
||||||
|
- Backend server: `api.snowballtools.com`
|
||||||
|
- Frontend app: `dashboard.snowballtools.com`
|
||||||
|
|
||||||
|
- Set the following values in [.env](packages/frontend/.env) file
|
||||||
|
|
||||||
|
```env
|
||||||
|
# Backend server endpoint
|
||||||
|
REACT_APP_SERVER_URL = 'https://api.snowballtools.com'
|
||||||
|
```
|
||||||
|
|
||||||
|
- Sign in to [wallet connect](https://cloud.walletconnect.com/sign-in) to create a project ID
|
||||||
|
- Create a project and add information to use wallet connect SDK
|
||||||
|
- Add project name and select project type as `App`
|
||||||
|
- Set project home page URL to `https://dashboard.snowballtools.com`
|
||||||
|
- On creation of project, use the `Project ID` and set it in `REACT_APP_WALLET_CONNECT_ID` in [.env](packages/frontend/.env) file
|
||||||
|
|
||||||
|
```env
|
||||||
|
REACT_APP_WALLET_CONNECT_ID = <PROJECT_ID>
|
||||||
|
```
|
||||||
|
|
||||||
|
- Build the React application
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn build
|
||||||
|
```
|
||||||
|
|
||||||
|
- Use a web server for hosting static built files
|
||||||
|
|
||||||
|
```bash
|
||||||
|
python3 -m http.server -d build 3000
|
||||||
|
```
|
||||||
|
|
||||||
|
### Frontend Development
|
||||||
|
|
||||||
|
- Copy the graphQL endpoint from terminal and add the endpoint in the [.env](packages/frontend/.env) file present in `packages/frontend`
|
||||||
|
|
||||||
|
```env
|
||||||
|
REACT_APP_SERVER_URL = 'http://localhost:8000'
|
||||||
|
```
|
||||||
|
|
||||||
|
- Sign in to [wallet connect](https://cloud.walletconnect.com/sign-in) to create a project ID.
|
||||||
|
- Create a project and add information to use wallet connect SDK
|
||||||
|
- Add project name and select project type as `App`
|
||||||
|
- Project home page URL is not required to be set
|
||||||
|
- On creation of project, use the `Project ID` and set it in `REACT_APP_WALLET_CONNECT_ID` in [.env](packages/frontend/.env) file
|
||||||
|
|
||||||
|
```env
|
||||||
|
REACT_APP_WALLET_CONNECT_ID = <Project_ID>
|
||||||
|
```
|
||||||
|
|
||||||
|
- Start the React application
|
||||||
|
|
||||||
|
```bash
|
||||||
|
yarn start
|
||||||
|
```
|
||||||
|
|
||||||
|
- The React application will be running in `http://localhost:3000/`
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
PKG_DIR="./packages/frontend"
|
PKG_DIR="./packages/frontend"
|
||||||
OUTPUT_DIR="${PKG_DIR}/dist"
|
OUTPUT_DIR="${PKG_DIR}/build"
|
||||||
DEST_DIR=${1:-/data}
|
DEST_DIR=${1:-/data}
|
||||||
|
|
||||||
if [[ -d "$DEST_DIR" ]]; then
|
if [[ -d "$DEST_DIR" ]]; then
|
||||||
@ -9,22 +9,24 @@ if [[ -d "$DEST_DIR" ]]; then
|
|||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
|
if [[ -f "$PKG_DIR/.env" ]]; then
|
||||||
|
echo "Using existing .env file"
|
||||||
|
else
|
||||||
|
mv "$PKG_DIR/.env.example" "$PKG_DIR/.env"
|
||||||
|
echo "Created .env file. Please populate with the correct values."
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
cat > $PKG_DIR/.env <<EOF
|
cat > $PKG_DIR/.env <<EOF
|
||||||
VITE_SERVER_URL = 'LACONIC_HOSTED_CONFIG_server_url'
|
REACT_APP_SERVER_URL = 'LACONIC_HOSTED_CONFIG_app_server_url'
|
||||||
VITE_GITHUB_CLIENT_ID = 'LACONIC_HOSTED_CONFIG_github_clientid'
|
REACT_APP_GITHUB_CLIENT_ID = 'LACONIC_HOSTED_CONFIG_app_github_clientid'
|
||||||
VITE_GITHUB_PWA_TEMPLATE_REPO = 'LACONIC_HOSTED_CONFIG_github_pwa_templaterepo'
|
REACT_APP_GITHUB_PWA_TEMPLATE_REPO = 'LACONIC_HOSTED_CONFIG_app_github_pwa_templaterepo'
|
||||||
VITE_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO = 'LACONIC_HOSTED_CONFIG_github_image_upload_templaterepo'
|
REACT_APP_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO = 'LACONIC_HOSTED_CONFIG_app_github_image_upload_templaterepo'
|
||||||
VITE_WALLET_CONNECT_ID = 'LACONIC_HOSTED_CONFIG_wallet_connect_id'
|
REACT_APP_WALLET_CONNECT_ID = 'LACONIC_HOSTED_CONFIG_app_wallet_connect_id'
|
||||||
VITE_LACONICD_CHAIN_ID = 'LACONIC_HOSTED_CONFIG_laconicd_chain_id'
|
|
||||||
VITE_LIT_RELAY_API_KEY = 'LACONIC_HOSTED_CONFIG_lit_relay_api_key'
|
|
||||||
VITE_BUGSNAG_API_KEY = 'LACONIC_HOSTED_CONFIG_bugsnag_api_key'
|
|
||||||
VITE_PASSKEY_WALLET_RPID = 'LACONIC_HOSTED_CONFIG_passkey_wallet_rpid'
|
|
||||||
VITE_TURNKEY_API_BASE_URL = 'LACONIC_HOSTED_CONFIG_turnkey_api_base_url'
|
|
||||||
VITE_TURNKEY_ORGANIZATION_ID = 'LACONIC_HOSTED_CONFIG_turnkey_organization_id'
|
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
yarn || exit 1
|
yarn || exit 1
|
||||||
yarn build --ignore backend || exit 1
|
yarn build || exit 1
|
||||||
|
|
||||||
if [[ ! -d "$OUTPUT_DIR" ]]; then
|
if [[ ! -d "$OUTPUT_DIR" ]]; then
|
||||||
echo "Missing output directory: $OUTPUT_DIR" 1>&2
|
echo "Missing output directory: $OUTPUT_DIR" 1>&2
|
||||||
|
@ -4,15 +4,15 @@
|
|||||||
"workspaces": [
|
"workspaces": [
|
||||||
"packages/*"
|
"packages/*"
|
||||||
],
|
],
|
||||||
|
"dependencies": {},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"depcheck": "^1.4.2",
|
|
||||||
"husky": "^8.0.3",
|
"husky": "^8.0.3",
|
||||||
"lerna": "^8.0.0",
|
"lerna": "^8.0.0",
|
||||||
"patch-package": "^8.0.0"
|
"depcheck": "^1.4.2"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"prepare": "husky install",
|
"prepare": "husky install",
|
||||||
"build": "lerna run build --stream",
|
"build": "lerna run build --stream",
|
||||||
"lint": "lerna run lint --stream"
|
"lint": "lerna run lint --stream -- --max-warnings=0"
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -1 +0,0 @@
|
|||||||
v20.12.1
|
|
@ -1,76 +1 @@
|
|||||||
# backend
|
# Backend for Snowball Tools
|
||||||
|
|
||||||
This backend is a [node.js](https://nodejs.org/) [express.js](https://expressjs.com/) [apollo server](https://www.apollographql.com/docs/apollo-server/) project in a [yarn workspace](https://yarnpkg.com/features/workspaces).
|
|
||||||
|
|
||||||
## Getting Started
|
|
||||||
|
|
||||||
### Install dependencies
|
|
||||||
|
|
||||||
In the root of the project, run:
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
yarn
|
|
||||||
```
|
|
||||||
|
|
||||||
### Build backend
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
yarn build --ignore frontend
|
|
||||||
```
|
|
||||||
|
|
||||||
### Environment variables
|
|
||||||
|
|
||||||
#### Local
|
|
||||||
|
|
||||||
Copy the `environments/local.toml.example` file to `environments/local.toml`:
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
cp environments/local.toml.example environments/local.toml
|
|
||||||
```
|
|
||||||
|
|
||||||
#### Staging environment variables
|
|
||||||
|
|
||||||
In the deployment repository, update staging [staging/configmaps/config/prod.toml](https://git.vdb.to/cerc-io/snowballtools-base-api-deployments/src/commit/318c2bc09f334dca79c3501838512749f9431bf1/deployments/staging/configmaps/config/prod.toml)
|
|
||||||
|
|
||||||
#### Production environment variables
|
|
||||||
|
|
||||||
In the deployment repository, update production [production/configmaps/config/prod.toml](https://git.vdb.to/cerc-io/snowballtools-base-api-deployments/src/commit/318c2bc09f334dca79c3501838512749f9431bf1/deployments/production/configmaps/config/prod.toml)
|
|
||||||
|
|
||||||
### Run development server
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
yarn start
|
|
||||||
```
|
|
||||||
|
|
||||||
## Deployment
|
|
||||||
|
|
||||||
Clone the [deployer repository](https://git.vdb.to/cerc-io/snowballtools-base-api-deployments):
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
git clone git@git.vdb.to:cerc-io/snowballtools-base-api-deployments.git
|
|
||||||
```
|
|
||||||
|
|
||||||
### Staging
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
echo trigger >> .gitea/workflows/triggers/staging-deploy
|
|
||||||
git commit -a -m "Deploy v0.0.8" # replace with version number
|
|
||||||
git push
|
|
||||||
```
|
|
||||||
|
|
||||||
### Production
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
echo trigger >> .gitea/workflows/triggers/production-deploy
|
|
||||||
git commit -a -m "Deploy v0.0.8" # replace with version number
|
|
||||||
git push
|
|
||||||
```
|
|
||||||
|
|
||||||
### Deployment status
|
|
||||||
|
|
||||||
Dumb for now
|
|
||||||
|
|
||||||
- [Staging](https://snowballtools-base-api.staging.apps.snowballtools.com/staging/version)
|
|
||||||
- [Production](https://snowballtools-base-api.apps.snowballtools.com/staging/version)
|
|
||||||
|
|
||||||
Update version number manually in [routes/staging.ts](/packages/backend/src/routes/staging.ts)
|
|
||||||
|
@ -3,12 +3,9 @@
|
|||||||
port = 8000
|
port = 8000
|
||||||
gqlPath = "/graphql"
|
gqlPath = "/graphql"
|
||||||
[server.session]
|
[server.session]
|
||||||
secret = ""
|
secret = "p4yfpkqnddkui2iw7t6hbhwq74lbqs7bhobvmfhrowoi"
|
||||||
# Frontend webapp URL origin
|
|
||||||
appOriginUrl = "http://localhost:3000"
|
appOriginUrl = "http://localhost:3000"
|
||||||
# Set to true if server running behind proxy
|
|
||||||
trustProxy = false
|
trustProxy = false
|
||||||
# Backend URL hostname
|
|
||||||
domain = "localhost"
|
domain = "localhost"
|
||||||
|
|
||||||
[database]
|
[database]
|
||||||
@ -22,7 +19,6 @@
|
|||||||
|
|
||||||
[registryConfig]
|
[registryConfig]
|
||||||
fetchDeploymentRecordDelay = 5000
|
fetchDeploymentRecordDelay = 5000
|
||||||
checkAuctionStatusDelay = 5000
|
|
||||||
restEndpoint = "http://localhost:1317"
|
restEndpoint = "http://localhost:1317"
|
||||||
gqlEndpoint = "http://localhost:9473/api"
|
gqlEndpoint = "http://localhost:9473/api"
|
||||||
chainId = "laconic_9000-1"
|
chainId = "laconic_9000-1"
|
||||||
@ -30,14 +26,9 @@
|
|||||||
bondId = ""
|
bondId = ""
|
||||||
authority = ""
|
authority = ""
|
||||||
[registryConfig.fee]
|
[registryConfig.fee]
|
||||||
gas = ""
|
amount = "200000"
|
||||||
fees = ""
|
denom = "aphoton"
|
||||||
gasPrice = "1alnt"
|
gas = "750000"
|
||||||
|
|
||||||
# Durations are set to 2 mins as deployers may take time with ongoing deployments and auctions
|
[misc]
|
||||||
[auction]
|
projectDomain = "apps.snowballtools.com"
|
||||||
commitFee = "100000"
|
|
||||||
commitsDuration = "120s"
|
|
||||||
revealFee = "100000"
|
|
||||||
revealsDuration = "120s"
|
|
||||||
denom = "alnt"
|
|
||||||
|
@ -1,25 +1,20 @@
|
|||||||
{
|
{
|
||||||
"name": "backend",
|
"name": "backend",
|
||||||
"license": "UNLICENSED",
|
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@cerc-io/registry-sdk": "^0.2.11",
|
"@cerc-io/laconic-sdk": "^0.1.14",
|
||||||
"@graphql-tools/schema": "^10.0.2",
|
"@graphql-tools/schema": "^10.0.2",
|
||||||
"@graphql-tools/utils": "^10.0.12",
|
"@graphql-tools/utils": "^10.0.12",
|
||||||
"@octokit/oauth-app": "^6.1.0",
|
"@octokit/oauth-app": "^6.1.0",
|
||||||
"@turnkey/sdk-server": "^0.1.0",
|
|
||||||
"@types/debug": "^4.1.5",
|
"@types/debug": "^4.1.5",
|
||||||
"@types/express": "^4.17.21",
|
"@types/express": "^4.17.21",
|
||||||
"@types/node": "^20.11.0",
|
"@types/node": "^20.11.0",
|
||||||
"@types/semver": "^7.5.8",
|
|
||||||
"apollo-server-core": "^3.13.0",
|
"apollo-server-core": "^3.13.0",
|
||||||
"apollo-server-express": "^3.13.0",
|
"apollo-server-express": "^3.13.0",
|
||||||
"cookie-session": "^2.1.0",
|
|
||||||
"cors": "^2.8.5",
|
"cors": "^2.8.5",
|
||||||
"debug": "^4.3.1",
|
"debug": "^4.3.1",
|
||||||
"express": "^4.18.2",
|
"express": "^4.18.2",
|
||||||
"express-async-errors": "^3.1.1",
|
|
||||||
"express-session": "^1.18.0",
|
"express-session": "^1.18.0",
|
||||||
"fs-extra": "^11.2.0",
|
"fs-extra": "^11.2.0",
|
||||||
"graphql": "^16.8.1",
|
"graphql": "^16.8.1",
|
||||||
@ -40,21 +35,30 @@
|
|||||||
"copy-assets": "copyfiles -u 1 src/**/*.gql dist/",
|
"copy-assets": "copyfiles -u 1 src/**/*.gql dist/",
|
||||||
"clean": "rm -rf ./dist",
|
"clean": "rm -rf ./dist",
|
||||||
"build": "yarn clean && tsc && yarn copy-assets",
|
"build": "yarn clean && tsc && yarn copy-assets",
|
||||||
|
"lint": "eslint .",
|
||||||
"format": "prettier --write .",
|
"format": "prettier --write .",
|
||||||
"format:check": "prettier --check .",
|
"format:check": "prettier --check .",
|
||||||
"lint": "tsc --noEmit",
|
|
||||||
"test:registry:init": "DEBUG=snowball:* ts-node ./test/initialize-registry.ts",
|
"test:registry:init": "DEBUG=snowball:* ts-node ./test/initialize-registry.ts",
|
||||||
"test:registry:publish-deploy-records": "DEBUG=snowball:* ts-node ./test/publish-deploy-records.ts",
|
"test:registry:publish-deploy-records": "DEBUG=snowball:* ts-node ./test/publish-deploy-records.ts",
|
||||||
"test:registry:publish-deployment-removal-records": "DEBUG=snowball:* ts-node ./test/publish-deployment-removal-records.ts",
|
|
||||||
"test:db:load:fixtures": "DEBUG=snowball:* ts-node ./test/initialize-db.ts",
|
"test:db:load:fixtures": "DEBUG=snowball:* ts-node ./test/initialize-db.ts",
|
||||||
"test:db:delete": "DEBUG=snowball:* ts-node ./test/delete-db.ts"
|
"test:db:delete": "DEBUG=snowball:* ts-node ./test/delete-db.ts"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/cookie-session": "^2.0.49",
|
|
||||||
"@types/express-session": "^1.17.10",
|
"@types/express-session": "^1.17.10",
|
||||||
"@types/fs-extra": "^11.0.4",
|
"@types/fs-extra": "^11.0.4",
|
||||||
|
"@typescript-eslint/eslint-plugin": "^6.18.1",
|
||||||
|
"@typescript-eslint/parser": "^6.18.1",
|
||||||
"better-sqlite3": "^9.2.2",
|
"better-sqlite3": "^9.2.2",
|
||||||
"copyfiles": "^2.4.1",
|
"copyfiles": "^2.4.1",
|
||||||
|
"eslint": "^8.56.0",
|
||||||
|
"eslint-config-prettier": "^9.1.0",
|
||||||
|
"eslint-config-semistandard": "^15.0.1",
|
||||||
|
"eslint-config-standard": "^16.0.3",
|
||||||
|
"eslint-plugin-import": "^2.27.5",
|
||||||
|
"eslint-plugin-node": "^11.1.0",
|
||||||
|
"eslint-plugin-prettier": "^5.1.3",
|
||||||
|
"eslint-plugin-promise": "^5.1.0",
|
||||||
|
"eslint-plugin-standard": "^5.0.0",
|
||||||
"prettier": "^3.1.1",
|
"prettier": "^3.1.1",
|
||||||
"workspace": "^0.0.1-preview.1"
|
"workspace": "^0.0.1-preview.1"
|
||||||
}
|
}
|
||||||
|
@ -34,21 +34,16 @@ export interface RegistryConfig {
|
|||||||
privateKey: string;
|
privateKey: string;
|
||||||
bondId: string;
|
bondId: string;
|
||||||
fetchDeploymentRecordDelay: number;
|
fetchDeploymentRecordDelay: number;
|
||||||
checkAuctionStatusDelay: number;
|
|
||||||
authority: string;
|
authority: string;
|
||||||
fee: {
|
fee: {
|
||||||
|
amount: string;
|
||||||
|
denom: string;
|
||||||
gas: string;
|
gas: string;
|
||||||
fees: string;
|
|
||||||
gasPrice: string;
|
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AuctionConfig {
|
export interface MiscConfig {
|
||||||
commitFee: string;
|
projectDomain: string;
|
||||||
commitsDuration: string;
|
|
||||||
revealFee: string;
|
|
||||||
revealsDuration: string;
|
|
||||||
denom: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface Config {
|
export interface Config {
|
||||||
@ -56,11 +51,5 @@ export interface Config {
|
|||||||
database: DatabaseConfig;
|
database: DatabaseConfig;
|
||||||
gitHub: GitHubConfig;
|
gitHub: GitHubConfig;
|
||||||
registryConfig: RegistryConfig;
|
registryConfig: RegistryConfig;
|
||||||
auction: AuctionConfig;
|
misc: MiscConfig;
|
||||||
turnkey: {
|
|
||||||
apiBaseUrl: string;
|
|
||||||
apiPublicKey: string;
|
|
||||||
apiPrivateKey: string;
|
|
||||||
defaultOrganizationId: string;
|
|
||||||
};
|
|
||||||
}
|
}
|
||||||
|
@ -3,9 +3,7 @@ import {
|
|||||||
DeepPartial,
|
DeepPartial,
|
||||||
FindManyOptions,
|
FindManyOptions,
|
||||||
FindOneOptions,
|
FindOneOptions,
|
||||||
FindOptionsWhere,
|
FindOptionsWhere
|
||||||
IsNull,
|
|
||||||
Not
|
|
||||||
} from 'typeorm';
|
} from 'typeorm';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import debug from 'debug';
|
import debug from 'debug';
|
||||||
@ -13,7 +11,7 @@ import assert from 'assert';
|
|||||||
import { customAlphabet } from 'nanoid';
|
import { customAlphabet } from 'nanoid';
|
||||||
import { lowercase, numbers } from 'nanoid-dictionary';
|
import { lowercase, numbers } from 'nanoid-dictionary';
|
||||||
|
|
||||||
import { DatabaseConfig } from './config';
|
import { DatabaseConfig, MiscConfig } from './config';
|
||||||
import { User } from './entity/User';
|
import { User } from './entity/User';
|
||||||
import { Organization } from './entity/Organization';
|
import { Organization } from './entity/Organization';
|
||||||
import { Project } from './entity/Project';
|
import { Project } from './entity/Project';
|
||||||
@ -23,7 +21,6 @@ import { EnvironmentVariable } from './entity/EnvironmentVariable';
|
|||||||
import { Domain } from './entity/Domain';
|
import { Domain } from './entity/Domain';
|
||||||
import { getEntities, loadAndSaveData } from './utils';
|
import { getEntities, loadAndSaveData } from './utils';
|
||||||
import { UserOrganization } from './entity/UserOrganization';
|
import { UserOrganization } from './entity/UserOrganization';
|
||||||
import { Deployer } from './entity/Deployer';
|
|
||||||
|
|
||||||
const ORGANIZATION_DATA_PATH = '../test/fixtures/organizations.json';
|
const ORGANIZATION_DATA_PATH = '../test/fixtures/organizations.json';
|
||||||
|
|
||||||
@ -34,8 +31,9 @@ const nanoid = customAlphabet(lowercase + numbers, 8);
|
|||||||
// TODO: Fix order of methods
|
// TODO: Fix order of methods
|
||||||
export class Database {
|
export class Database {
|
||||||
private dataSource: DataSource;
|
private dataSource: DataSource;
|
||||||
|
private projectDomain: string;
|
||||||
|
|
||||||
constructor({ dbPath }: DatabaseConfig) {
|
constructor ({ dbPath } : DatabaseConfig, { projectDomain } : MiscConfig) {
|
||||||
this.dataSource = new DataSource({
|
this.dataSource = new DataSource({
|
||||||
type: 'better-sqlite3',
|
type: 'better-sqlite3',
|
||||||
database: dbPath,
|
database: dbPath,
|
||||||
@ -43,9 +41,11 @@ export class Database {
|
|||||||
synchronize: true,
|
synchronize: true,
|
||||||
logging: false
|
logging: false
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.projectDomain = projectDomain;
|
||||||
}
|
}
|
||||||
|
|
||||||
async init(): Promise<void> {
|
async init (): Promise<void> {
|
||||||
await this.dataSource.initialize();
|
await this.dataSource.initialize();
|
||||||
log('database initialized');
|
log('database initialized');
|
||||||
|
|
||||||
@ -58,21 +58,21 @@ export class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async getUser(options: FindOneOptions<User>): Promise<User | null> {
|
async getUser (options: FindOneOptions<User>): Promise<User | null> {
|
||||||
const userRepository = this.dataSource.getRepository(User);
|
const userRepository = this.dataSource.getRepository(User);
|
||||||
const user = await userRepository.findOne(options);
|
const user = await userRepository.findOne(options);
|
||||||
|
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addUser(data: DeepPartial<User>): Promise<User> {
|
async addUser (data: DeepPartial<User>): Promise<User> {
|
||||||
const userRepository = this.dataSource.getRepository(User);
|
const userRepository = this.dataSource.getRepository(User);
|
||||||
const user = await userRepository.save(data);
|
const user = await userRepository.save(data);
|
||||||
|
|
||||||
return user;
|
return user;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateUser(user: User, data: DeepPartial<User>): Promise<boolean> {
|
async updateUser (user: User, data: DeepPartial<User>): Promise<boolean> {
|
||||||
const userRepository = this.dataSource.getRepository(User);
|
const userRepository = this.dataSource.getRepository(User);
|
||||||
const updateResult = await userRepository.update({ id: user.id }, data);
|
const updateResult = await userRepository.update({ id: user.id }, data);
|
||||||
assert(updateResult.affected);
|
assert(updateResult.affected);
|
||||||
@ -80,7 +80,7 @@ export class Database {
|
|||||||
return updateResult.affected > 0;
|
return updateResult.affected > 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getOrganizations(
|
async getOrganizations (
|
||||||
options: FindManyOptions<Organization>
|
options: FindManyOptions<Organization>
|
||||||
): Promise<Organization[]> {
|
): Promise<Organization[]> {
|
||||||
const organizationRepository = this.dataSource.getRepository(Organization);
|
const organizationRepository = this.dataSource.getRepository(Organization);
|
||||||
@ -89,7 +89,7 @@ export class Database {
|
|||||||
return organizations;
|
return organizations;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getOrganization(
|
async getOrganization (
|
||||||
options: FindOneOptions<Organization>
|
options: FindOneOptions<Organization>
|
||||||
): Promise<Organization | null> {
|
): Promise<Organization | null> {
|
||||||
const organizationRepository = this.dataSource.getRepository(Organization);
|
const organizationRepository = this.dataSource.getRepository(Organization);
|
||||||
@ -98,7 +98,7 @@ export class Database {
|
|||||||
return organization;
|
return organization;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getOrganizationsByUserId(userId: string): Promise<Organization[]> {
|
async getOrganizationsByUserId (userId: string): Promise<Organization[]> {
|
||||||
const organizationRepository = this.dataSource.getRepository(Organization);
|
const organizationRepository = this.dataSource.getRepository(Organization);
|
||||||
|
|
||||||
const userOrgs = await organizationRepository.find({
|
const userOrgs = await organizationRepository.find({
|
||||||
@ -114,21 +114,21 @@ export class Database {
|
|||||||
return userOrgs;
|
return userOrgs;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addUserOrganization(data: DeepPartial<UserOrganization>): Promise<UserOrganization> {
|
async addUserOrganization (data: DeepPartial<UserOrganization>): Promise<UserOrganization> {
|
||||||
const userOrganizationRepository = this.dataSource.getRepository(UserOrganization);
|
const userOrganizationRepository = this.dataSource.getRepository(UserOrganization);
|
||||||
const newUserOrganization = await userOrganizationRepository.save(data);
|
const newUserOrganization = await userOrganizationRepository.save(data);
|
||||||
|
|
||||||
return newUserOrganization;
|
return newUserOrganization;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjects(options: FindManyOptions<Project>): Promise<Project[]> {
|
async getProjects (options: FindManyOptions<Project>): Promise<Project[]> {
|
||||||
const projectRepository = this.dataSource.getRepository(Project);
|
const projectRepository = this.dataSource.getRepository(Project);
|
||||||
const projects = await projectRepository.find(options);
|
const projects = await projectRepository.find(options);
|
||||||
|
|
||||||
return projects;
|
return projects;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectById(projectId: string): Promise<Project | null> {
|
async getProjectById (projectId: string): Promise<Project | null> {
|
||||||
const projectRepository = this.dataSource.getRepository(Project);
|
const projectRepository = this.dataSource.getRepository(Project);
|
||||||
|
|
||||||
const project = await projectRepository
|
const project = await projectRepository
|
||||||
@ -140,9 +140,7 @@ export class Database {
|
|||||||
)
|
)
|
||||||
.leftJoinAndSelect('deployments.createdBy', 'user')
|
.leftJoinAndSelect('deployments.createdBy', 'user')
|
||||||
.leftJoinAndSelect('deployments.domain', 'domain')
|
.leftJoinAndSelect('deployments.domain', 'domain')
|
||||||
.leftJoinAndSelect('deployments.deployer', 'deployer')
|
|
||||||
.leftJoinAndSelect('project.owner', 'owner')
|
.leftJoinAndSelect('project.owner', 'owner')
|
||||||
.leftJoinAndSelect('project.deployers', 'deployers')
|
|
||||||
.leftJoinAndSelect('project.organization', 'organization')
|
.leftJoinAndSelect('project.organization', 'organization')
|
||||||
.where('project.id = :projectId', {
|
.where('project.id = :projectId', {
|
||||||
projectId
|
projectId
|
||||||
@ -152,25 +150,7 @@ export class Database {
|
|||||||
return project;
|
return project;
|
||||||
}
|
}
|
||||||
|
|
||||||
async allProjectsWithoutDeployments(): Promise<Project[]> {
|
async getProjectsInOrganization (
|
||||||
const allProjects = await this.getProjects({
|
|
||||||
where: {
|
|
||||||
auctionId: Not(IsNull()),
|
|
||||||
},
|
|
||||||
relations: ['deployments'],
|
|
||||||
withDeleted: true,
|
|
||||||
});
|
|
||||||
|
|
||||||
const projects = allProjects.filter(project => {
|
|
||||||
if (project.deletedAt !== null) return false;
|
|
||||||
|
|
||||||
return project.deployments.length === 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
return projects;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getProjectsInOrganization(
|
|
||||||
userId: string,
|
userId: string,
|
||||||
organizationSlug: string
|
organizationSlug: string
|
||||||
): Promise<Project[]> {
|
): Promise<Project[]> {
|
||||||
@ -201,7 +181,7 @@ export class Database {
|
|||||||
/**
|
/**
|
||||||
* Get deployments with specified filter
|
* Get deployments with specified filter
|
||||||
*/
|
*/
|
||||||
async getDeployments(
|
async getDeployments (
|
||||||
options: FindManyOptions<Deployment>
|
options: FindManyOptions<Deployment>
|
||||||
): Promise<Deployment[]> {
|
): Promise<Deployment[]> {
|
||||||
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
||||||
@ -210,13 +190,12 @@ export class Database {
|
|||||||
return deployments;
|
return deployments;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDeploymentsByProjectId(projectId: string): Promise<Deployment[]> {
|
async getDeploymentsByProjectId (projectId: string): Promise<Deployment[]> {
|
||||||
return this.getDeployments({
|
return this.getDeployments({
|
||||||
relations: {
|
relations: {
|
||||||
project: true,
|
project: true,
|
||||||
domain: true,
|
domain: true,
|
||||||
createdBy: true,
|
createdBy: true
|
||||||
deployer: true,
|
|
||||||
},
|
},
|
||||||
where: {
|
where: {
|
||||||
project: {
|
project: {
|
||||||
@ -229,7 +208,7 @@ export class Database {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDeployment(
|
async getDeployment (
|
||||||
options: FindOneOptions<Deployment>
|
options: FindOneOptions<Deployment>
|
||||||
): Promise<Deployment | null> {
|
): Promise<Deployment | null> {
|
||||||
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
||||||
@ -238,14 +217,14 @@ export class Database {
|
|||||||
return deployment;
|
return deployment;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDomains(options: FindManyOptions<Domain>): Promise<Domain[]> {
|
async getDomains (options: FindManyOptions<Domain>): Promise<Domain[]> {
|
||||||
const domainRepository = this.dataSource.getRepository(Domain);
|
const domainRepository = this.dataSource.getRepository(Domain);
|
||||||
const domains = await domainRepository.find(options);
|
const domains = await domainRepository.find(options);
|
||||||
|
|
||||||
return domains;
|
return domains;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addDeployment(data: DeepPartial<Deployment>): Promise<Deployment> {
|
async addDeployment (data: DeepPartial<Deployment>): Promise<Deployment> {
|
||||||
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
||||||
|
|
||||||
const id = nanoid();
|
const id = nanoid();
|
||||||
@ -259,7 +238,7 @@ export class Database {
|
|||||||
return deployment;
|
return deployment;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectMembersByProjectId(
|
async getProjectMembersByProjectId (
|
||||||
projectId: string
|
projectId: string
|
||||||
): Promise<ProjectMember[]> {
|
): Promise<ProjectMember[]> {
|
||||||
const projectMemberRepository =
|
const projectMemberRepository =
|
||||||
@ -280,7 +259,7 @@ export class Database {
|
|||||||
return projectMembers;
|
return projectMembers;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getEnvironmentVariablesByProjectId(
|
async getEnvironmentVariablesByProjectId (
|
||||||
projectId: string,
|
projectId: string,
|
||||||
filter?: FindOptionsWhere<EnvironmentVariable>
|
filter?: FindOptionsWhere<EnvironmentVariable>
|
||||||
): Promise<EnvironmentVariable[]> {
|
): Promise<EnvironmentVariable[]> {
|
||||||
@ -299,7 +278,7 @@ export class Database {
|
|||||||
return environmentVariables;
|
return environmentVariables;
|
||||||
}
|
}
|
||||||
|
|
||||||
async removeProjectMemberById(projectMemberId: string): Promise<boolean> {
|
async removeProjectMemberById (projectMemberId: string): Promise<boolean> {
|
||||||
const projectMemberRepository =
|
const projectMemberRepository =
|
||||||
this.dataSource.getRepository(ProjectMember);
|
this.dataSource.getRepository(ProjectMember);
|
||||||
|
|
||||||
@ -314,7 +293,7 @@ export class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateProjectMemberById(
|
async updateProjectMemberById (
|
||||||
projectMemberId: string,
|
projectMemberId: string,
|
||||||
data: DeepPartial<ProjectMember>
|
data: DeepPartial<ProjectMember>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
@ -328,7 +307,7 @@ export class Database {
|
|||||||
return Boolean(updateResult.affected);
|
return Boolean(updateResult.affected);
|
||||||
}
|
}
|
||||||
|
|
||||||
async addProjectMember(
|
async addProjectMember (
|
||||||
data: DeepPartial<ProjectMember>
|
data: DeepPartial<ProjectMember>
|
||||||
): Promise<ProjectMember> {
|
): Promise<ProjectMember> {
|
||||||
const projectMemberRepository =
|
const projectMemberRepository =
|
||||||
@ -338,7 +317,7 @@ export class Database {
|
|||||||
return newProjectMember;
|
return newProjectMember;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addEnvironmentVariables(
|
async addEnvironmentVariables (
|
||||||
data: DeepPartial<EnvironmentVariable>[]
|
data: DeepPartial<EnvironmentVariable>[]
|
||||||
): Promise<EnvironmentVariable[]> {
|
): Promise<EnvironmentVariable[]> {
|
||||||
const environmentVariableRepository =
|
const environmentVariableRepository =
|
||||||
@ -349,7 +328,7 @@ export class Database {
|
|||||||
return savedEnvironmentVariables;
|
return savedEnvironmentVariables;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateEnvironmentVariable(
|
async updateEnvironmentVariable (
|
||||||
environmentVariableId: string,
|
environmentVariableId: string,
|
||||||
data: DeepPartial<EnvironmentVariable>
|
data: DeepPartial<EnvironmentVariable>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
@ -363,7 +342,7 @@ export class Database {
|
|||||||
return Boolean(updateResult.affected);
|
return Boolean(updateResult.affected);
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteEnvironmentVariable(
|
async deleteEnvironmentVariable (
|
||||||
environmentVariableId: string
|
environmentVariableId: string
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
const environmentVariableRepository =
|
const environmentVariableRepository =
|
||||||
@ -379,7 +358,7 @@ export class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectMemberById(projectMemberId: string): Promise<ProjectMember> {
|
async getProjectMemberById (projectMemberId: string): Promise<ProjectMember> {
|
||||||
const projectMemberRepository =
|
const projectMemberRepository =
|
||||||
this.dataSource.getRepository(ProjectMember);
|
this.dataSource.getRepository(ProjectMember);
|
||||||
|
|
||||||
@ -402,7 +381,7 @@ export class Database {
|
|||||||
return projectMemberWithProject[0];
|
return projectMemberWithProject[0];
|
||||||
}
|
}
|
||||||
|
|
||||||
async getProjectsBySearchText(
|
async getProjectsBySearchText (
|
||||||
userId: string,
|
userId: string,
|
||||||
searchText: string
|
searchText: string
|
||||||
): Promise<Project[]> {
|
): Promise<Project[]> {
|
||||||
@ -424,14 +403,14 @@ export class Database {
|
|||||||
return projects;
|
return projects;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDeploymentById(
|
async updateDeploymentById (
|
||||||
deploymentId: string,
|
deploymentId: string,
|
||||||
data: DeepPartial<Deployment>
|
data: DeepPartial<Deployment>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
return this.updateDeployment({ id: deploymentId }, data);
|
return this.updateDeployment({ id: deploymentId }, data);
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDeployment(
|
async updateDeployment (
|
||||||
criteria: FindOptionsWhere<Deployment>,
|
criteria: FindOptionsWhere<Deployment>,
|
||||||
data: DeepPartial<Deployment>
|
data: DeepPartial<Deployment>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
@ -441,7 +420,7 @@ export class Database {
|
|||||||
return Boolean(updateResult.affected);
|
return Boolean(updateResult.affected);
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDeploymentsByProjectIds(
|
async updateDeploymentsByProjectIds (
|
||||||
projectIds: string[],
|
projectIds: string[],
|
||||||
data: DeepPartial<Deployment>
|
data: DeepPartial<Deployment>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
@ -457,20 +436,7 @@ export class Database {
|
|||||||
return Boolean(updateResult.affected);
|
return Boolean(updateResult.affected);
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteDeploymentById(deploymentId: string): Promise<boolean> {
|
async addProject (user: User, organizationId: string, data: DeepPartial<Project>): Promise<Project> {
|
||||||
const deploymentRepository = this.dataSource.getRepository(Deployment);
|
|
||||||
const deployment = await deploymentRepository.findOneOrFail({
|
|
||||||
where: {
|
|
||||||
id: deploymentId
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
const deleteResult = await deploymentRepository.softRemove(deployment);
|
|
||||||
|
|
||||||
return Boolean(deleteResult);
|
|
||||||
}
|
|
||||||
|
|
||||||
async addProject(user: User, organizationId: string, data: DeepPartial<Project>): Promise<Project> {
|
|
||||||
const projectRepository = this.dataSource.getRepository(Project);
|
const projectRepository = this.dataSource.getRepository(Project);
|
||||||
|
|
||||||
// TODO: Check if organization exists
|
// TODO: Check if organization exists
|
||||||
@ -486,16 +452,12 @@ export class Database {
|
|||||||
id: organizationId
|
id: organizationId
|
||||||
});
|
});
|
||||||
|
|
||||||
|
newProject.subDomain = `${newProject.name}.${this.projectDomain}`;
|
||||||
|
|
||||||
return projectRepository.save(newProject);
|
return projectRepository.save(newProject);
|
||||||
}
|
}
|
||||||
|
|
||||||
async saveProject(project: Project): Promise<Project> {
|
async updateProjectById (
|
||||||
const projectRepository = this.dataSource.getRepository(Project);
|
|
||||||
|
|
||||||
return projectRepository.save(project);
|
|
||||||
}
|
|
||||||
|
|
||||||
async updateProjectById(
|
|
||||||
projectId: string,
|
projectId: string,
|
||||||
data: DeepPartial<Project>
|
data: DeepPartial<Project>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
@ -508,7 +470,7 @@ export class Database {
|
|||||||
return Boolean(updateResult.affected);
|
return Boolean(updateResult.affected);
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteProjectById(projectId: string): Promise<boolean> {
|
async deleteProjectById (projectId: string): Promise<boolean> {
|
||||||
const projectRepository = this.dataSource.getRepository(Project);
|
const projectRepository = this.dataSource.getRepository(Project);
|
||||||
const project = await projectRepository.findOneOrFail({
|
const project = await projectRepository.findOneOrFail({
|
||||||
where: {
|
where: {
|
||||||
@ -524,7 +486,7 @@ export class Database {
|
|||||||
return Boolean(deleteResult);
|
return Boolean(deleteResult);
|
||||||
}
|
}
|
||||||
|
|
||||||
async deleteDomainById(domainId: string): Promise<boolean> {
|
async deleteDomainById (domainId: string): Promise<boolean> {
|
||||||
const domainRepository = this.dataSource.getRepository(Domain);
|
const domainRepository = this.dataSource.getRepository(Domain);
|
||||||
|
|
||||||
const deleteResult = await domainRepository.softDelete({ id: domainId });
|
const deleteResult = await domainRepository.softDelete({ id: domainId });
|
||||||
@ -536,21 +498,21 @@ export class Database {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async addDomain(data: DeepPartial<Domain>): Promise<Domain> {
|
async addDomain (data: DeepPartial<Domain>): Promise<Domain> {
|
||||||
const domainRepository = this.dataSource.getRepository(Domain);
|
const domainRepository = this.dataSource.getRepository(Domain);
|
||||||
const newDomain = await domainRepository.save(data);
|
const newDomain = await domainRepository.save(data);
|
||||||
|
|
||||||
return newDomain;
|
return newDomain;
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDomain(options: FindOneOptions<Domain>): Promise<Domain | null> {
|
async getDomain (options: FindOneOptions<Domain>): Promise<Domain | null> {
|
||||||
const domainRepository = this.dataSource.getRepository(Domain);
|
const domainRepository = this.dataSource.getRepository(Domain);
|
||||||
const domain = await domainRepository.findOne(options);
|
const domain = await domainRepository.findOne(options);
|
||||||
|
|
||||||
return domain;
|
return domain;
|
||||||
}
|
}
|
||||||
|
|
||||||
async updateDomainById(
|
async updateDomainById (
|
||||||
domainId: string,
|
domainId: string,
|
||||||
data: DeepPartial<Domain>
|
data: DeepPartial<Domain>
|
||||||
): Promise<boolean> {
|
): Promise<boolean> {
|
||||||
@ -560,7 +522,7 @@ export class Database {
|
|||||||
return Boolean(updateResult.affected);
|
return Boolean(updateResult.affected);
|
||||||
}
|
}
|
||||||
|
|
||||||
async getDomainsByProjectId(
|
async getDomainsByProjectId (
|
||||||
projectId: string,
|
projectId: string,
|
||||||
filter?: FindOptionsWhere<Domain>
|
filter?: FindOptionsWhere<Domain>
|
||||||
): Promise<Domain[]> {
|
): Promise<Domain[]> {
|
||||||
@ -580,24 +542,4 @@ export class Database {
|
|||||||
|
|
||||||
return domains;
|
return domains;
|
||||||
}
|
}
|
||||||
|
|
||||||
async addDeployer(data: DeepPartial<Deployer>): Promise<Deployer> {
|
|
||||||
const deployerRepository = this.dataSource.getRepository(Deployer);
|
|
||||||
const newDomain = await deployerRepository.save(data);
|
|
||||||
|
|
||||||
return newDomain;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDeployers(): Promise<Deployer[]> {
|
|
||||||
const deployerRepository = this.dataSource.getRepository(Deployer);
|
|
||||||
const deployers = await deployerRepository.find();
|
|
||||||
return deployers;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getDeployerByLRN(deployerLrn: string): Promise<Deployer | null> {
|
|
||||||
const deployerRepository = this.dataSource.getRepository(Deployer);
|
|
||||||
const deployer = await deployerRepository.findOne({ where: { deployerLrn } });
|
|
||||||
|
|
||||||
return deployer;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -1,26 +0,0 @@
|
|||||||
import { Entity, PrimaryColumn, Column, ManyToMany } from 'typeorm';
|
|
||||||
import { Project } from './Project';
|
|
||||||
|
|
||||||
@Entity()
|
|
||||||
export class Deployer {
|
|
||||||
@PrimaryColumn('varchar')
|
|
||||||
deployerLrn!: string;
|
|
||||||
|
|
||||||
@Column('varchar')
|
|
||||||
deployerId!: string;
|
|
||||||
|
|
||||||
@Column('varchar')
|
|
||||||
deployerApiUrl!: string;
|
|
||||||
|
|
||||||
@Column('varchar')
|
|
||||||
baseDomain!: string;
|
|
||||||
|
|
||||||
@Column('varchar', { nullable: true })
|
|
||||||
minimumPayment!: string | null;
|
|
||||||
|
|
||||||
@Column('varchar', { nullable: true })
|
|
||||||
paymentAddress!: string | null;
|
|
||||||
|
|
||||||
@ManyToMany(() => Project, (project) => project.deployers)
|
|
||||||
projects!: Project[];
|
|
||||||
}
|
|
@ -6,15 +6,13 @@ import {
|
|||||||
UpdateDateColumn,
|
UpdateDateColumn,
|
||||||
ManyToOne,
|
ManyToOne,
|
||||||
OneToOne,
|
OneToOne,
|
||||||
JoinColumn,
|
JoinColumn
|
||||||
DeleteDateColumn
|
|
||||||
} from 'typeorm';
|
} from 'typeorm';
|
||||||
|
|
||||||
import { Project } from './Project';
|
import { Project } from './Project';
|
||||||
import { Domain } from './Domain';
|
import { Domain } from './Domain';
|
||||||
import { User } from './User';
|
import { User } from './User';
|
||||||
import { Deployer } from './Deployer';
|
import { AppDeploymentRecordAttributes } from '../types';
|
||||||
import { AppDeploymentRecordAttributes, AppDeploymentRemovalRecordAttributes } from '../types';
|
|
||||||
|
|
||||||
export enum Environment {
|
export enum Environment {
|
||||||
Production = 'Production',
|
Production = 'Production',
|
||||||
@ -26,7 +24,6 @@ export enum DeploymentStatus {
|
|||||||
Building = 'Building',
|
Building = 'Building',
|
||||||
Ready = 'Ready',
|
Ready = 'Ready',
|
||||||
Error = 'Error',
|
Error = 'Error',
|
||||||
Deleting = 'Deleting',
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ApplicationDeploymentRequest {
|
export interface ApplicationDeploymentRequest {
|
||||||
@ -34,22 +31,10 @@ export interface ApplicationDeploymentRequest {
|
|||||||
version: string;
|
version: string;
|
||||||
name: string;
|
name: string;
|
||||||
application: string;
|
application: string;
|
||||||
lrn?: string;
|
|
||||||
auction?: string;
|
|
||||||
config: string;
|
config: string;
|
||||||
meta: string;
|
meta: string;
|
||||||
payment?: string;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface ApplicationDeploymentRemovalRequest {
|
|
||||||
type: string;
|
|
||||||
version: string;
|
|
||||||
deployment: string;
|
|
||||||
auction?: string;
|
|
||||||
payment?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
export interface ApplicationRecord {
|
export interface ApplicationRecord {
|
||||||
type: string;
|
type: string;
|
||||||
version: string;
|
version: string;
|
||||||
@ -114,22 +99,6 @@ export class Deployment {
|
|||||||
@Column('simple-json', { nullable: true })
|
@Column('simple-json', { nullable: true })
|
||||||
applicationDeploymentRecordData!: AppDeploymentRecordAttributes | null;
|
applicationDeploymentRecordData!: AppDeploymentRecordAttributes | null;
|
||||||
|
|
||||||
@Column('varchar', { nullable: true })
|
|
||||||
applicationDeploymentRemovalRequestId!: string | null;
|
|
||||||
|
|
||||||
@Column('simple-json', { nullable: true })
|
|
||||||
applicationDeploymentRemovalRequestData!: ApplicationDeploymentRemovalRequest | null;
|
|
||||||
|
|
||||||
@Column('varchar', { nullable: true })
|
|
||||||
applicationDeploymentRemovalRecordId!: string | null;
|
|
||||||
|
|
||||||
@Column('simple-json', { nullable: true })
|
|
||||||
applicationDeploymentRemovalRecordData!: AppDeploymentRemovalRecordAttributes | null;
|
|
||||||
|
|
||||||
@ManyToOne(() => Deployer)
|
|
||||||
@JoinColumn({ name: 'deployerLrn' })
|
|
||||||
deployer!: Deployer;
|
|
||||||
|
|
||||||
@Column({
|
@Column({
|
||||||
enum: Environment
|
enum: Environment
|
||||||
})
|
})
|
||||||
@ -152,7 +121,4 @@ export class Deployment {
|
|||||||
|
|
||||||
@UpdateDateColumn()
|
@UpdateDateColumn()
|
||||||
updatedAt!: Date;
|
updatedAt!: Date;
|
||||||
|
|
||||||
@DeleteDateColumn()
|
|
||||||
deletedAt!: Date | null;
|
|
||||||
}
|
}
|
||||||
|
@ -7,16 +7,13 @@ import {
|
|||||||
ManyToOne,
|
ManyToOne,
|
||||||
JoinColumn,
|
JoinColumn,
|
||||||
OneToMany,
|
OneToMany,
|
||||||
DeleteDateColumn,
|
DeleteDateColumn
|
||||||
JoinTable,
|
|
||||||
ManyToMany
|
|
||||||
} from 'typeorm';
|
} from 'typeorm';
|
||||||
|
|
||||||
import { User } from './User';
|
import { User } from './User';
|
||||||
import { Organization } from './Organization';
|
import { Organization } from './Organization';
|
||||||
import { ProjectMember } from './ProjectMember';
|
import { ProjectMember } from './ProjectMember';
|
||||||
import { Deployment } from './Deployment';
|
import { Deployment } from './Deployment';
|
||||||
import { Deployer } from './Deployer';
|
|
||||||
|
|
||||||
@Entity()
|
@Entity()
|
||||||
export class Project {
|
export class Project {
|
||||||
@ -49,20 +46,6 @@ export class Project {
|
|||||||
@Column('text', { default: '' })
|
@Column('text', { default: '' })
|
||||||
description!: string;
|
description!: string;
|
||||||
|
|
||||||
@Column('varchar', { nullable: true })
|
|
||||||
auctionId!: string | null;
|
|
||||||
|
|
||||||
// Tx hash for sending coins from snowball to deployer
|
|
||||||
@Column('varchar', { nullable: true })
|
|
||||||
txHash!: string | null;
|
|
||||||
|
|
||||||
@ManyToMany(() => Deployer, (deployer) => (deployer.projects))
|
|
||||||
@JoinTable()
|
|
||||||
deployers!: Deployer[]
|
|
||||||
|
|
||||||
@Column('boolean', { default: false, nullable: true })
|
|
||||||
fundsReleased!: boolean;
|
|
||||||
|
|
||||||
// TODO: Compute template & framework in import repository
|
// TODO: Compute template & framework in import repository
|
||||||
@Column('varchar', { nullable: true })
|
@Column('varchar', { nullable: true })
|
||||||
template!: string | null;
|
template!: string | null;
|
||||||
@ -70,10 +53,6 @@ export class Project {
|
|||||||
@Column('varchar', { nullable: true })
|
@Column('varchar', { nullable: true })
|
||||||
framework!: string | null;
|
framework!: string | null;
|
||||||
|
|
||||||
// Address of the user who created the project i.e. requested deployments
|
|
||||||
@Column('varchar')
|
|
||||||
paymentAddress!: string;
|
|
||||||
|
|
||||||
@Column({
|
@Column({
|
||||||
type: 'simple-array'
|
type: 'simple-array'
|
||||||
})
|
})
|
||||||
@ -82,6 +61,9 @@ export class Project {
|
|||||||
@Column('varchar')
|
@Column('varchar')
|
||||||
icon!: string;
|
icon!: string;
|
||||||
|
|
||||||
|
@Column('varchar')
|
||||||
|
subDomain!: string;
|
||||||
|
|
||||||
@CreateDateColumn()
|
@CreateDateColumn()
|
||||||
createdAt!: Date;
|
createdAt!: Date;
|
||||||
|
|
||||||
|
@ -39,12 +39,6 @@ export class User {
|
|||||||
@CreateDateColumn()
|
@CreateDateColumn()
|
||||||
updatedAt!: Date;
|
updatedAt!: Date;
|
||||||
|
|
||||||
@Column()
|
|
||||||
subOrgId!: string;
|
|
||||||
|
|
||||||
@Column()
|
|
||||||
turnkeyWalletId!: string;
|
|
||||||
|
|
||||||
@OneToMany(() => ProjectMember, (projectMember) => projectMember.project, {
|
@OneToMany(() => ProjectMember, (projectMember) => projectMember.project, {
|
||||||
cascade: ['soft-remove']
|
cascade: ['soft-remove']
|
||||||
})
|
})
|
||||||
|
@ -1,4 +1,3 @@
|
|||||||
import 'express-async-errors';
|
|
||||||
import 'reflect-metadata';
|
import 'reflect-metadata';
|
||||||
import debug from 'debug';
|
import debug from 'debug';
|
||||||
import fs from 'fs';
|
import fs from 'fs';
|
||||||
@ -10,6 +9,8 @@ import { Database } from './database';
|
|||||||
import { createAndStartServer } from './server';
|
import { createAndStartServer } from './server';
|
||||||
import { createResolvers } from './resolvers';
|
import { createResolvers } from './resolvers';
|
||||||
import { getConfig } from './utils';
|
import { getConfig } from './utils';
|
||||||
|
import { Config } from './config';
|
||||||
|
import { DEFAULT_CONFIG_FILE_PATH } from './constants';
|
||||||
import { Service } from './service';
|
import { Service } from './service';
|
||||||
import { Registry } from './registry';
|
import { Registry } from './registry';
|
||||||
|
|
||||||
@ -17,15 +18,16 @@ const log = debug('snowball:server');
|
|||||||
const OAUTH_CLIENT_TYPE = 'oauth-app';
|
const OAUTH_CLIENT_TYPE = 'oauth-app';
|
||||||
|
|
||||||
export const main = async (): Promise<void> => {
|
export const main = async (): Promise<void> => {
|
||||||
const { server, database, gitHub, registryConfig } = await getConfig();
|
// TODO: get config path using cli
|
||||||
|
const { server, database, gitHub, registryConfig, misc } = await getConfig<Config>(DEFAULT_CONFIG_FILE_PATH);
|
||||||
|
|
||||||
const app = new OAuthApp({
|
const app = new OAuthApp({
|
||||||
clientType: OAUTH_CLIENT_TYPE,
|
clientType: OAUTH_CLIENT_TYPE,
|
||||||
clientId: gitHub.oAuth.clientId,
|
clientId: gitHub.oAuth.clientId,
|
||||||
clientSecret: gitHub.oAuth.clientSecret,
|
clientSecret: gitHub.oAuth.clientSecret
|
||||||
});
|
});
|
||||||
|
|
||||||
const db = new Database(database);
|
const db = new Database(database, misc);
|
||||||
await db.init();
|
await db.init();
|
||||||
|
|
||||||
const registry = new Registry(registryConfig);
|
const registry = new Registry(registryConfig);
|
||||||
@ -33,7 +35,7 @@ export const main = async (): Promise<void> => {
|
|||||||
{ gitHubConfig: gitHub, registryConfig },
|
{ gitHubConfig: gitHub, registryConfig },
|
||||||
db,
|
db,
|
||||||
app,
|
app,
|
||||||
registry,
|
registry
|
||||||
);
|
);
|
||||||
|
|
||||||
const typeDefs = fs
|
const typeDefs = fs
|
||||||
|
@ -1,67 +1,55 @@
|
|||||||
import assert from 'assert';
|
|
||||||
import debug from 'debug';
|
import debug from 'debug';
|
||||||
import { DateTime } from 'luxon';
|
import assert from 'assert';
|
||||||
import { Octokit } from 'octokit';
|
|
||||||
import { inc as semverInc } from 'semver';
|
import { inc as semverInc } from 'semver';
|
||||||
import { DeepPartial } from 'typeorm';
|
import { DateTime } from 'luxon';
|
||||||
|
|
||||||
import { Account, DEFAULT_GAS_ESTIMATION_MULTIPLIER, Registry as LaconicRegistry, getGasPrice, parseGasAndFees } from '@cerc-io/registry-sdk';
|
import { Registry as LaconicRegistry } from '@cerc-io/laconic-sdk';
|
||||||
import { DeliverTxResponse, IndexedTx } from '@cosmjs/stargate';
|
|
||||||
|
|
||||||
import { RegistryConfig } from './config';
|
import { RegistryConfig } from './config';
|
||||||
import {
|
import {
|
||||||
ApplicationRecord,
|
ApplicationRecord,
|
||||||
Deployment,
|
Deployment,
|
||||||
ApplicationDeploymentRequest,
|
ApplicationDeploymentRequest
|
||||||
ApplicationDeploymentRemovalRequest
|
|
||||||
} from './entity/Deployment';
|
} from './entity/Deployment';
|
||||||
import { AppDeploymentRecord, AppDeploymentRemovalRecord, AuctionParams, DeployerRecord } from './types';
|
import { AppDeploymentRecord, PackageJSON } from './types';
|
||||||
import { getConfig, getRepoDetails, registryTransactionWithRetry, sleep } from './utils';
|
|
||||||
|
|
||||||
const log = debug('snowball:registry');
|
const log = debug('snowball:registry');
|
||||||
|
|
||||||
const APP_RECORD_TYPE = 'ApplicationRecord';
|
const APP_RECORD_TYPE = 'ApplicationRecord';
|
||||||
const APP_DEPLOYMENT_AUCTION_RECORD_TYPE = 'ApplicationDeploymentAuction';
|
|
||||||
const APP_DEPLOYMENT_REQUEST_TYPE = 'ApplicationDeploymentRequest';
|
const APP_DEPLOYMENT_REQUEST_TYPE = 'ApplicationDeploymentRequest';
|
||||||
const APP_DEPLOYMENT_REMOVAL_REQUEST_TYPE = 'ApplicationDeploymentRemovalRequest';
|
|
||||||
const APP_DEPLOYMENT_RECORD_TYPE = 'ApplicationDeploymentRecord';
|
const APP_DEPLOYMENT_RECORD_TYPE = 'ApplicationDeploymentRecord';
|
||||||
const APP_DEPLOYMENT_REMOVAL_RECORD_TYPE = 'ApplicationDeploymentRemovalRecord';
|
|
||||||
const WEBAPP_DEPLOYER_RECORD_TYPE = 'WebappDeployer'
|
|
||||||
const SLEEP_DURATION = 1000;
|
|
||||||
|
|
||||||
// TODO: Move registry code to registry-sdk/watcher-ts
|
// TODO: Move registry code to laconic-sdk/watcher-ts
|
||||||
export class Registry {
|
export class Registry {
|
||||||
private registry: LaconicRegistry;
|
private registry: LaconicRegistry;
|
||||||
private registryConfig: RegistryConfig;
|
private registryConfig: RegistryConfig;
|
||||||
|
|
||||||
constructor(registryConfig: RegistryConfig) {
|
constructor (registryConfig: RegistryConfig) {
|
||||||
this.registryConfig = registryConfig;
|
this.registryConfig = registryConfig;
|
||||||
|
|
||||||
const gasPrice = getGasPrice(registryConfig.fee.gasPrice);
|
|
||||||
|
|
||||||
this.registry = new LaconicRegistry(
|
this.registry = new LaconicRegistry(
|
||||||
registryConfig.gqlEndpoint,
|
registryConfig.gqlEndpoint,
|
||||||
registryConfig.restEndpoint,
|
registryConfig.restEndpoint,
|
||||||
{ chainId: registryConfig.chainId, gasPrice }
|
registryConfig.chainId
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
async createApplicationRecord({
|
async createApplicationRecord ({
|
||||||
octokit,
|
appName,
|
||||||
repository,
|
packageJSON,
|
||||||
commitHash,
|
commitHash,
|
||||||
appType,
|
appType,
|
||||||
|
repoUrl
|
||||||
}: {
|
}: {
|
||||||
octokit: Octokit
|
appName: string;
|
||||||
repository: string;
|
packageJSON: PackageJSON;
|
||||||
commitHash: string;
|
commitHash: string;
|
||||||
appType: string;
|
appType: string;
|
||||||
|
repoUrl: string;
|
||||||
}): Promise<{
|
}): Promise<{
|
||||||
applicationRecordId: string;
|
applicationRecordId: string;
|
||||||
applicationRecordData: ApplicationRecord;
|
applicationRecordData: ApplicationRecord;
|
||||||
}> {
|
}> {
|
||||||
const { repo, repoUrl, packageJSON } = await getRepoDetails(octokit, repository, commitHash)
|
// Use laconic-sdk to publish record
|
||||||
// Use registry-sdk to publish record
|
|
||||||
// Reference: https://git.vdb.to/cerc-io/test-progressive-web-app/src/branch/main/scripts/publish-app-record.sh
|
// Reference: https://git.vdb.to/cerc-io/test-progressive-web-app/src/branch/main/scripts/publish-app-record.sh
|
||||||
// Fetch previous records
|
// Fetch previous records
|
||||||
const records = await this.registry.queryRecords(
|
const records = await this.registry.queryRecords(
|
||||||
@ -94,7 +82,7 @@ export class Registry {
|
|||||||
repository_ref: commitHash,
|
repository_ref: commitHash,
|
||||||
repository: [repoUrl],
|
repository: [repoUrl],
|
||||||
app_type: appType,
|
app_type: appType,
|
||||||
name: repo,
|
name: appName,
|
||||||
...(packageJSON.description && { description: packageJSON.description }),
|
...(packageJSON.description && { description: packageJSON.description }),
|
||||||
...(packageJSON.homepage && { homepage: packageJSON.homepage }),
|
...(packageJSON.homepage && { homepage: packageJSON.homepage }),
|
||||||
...(packageJSON.license && { license: packageJSON.license }),
|
...(packageJSON.license && { license: packageJSON.license }),
|
||||||
@ -107,158 +95,63 @@ export class Registry {
|
|||||||
...(packageJSON.version && { app_version: packageJSON.version })
|
...(packageJSON.version && { app_version: packageJSON.version })
|
||||||
};
|
};
|
||||||
|
|
||||||
const fee = parseGasAndFees(this.registryConfig.fee.gas, this.registryConfig.fee.fees);
|
const result = await this.registry.setRecord(
|
||||||
|
{
|
||||||
const result = await registryTransactionWithRetry(() =>
|
privateKey: this.registryConfig.privateKey,
|
||||||
this.registry.setRecord(
|
record: applicationRecord,
|
||||||
{
|
bondId: this.registryConfig.bondId
|
||||||
privateKey: this.registryConfig.privateKey,
|
},
|
||||||
record: applicationRecord,
|
'',
|
||||||
bondId: this.registryConfig.bondId
|
this.registryConfig.fee
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
log(`Published application record ${result.id}`);
|
|
||||||
log('Application record data:', applicationRecord);
|
log('Application record data:', applicationRecord);
|
||||||
|
|
||||||
// TODO: Discuss computation of LRN
|
// TODO: Discuss computation of CRN
|
||||||
const lrn = this.getLrn(repo);
|
const crn = this.getCrn(appName);
|
||||||
log(`Setting name: ${lrn} for record ID: ${result.id}`);
|
log(`Setting name: ${crn} for record ID: ${result.data.id}`);
|
||||||
|
|
||||||
await sleep(SLEEP_DURATION);
|
await this.registry.setName(
|
||||||
await registryTransactionWithRetry(() =>
|
{ cid: result.data.id, crn },
|
||||||
this.registry.setName(
|
this.registryConfig.privateKey,
|
||||||
{
|
this.registryConfig.fee
|
||||||
cid: result.id,
|
|
||||||
lrn
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
await this.registry.setName(
|
||||||
await sleep(SLEEP_DURATION);
|
{ cid: result.data.id, crn: `${crn}@${applicationRecord.app_version}` },
|
||||||
await registryTransactionWithRetry(() =>
|
this.registryConfig.privateKey,
|
||||||
this.registry.setName(
|
this.registryConfig.fee
|
||||||
{
|
|
||||||
cid: result.id,
|
|
||||||
lrn: `${lrn}@${applicationRecord.app_version}`
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
await this.registry.setName(
|
||||||
await sleep(SLEEP_DURATION);
|
{
|
||||||
await registryTransactionWithRetry(() =>
|
cid: result.data.id,
|
||||||
this.registry.setName(
|
crn: `${crn}@${applicationRecord.repository_ref}`
|
||||||
{
|
},
|
||||||
cid: result.id,
|
this.registryConfig.privateKey,
|
||||||
lrn: `${lrn}@${applicationRecord.repository_ref}`
|
this.registryConfig.fee
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
applicationRecordId: result.id,
|
applicationRecordId: result.data.id,
|
||||||
applicationRecordData: applicationRecord
|
applicationRecordData: applicationRecord
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async createApplicationDeploymentAuction(
|
async createApplicationDeploymentRequest (data: {
|
||||||
appName: string,
|
|
||||||
octokit: Octokit,
|
|
||||||
auctionParams: AuctionParams,
|
|
||||||
data: DeepPartial<Deployment>,
|
|
||||||
): Promise<{
|
|
||||||
applicationDeploymentAuctionId: string;
|
|
||||||
}> {
|
|
||||||
assert(data.project?.repository, 'Project repository not found');
|
|
||||||
|
|
||||||
await this.createApplicationRecord({
|
|
||||||
octokit,
|
|
||||||
repository: data.project.repository,
|
|
||||||
appType: data.project!.template!,
|
|
||||||
commitHash: data.commitHash!,
|
|
||||||
});
|
|
||||||
|
|
||||||
const lrn = this.getLrn(appName);
|
|
||||||
const config = await getConfig();
|
|
||||||
const auctionConfig = config.auction;
|
|
||||||
|
|
||||||
const fee = parseGasAndFees(this.registryConfig.fee.gas, this.registryConfig.fee.fees);
|
|
||||||
const auctionResult = await registryTransactionWithRetry(() =>
|
|
||||||
this.registry.createProviderAuction(
|
|
||||||
{
|
|
||||||
commitFee: auctionConfig.commitFee,
|
|
||||||
commitsDuration: auctionConfig.commitsDuration,
|
|
||||||
revealFee: auctionConfig.revealFee,
|
|
||||||
revealsDuration: auctionConfig.revealsDuration,
|
|
||||||
denom: auctionConfig.denom,
|
|
||||||
maxPrice: auctionParams.maxPrice,
|
|
||||||
numProviders: auctionParams.numProviders,
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
if (!auctionResult.auction) {
|
|
||||||
throw new Error('Error creating auction');
|
|
||||||
}
|
|
||||||
|
|
||||||
// Create record of type applicationDeploymentAuction and publish
|
|
||||||
const applicationDeploymentAuction = {
|
|
||||||
application: lrn,
|
|
||||||
auction: auctionResult.auction.id,
|
|
||||||
type: APP_DEPLOYMENT_AUCTION_RECORD_TYPE,
|
|
||||||
};
|
|
||||||
|
|
||||||
const result = await registryTransactionWithRetry(() =>
|
|
||||||
this.registry.setRecord(
|
|
||||||
{
|
|
||||||
privateKey: this.registryConfig.privateKey,
|
|
||||||
record: applicationDeploymentAuction,
|
|
||||||
bondId: this.registryConfig.bondId
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
log(`Application deployment auction created: ${auctionResult.auction.id}`);
|
|
||||||
log(`Application deployment auction record published: ${result.id}`);
|
|
||||||
log('Application deployment auction data:', applicationDeploymentAuction);
|
|
||||||
|
|
||||||
return {
|
|
||||||
applicationDeploymentAuctionId: auctionResult.auction.id,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
async createApplicationDeploymentRequest(data: {
|
|
||||||
deployment: Deployment,
|
deployment: Deployment,
|
||||||
appName: string,
|
appName: string,
|
||||||
|
packageJsonName: string,
|
||||||
repository: string,
|
repository: string,
|
||||||
auctionId?: string | null,
|
environmentVariables: { [key: string]: string }
|
||||||
lrn: string,
|
|
||||||
environmentVariables: { [key: string]: string },
|
|
||||||
dns: string,
|
|
||||||
payment?: string | null
|
|
||||||
}): Promise<{
|
}): Promise<{
|
||||||
applicationDeploymentRequestId: string;
|
applicationDeploymentRequestId: string;
|
||||||
applicationDeploymentRequestData: ApplicationDeploymentRequest;
|
applicationDeploymentRequestData: ApplicationDeploymentRequest;
|
||||||
}> {
|
}> {
|
||||||
const lrn = this.getLrn(data.appName);
|
const crn = this.getCrn(data.appName);
|
||||||
const records = await this.registry.resolveNames([lrn]);
|
const records = await this.registry.resolveNames([crn]);
|
||||||
const applicationRecord = records[0];
|
const applicationRecord = records[0];
|
||||||
|
|
||||||
if (!applicationRecord) {
|
if (!applicationRecord) {
|
||||||
throw new Error(`No record found for ${lrn}`);
|
throw new Error(`No record found for ${crn}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create record of type ApplicationDeploymentRequest and publish
|
// Create record of type ApplicationDeploymentRequest and publish
|
||||||
@ -266,8 +159,11 @@ export class Registry {
|
|||||||
type: APP_DEPLOYMENT_REQUEST_TYPE,
|
type: APP_DEPLOYMENT_REQUEST_TYPE,
|
||||||
version: '1.0.0',
|
version: '1.0.0',
|
||||||
name: `${applicationRecord.attributes.name}@${applicationRecord.attributes.app_version}`,
|
name: `${applicationRecord.attributes.name}@${applicationRecord.attributes.app_version}`,
|
||||||
application: `${lrn}@${applicationRecord.attributes.app_version}`,
|
application: `${crn}@${applicationRecord.attributes.app_version}`,
|
||||||
dns: data.dns,
|
dns: `${data.deployment.project.name}-${data.deployment.id}`,
|
||||||
|
|
||||||
|
// TODO: Not set in test-progressive-web-app CI
|
||||||
|
// deployment: '$CERC_REGISTRY_DEPLOYMENT_CRN',
|
||||||
|
|
||||||
// https://git.vdb.to/cerc-io/laconic-registry-cli/commit/129019105dfb93bebcea02fde0ed64d0f8e5983b
|
// https://git.vdb.to/cerc-io/laconic-registry-cli/commit/129019105dfb93bebcea02fde0ed64d0f8e5983b
|
||||||
config: JSON.stringify({
|
config: JSON.stringify({
|
||||||
@ -279,87 +175,31 @@ export class Registry {
|
|||||||
)}`,
|
)}`,
|
||||||
repository: data.repository,
|
repository: data.repository,
|
||||||
repository_ref: data.deployment.commitHash
|
repository_ref: data.deployment.commitHash
|
||||||
}),
|
})
|
||||||
deployer: data.lrn,
|
|
||||||
...(data.auctionId && { auction: data.auctionId }),
|
|
||||||
...(data.payment && { payment: data.payment }),
|
|
||||||
};
|
};
|
||||||
|
|
||||||
await sleep(SLEEP_DURATION);
|
const result = await this.registry.setRecord(
|
||||||
|
{
|
||||||
const fee = parseGasAndFees(this.registryConfig.fee.gas, this.registryConfig.fee.fees);
|
privateKey: this.registryConfig.privateKey,
|
||||||
|
record: applicationDeploymentRequest,
|
||||||
const result = await registryTransactionWithRetry(() =>
|
bondId: this.registryConfig.bondId
|
||||||
this.registry.setRecord(
|
},
|
||||||
{
|
'',
|
||||||
privateKey: this.registryConfig.privateKey,
|
this.registryConfig.fee
|
||||||
record: applicationDeploymentRequest,
|
|
||||||
bondId: this.registryConfig.bondId
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
log(`Application deployment request record published: ${result.data.id}`);
|
||||||
log(`Application deployment request record published: ${result.id}`);
|
|
||||||
log('Application deployment request data:', applicationDeploymentRequest);
|
log('Application deployment request data:', applicationDeploymentRequest);
|
||||||
|
|
||||||
return {
|
return {
|
||||||
applicationDeploymentRequestId: result.id,
|
applicationDeploymentRequestId: result.data.id,
|
||||||
applicationDeploymentRequestData: applicationDeploymentRequest
|
applicationDeploymentRequestData: applicationDeploymentRequest
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
async getAuctionWinningDeployerRecords(
|
|
||||||
auctionId: string
|
|
||||||
): Promise<DeployerRecord[]> {
|
|
||||||
const records = await this.registry.getAuctionsByIds([auctionId]);
|
|
||||||
const auctionResult = records[0];
|
|
||||||
|
|
||||||
let deployerRecords = [];
|
|
||||||
const { winnerAddresses } = auctionResult;
|
|
||||||
|
|
||||||
for (const auctionWinner of winnerAddresses) {
|
|
||||||
const records = await this.getDeployerRecordsByFilter({
|
|
||||||
paymentAddress: auctionWinner,
|
|
||||||
});
|
|
||||||
|
|
||||||
const newRecords = records.filter(record => {
|
|
||||||
return record.names !== null && record.names.length > 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
for (const record of newRecords) {
|
|
||||||
if (record.id) {
|
|
||||||
deployerRecords.push(record);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return deployerRecords;
|
|
||||||
}
|
|
||||||
|
|
||||||
async releaseDeployerFunds(
|
|
||||||
auctionId: string
|
|
||||||
): Promise<any> {
|
|
||||||
const fee = parseGasAndFees(this.registryConfig.fee.gas, this.registryConfig.fee.fees);
|
|
||||||
const auction = await registryTransactionWithRetry(() =>
|
|
||||||
this.registry.releaseFunds(
|
|
||||||
{
|
|
||||||
auctionId
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
return auction;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Fetch ApplicationDeploymentRecords for deployments
|
* Fetch ApplicationDeploymentRecords for deployments
|
||||||
*/
|
*/
|
||||||
async getDeploymentRecords(
|
async getDeploymentRecords (
|
||||||
deployments: Deployment[]
|
deployments: Deployment[]
|
||||||
): Promise<AppDeploymentRecord[]> {
|
): Promise<AppDeploymentRecord[]> {
|
||||||
// Fetch ApplicationDeploymentRecords for corresponding ApplicationRecord set in deployments
|
// Fetch ApplicationDeploymentRecords for corresponding ApplicationRecord set in deployments
|
||||||
@ -371,165 +211,17 @@ export class Registry {
|
|||||||
true
|
true
|
||||||
);
|
);
|
||||||
|
|
||||||
// Filter records with ApplicationDeploymentRequestId ID and Deployment specific URL
|
// Filter records with ApplicationRecord ids
|
||||||
return records.filter((record: AppDeploymentRecord) =>
|
return records.filter((record: AppDeploymentRecord) =>
|
||||||
deployments.some(
|
deployments.some(
|
||||||
(deployment) =>
|
(deployment) =>
|
||||||
deployment.applicationDeploymentRequestId === record.attributes.request &&
|
deployment.applicationRecordId === record.attributes.application
|
||||||
record.attributes.url.includes(deployment.id)
|
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
getCrn (appName: string): string {
|
||||||
* Fetch WebappDeployer Records by filter
|
|
||||||
*/
|
|
||||||
async getDeployerRecordsByFilter(filter: { [key: string]: any }): Promise<DeployerRecord[]> {
|
|
||||||
return this.registry.queryRecords(
|
|
||||||
{
|
|
||||||
type: WEBAPP_DEPLOYER_RECORD_TYPE,
|
|
||||||
...filter
|
|
||||||
},
|
|
||||||
true
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fetch ApplicationDeploymentRecords by filter
|
|
||||||
*/
|
|
||||||
async getDeploymentRecordsByFilter(filter: { [key: string]: any }): Promise<AppDeploymentRecord[]> {
|
|
||||||
return this.registry.queryRecords(
|
|
||||||
{
|
|
||||||
type: APP_DEPLOYMENT_RECORD_TYPE,
|
|
||||||
...filter
|
|
||||||
},
|
|
||||||
true
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Fetch ApplicationDeploymentRemovalRecords for deployments
|
|
||||||
*/
|
|
||||||
async getDeploymentRemovalRecords(
|
|
||||||
deployments: Deployment[]
|
|
||||||
): Promise<AppDeploymentRemovalRecord[]> {
|
|
||||||
// Fetch ApplicationDeploymentRemovalRecords for corresponding ApplicationDeploymentRecord set in deployments
|
|
||||||
const records = await this.registry.queryRecords(
|
|
||||||
{
|
|
||||||
type: APP_DEPLOYMENT_REMOVAL_RECORD_TYPE
|
|
||||||
},
|
|
||||||
true
|
|
||||||
);
|
|
||||||
|
|
||||||
// Filter records with ApplicationDeploymentRecord and ApplicationDeploymentRemovalRequest IDs
|
|
||||||
return records.filter((record: AppDeploymentRemovalRecord) =>
|
|
||||||
deployments.some(
|
|
||||||
(deployment) =>
|
|
||||||
deployment.applicationDeploymentRemovalRequestId === record.attributes.request &&
|
|
||||||
deployment.applicationDeploymentRecordId === record.attributes.deployment
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
async createApplicationDeploymentRemovalRequest(data: {
|
|
||||||
deploymentId: string;
|
|
||||||
deployerLrn: string;
|
|
||||||
auctionId?: string | null;
|
|
||||||
payment?: string | null;
|
|
||||||
}): Promise<{
|
|
||||||
applicationDeploymentRemovalRequestId: string;
|
|
||||||
applicationDeploymentRemovalRequestData: ApplicationDeploymentRemovalRequest;
|
|
||||||
}> {
|
|
||||||
const applicationDeploymentRemovalRequest = {
|
|
||||||
type: APP_DEPLOYMENT_REMOVAL_REQUEST_TYPE,
|
|
||||||
version: '1.0.0',
|
|
||||||
deployment: data.deploymentId,
|
|
||||||
deployer: data.deployerLrn,
|
|
||||||
...(data.auctionId && { auction: data.auctionId }),
|
|
||||||
...(data.payment && { payment: data.payment }),
|
|
||||||
};
|
|
||||||
|
|
||||||
const fee = parseGasAndFees(this.registryConfig.fee.gas, this.registryConfig.fee.fees);
|
|
||||||
|
|
||||||
const result = await registryTransactionWithRetry(() =>
|
|
||||||
this.registry.setRecord(
|
|
||||||
{
|
|
||||||
privateKey: this.registryConfig.privateKey,
|
|
||||||
record: applicationDeploymentRemovalRequest,
|
|
||||||
bondId: this.registryConfig.bondId
|
|
||||||
},
|
|
||||||
this.registryConfig.privateKey,
|
|
||||||
fee
|
|
||||||
)
|
|
||||||
);
|
|
||||||
|
|
||||||
log(`Application deployment removal request record published: ${result.id}`);
|
|
||||||
log('Application deployment removal request data:', applicationDeploymentRemovalRequest);
|
|
||||||
|
|
||||||
return {
|
|
||||||
applicationDeploymentRemovalRequestId: result.id,
|
|
||||||
applicationDeploymentRemovalRequestData: applicationDeploymentRemovalRequest
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
async getCompletedAuctionIds(auctionIds: string[]): Promise<string[]> {
|
|
||||||
if (auctionIds.length === 0) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
|
|
||||||
const auctions = await this.registry.getAuctionsByIds(auctionIds);
|
|
||||||
|
|
||||||
const completedAuctions = auctions
|
|
||||||
.filter((auction: { id: string, status: string }) => auction.status === 'completed')
|
|
||||||
.map((auction: { id: string, status: string }) => auction.id);
|
|
||||||
|
|
||||||
return completedAuctions;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getRecordsByName(name: string): Promise<any> {
|
|
||||||
return this.registry.resolveNames([name]);
|
|
||||||
}
|
|
||||||
|
|
||||||
async getAuctionData(auctionId: string): Promise<any> {
|
|
||||||
return this.registry.getAuctionsByIds([auctionId]);
|
|
||||||
}
|
|
||||||
|
|
||||||
async sendTokensToAccount(receiverAddress: string, amount: string): Promise<DeliverTxResponse> {
|
|
||||||
const fee = parseGasAndFees(this.registryConfig.fee.gas, this.registryConfig.fee.fees);
|
|
||||||
const account = await this.getAccount();
|
|
||||||
const laconicClient = await this.registry.getLaconicClient(account);
|
|
||||||
const txResponse: DeliverTxResponse =
|
|
||||||
await registryTransactionWithRetry(() =>
|
|
||||||
laconicClient.sendTokens(account.address, receiverAddress,
|
|
||||||
[
|
|
||||||
{
|
|
||||||
denom: 'alnt',
|
|
||||||
amount
|
|
||||||
}
|
|
||||||
],
|
|
||||||
fee || DEFAULT_GAS_ESTIMATION_MULTIPLIER)
|
|
||||||
);
|
|
||||||
|
|
||||||
return txResponse;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getAccount(): Promise<Account> {
|
|
||||||
const account = new Account(Buffer.from(this.registryConfig.privateKey, 'hex'));
|
|
||||||
await account.init();
|
|
||||||
|
|
||||||
return account;
|
|
||||||
}
|
|
||||||
|
|
||||||
async getTxResponse(txHash: string): Promise<IndexedTx | null> {
|
|
||||||
const account = await this.getAccount();
|
|
||||||
const laconicClient = await this.registry.getLaconicClient(account);
|
|
||||||
const txResponse: IndexedTx | null = await laconicClient.getTx(txHash);
|
|
||||||
|
|
||||||
return txResponse;
|
|
||||||
}
|
|
||||||
|
|
||||||
getLrn(appName: string): string {
|
|
||||||
assert(this.registryConfig.authority, "Authority doesn't exist");
|
assert(this.registryConfig.authority, "Authority doesn't exist");
|
||||||
return `lrn://${this.registryConfig.authority}/applications/${appName}`;
|
return `crn://${this.registryConfig.authority}/applications/${appName}`;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -6,7 +6,6 @@ import { Permission } from './entity/ProjectMember';
|
|||||||
import { Domain } from './entity/Domain';
|
import { Domain } from './entity/Domain';
|
||||||
import { Project } from './entity/Project';
|
import { Project } from './entity/Project';
|
||||||
import { EnvironmentVariable } from './entity/EnvironmentVariable';
|
import { EnvironmentVariable } from './entity/EnvironmentVariable';
|
||||||
import { AddProjectFromTemplateInput, AuctionParams, EnvironmentVariables } from './types';
|
|
||||||
|
|
||||||
const log = debug('snowball:resolver');
|
const log = debug('snowball:resolver');
|
||||||
|
|
||||||
@ -18,23 +17,16 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
return context.user;
|
return context.user;
|
||||||
},
|
},
|
||||||
|
|
||||||
organizations: async (_: any, __: any, context: any) => {
|
organizations: async (_:any, __: any, context: any) => {
|
||||||
return service.getOrganizationsByUserId(context.user);
|
return service.getOrganizationsByUserId(context.user);
|
||||||
},
|
},
|
||||||
|
|
||||||
project: async (_: any, { projectId }: { projectId: string }, context: any) => {
|
project: async (_: any, { projectId }: { projectId: string }) => {
|
||||||
return service.getProjectById(context.user, projectId);
|
return service.getProjectById(projectId);
|
||||||
},
|
},
|
||||||
|
|
||||||
projectsInOrganization: async (
|
projectsInOrganization: async (_: any, { organizationSlug }: {organizationSlug: string }, context: any) => {
|
||||||
_: any,
|
return service.getProjectsInOrganization(context.user, organizationSlug);
|
||||||
{ organizationSlug }: { organizationSlug: string },
|
|
||||||
context: any,
|
|
||||||
) => {
|
|
||||||
return service.getProjectsInOrganization(
|
|
||||||
context.user,
|
|
||||||
organizationSlug,
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
|
|
||||||
deployments: async (_: any, { projectId }: { projectId: string }) => {
|
deployments: async (_: any, { projectId }: { projectId: string }) => {
|
||||||
@ -43,7 +35,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
environmentVariables: async (
|
environmentVariables: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectId }: { projectId: string },
|
{ projectId }: { projectId: string }
|
||||||
) => {
|
) => {
|
||||||
return service.getEnvironmentVariablesByProjectId(projectId);
|
return service.getEnvironmentVariablesByProjectId(projectId);
|
||||||
},
|
},
|
||||||
@ -52,11 +44,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
return service.getProjectMembersByProjectId(projectId);
|
return service.getProjectMembersByProjectId(projectId);
|
||||||
},
|
},
|
||||||
|
|
||||||
searchProjects: async (
|
searchProjects: async (_: any, { searchText }: { searchText: string }, context: any) => {
|
||||||
_: any,
|
|
||||||
{ searchText }: { searchText: string },
|
|
||||||
context: any,
|
|
||||||
) => {
|
|
||||||
return service.searchProjects(context.user, searchText);
|
return service.searchProjects(context.user, searchText);
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -64,37 +52,11 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
filter,
|
filter
|
||||||
}: { projectId: string; filter?: FindOptionsWhere<Domain> },
|
}: { projectId: string; filter?: FindOptionsWhere<Domain> }
|
||||||
) => {
|
) => {
|
||||||
return service.getDomainsByProjectId(projectId, filter);
|
return service.getDomainsByProjectId(projectId, filter);
|
||||||
},
|
}
|
||||||
|
|
||||||
getAuctionData: async (
|
|
||||||
_: any,
|
|
||||||
{ auctionId }: { auctionId: string },
|
|
||||||
) => {
|
|
||||||
return service.getAuctionData(auctionId);
|
|
||||||
},
|
|
||||||
|
|
||||||
deployers: async (_: any, __: any, context: any) => {
|
|
||||||
return service.getDeployers();
|
|
||||||
},
|
|
||||||
|
|
||||||
address: async (_: any, __: any, context: any) => {
|
|
||||||
return service.getAddress();
|
|
||||||
},
|
|
||||||
|
|
||||||
verifyTx: async (
|
|
||||||
_: any,
|
|
||||||
{
|
|
||||||
txHash,
|
|
||||||
amount,
|
|
||||||
senderAddress,
|
|
||||||
}: { txHash: string; amount: string; senderAddress: string },
|
|
||||||
) => {
|
|
||||||
return service.verifyTx(txHash, amount, senderAddress);
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
|
|
||||||
// TODO: Return error in GQL response
|
// TODO: Return error in GQL response
|
||||||
@ -102,13 +64,10 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
removeProjectMember: async (
|
removeProjectMember: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectMemberId }: { projectMemberId: string },
|
{ projectMemberId }: { projectMemberId: string },
|
||||||
context: any,
|
context: any
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.removeProjectMember(
|
return await service.removeProjectMember(context.user, projectMemberId);
|
||||||
context.user,
|
|
||||||
projectMemberId,
|
|
||||||
);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
@ -119,13 +78,13 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectMemberId,
|
projectMemberId,
|
||||||
data,
|
data
|
||||||
}: {
|
}: {
|
||||||
projectMemberId: string;
|
projectMemberId: string;
|
||||||
data: {
|
data: {
|
||||||
permissions: Permission[];
|
permissions: Permission[];
|
||||||
};
|
};
|
||||||
},
|
}
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateProjectMember(projectMemberId, data);
|
return await service.updateProjectMember(projectMemberId, data);
|
||||||
@ -139,14 +98,14 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
data,
|
data
|
||||||
}: {
|
}: {
|
||||||
projectId: string;
|
projectId: string;
|
||||||
data: {
|
data: {
|
||||||
email: string;
|
email: string;
|
||||||
permissions: Permission[];
|
permissions: Permission[];
|
||||||
};
|
};
|
||||||
},
|
}
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(await service.addProjectMember(projectId, data));
|
return Boolean(await service.addProjectMember(projectId, data));
|
||||||
@ -160,15 +119,15 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
data,
|
data
|
||||||
}: {
|
}: {
|
||||||
projectId: string;
|
projectId: string;
|
||||||
data: { environments: string[]; key: string; value: string }[];
|
data: { environments: string[]; key: string; value: string }[];
|
||||||
},
|
}
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(
|
return Boolean(
|
||||||
await service.addEnvironmentVariables(projectId, data),
|
await service.addEnvironmentVariables(projectId, data)
|
||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
@ -180,16 +139,16 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
environmentVariableId,
|
environmentVariableId,
|
||||||
data,
|
data
|
||||||
}: {
|
}: {
|
||||||
environmentVariableId: string;
|
environmentVariableId: string;
|
||||||
data: DeepPartial<EnvironmentVariable>;
|
data: DeepPartial<EnvironmentVariable>;
|
||||||
},
|
}
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateEnvironmentVariable(
|
return await service.updateEnvironmentVariable(
|
||||||
environmentVariableId,
|
environmentVariableId,
|
||||||
data,
|
data
|
||||||
);
|
);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
@ -199,7 +158,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
removeEnvironmentVariable: async (
|
removeEnvironmentVariable: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ environmentVariableId }: { environmentVariableId: string },
|
{ environmentVariableId }: { environmentVariableId: string }
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.removeEnvironmentVariable(environmentVariableId);
|
return await service.removeEnvironmentVariable(environmentVariableId);
|
||||||
@ -212,76 +171,26 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
updateDeploymentToProd: async (
|
updateDeploymentToProd: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ deploymentId }: { deploymentId: string },
|
{ deploymentId }: { deploymentId: string },
|
||||||
context: any,
|
context: any
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(
|
return Boolean(await service.updateDeploymentToProd(context.user, deploymentId));
|
||||||
await service.updateDeploymentToProd(context.user, deploymentId),
|
|
||||||
);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
addProjectFromTemplate: async (
|
|
||||||
_: any,
|
|
||||||
{
|
|
||||||
organizationSlug,
|
|
||||||
data,
|
|
||||||
lrn,
|
|
||||||
auctionParams,
|
|
||||||
environmentVariables
|
|
||||||
}: {
|
|
||||||
organizationSlug: string;
|
|
||||||
data: AddProjectFromTemplateInput;
|
|
||||||
lrn: string;
|
|
||||||
auctionParams: AuctionParams;
|
|
||||||
environmentVariables: EnvironmentVariables[];
|
|
||||||
},
|
|
||||||
context: any,
|
|
||||||
) => {
|
|
||||||
try {
|
|
||||||
return await service.addProjectFromTemplate(
|
|
||||||
context.user,
|
|
||||||
organizationSlug,
|
|
||||||
data,
|
|
||||||
lrn,
|
|
||||||
auctionParams,
|
|
||||||
environmentVariables
|
|
||||||
);
|
|
||||||
} catch (err) {
|
|
||||||
log(err);
|
|
||||||
throw err;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
addProject: async (
|
addProject: async (
|
||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
organizationSlug,
|
organizationSlug,
|
||||||
data,
|
data
|
||||||
lrn,
|
}: { organizationSlug: string; data: DeepPartial<Project> },
|
||||||
auctionParams,
|
context: any
|
||||||
environmentVariables
|
|
||||||
}: {
|
|
||||||
organizationSlug: string;
|
|
||||||
data: DeepPartial<Project>;
|
|
||||||
lrn: string;
|
|
||||||
auctionParams: AuctionParams,
|
|
||||||
environmentVariables: EnvironmentVariables[];
|
|
||||||
},
|
|
||||||
context: any,
|
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.addProject(
|
return await service.addProject(context.user, organizationSlug, data);
|
||||||
context.user,
|
|
||||||
organizationSlug,
|
|
||||||
data,
|
|
||||||
lrn,
|
|
||||||
auctionParams,
|
|
||||||
environmentVariables
|
|
||||||
);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
throw err;
|
throw err;
|
||||||
@ -290,7 +199,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
updateProject: async (
|
updateProject: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectId, data }: { projectId: string; data: DeepPartial<Project> },
|
{ projectId, data }: { projectId: string; data: DeepPartial<Project> }
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateProject(projectId, data);
|
return await service.updateProject(projectId, data);
|
||||||
@ -303,12 +212,10 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
redeployToProd: async (
|
redeployToProd: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ deploymentId }: { deploymentId: string },
|
{ deploymentId }: { deploymentId: string },
|
||||||
context: any,
|
context: any
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(
|
return Boolean(await service.redeployToProd(context.user, deploymentId));
|
||||||
await service.redeployToProd(context.user, deploymentId),
|
|
||||||
);
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
@ -337,8 +244,8 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
_: any,
|
_: any,
|
||||||
{
|
{
|
||||||
projectId,
|
projectId,
|
||||||
deploymentId,
|
deploymentId
|
||||||
}: { deploymentId: string; projectId: string },
|
}: { deploymentId: string; projectId: string }
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.rollbackDeployment(projectId, deploymentId);
|
return await service.rollbackDeployment(projectId, deploymentId);
|
||||||
@ -348,21 +255,9 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
deleteDeployment: async (
|
|
||||||
_: any,
|
|
||||||
{ deploymentId }: { deploymentId: string },
|
|
||||||
) => {
|
|
||||||
try {
|
|
||||||
return await service.deleteDeployment(deploymentId);
|
|
||||||
} catch (err) {
|
|
||||||
log(err);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
},
|
|
||||||
|
|
||||||
addDomain: async (
|
addDomain: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ projectId, data }: { projectId: string; data: { name: string } },
|
{ projectId, data }: { projectId: string; data: { name: string } }
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return Boolean(await service.addDomain(projectId, data));
|
return Boolean(await service.addDomain(projectId, data));
|
||||||
@ -374,7 +269,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
updateDomain: async (
|
updateDomain: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ domainId, data }: { domainId: string; data: DeepPartial<Domain> },
|
{ domainId, data }: { domainId: string; data: DeepPartial<Domain> }
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.updateDomain(domainId, data);
|
return await service.updateDomain(domainId, data);
|
||||||
@ -387,7 +282,7 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
authenticateGitHub: async (
|
authenticateGitHub: async (
|
||||||
_: any,
|
_: any,
|
||||||
{ code }: { code: string },
|
{ code }: { code: string },
|
||||||
context: any,
|
context: any
|
||||||
) => {
|
) => {
|
||||||
try {
|
try {
|
||||||
return await service.authenticateGitHub(code, context.user);
|
return await service.authenticateGitHub(code, context.user);
|
||||||
@ -399,14 +294,12 @@ export const createResolvers = async (service: Service): Promise<any> => {
|
|||||||
|
|
||||||
unauthenticateGitHub: async (_: any, __: object, context: any) => {
|
unauthenticateGitHub: async (_: any, __: object, context: any) => {
|
||||||
try {
|
try {
|
||||||
return service.unauthenticateGitHub(context.user, {
|
return service.unauthenticateGitHub(context.user, { gitHubToken: null });
|
||||||
gitHubToken: null,
|
|
||||||
});
|
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
log(err);
|
log(err);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
},
|
}
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
@ -1,94 +1,29 @@
|
|||||||
import { Router } from 'express';
|
import { Router } from 'express';
|
||||||
import { SiweMessage } from 'siwe';
|
import { SiweMessage, generateNonce } from 'siwe';
|
||||||
import { Service } from '../service';
|
|
||||||
import { authenticateUser, createUser } from '../turnkey-backend';
|
|
||||||
|
|
||||||
const router = Router();
|
const router = Router();
|
||||||
|
|
||||||
//
|
router.get('/nonce', async (_, res) => {
|
||||||
// Turnkey
|
res.send(generateNonce());
|
||||||
//
|
|
||||||
router.get('/registration/:email', async (req, res) => {
|
|
||||||
const service: Service = req.app.get('service');
|
|
||||||
const user = await service.getUserByEmail(req.params.email);
|
|
||||||
if (user) {
|
|
||||||
return res.send({ subOrganizationId: user?.subOrgId });
|
|
||||||
} else {
|
|
||||||
return res.sendStatus(204);
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
|
|
||||||
router.post('/register', async (req, res) => {
|
|
||||||
console.log('Register', req.body);
|
|
||||||
const { email, challenge, attestation } = req.body;
|
|
||||||
const user = await createUser(req.app.get('service'), {
|
|
||||||
challenge,
|
|
||||||
attestation,
|
|
||||||
userEmail: email,
|
|
||||||
userName: email.split('@')[0],
|
|
||||||
});
|
|
||||||
req.session.address = user.id;
|
|
||||||
res.sendStatus(200);
|
|
||||||
});
|
|
||||||
|
|
||||||
router.post('/authenticate', async (req, res) => {
|
|
||||||
console.log('Authenticate', req.body);
|
|
||||||
const { signedWhoamiRequest } = req.body;
|
|
||||||
const user = await authenticateUser(
|
|
||||||
req.app.get('service'),
|
|
||||||
signedWhoamiRequest,
|
|
||||||
);
|
|
||||||
if (user) {
|
|
||||||
req.session.address = user.id;
|
|
||||||
res.sendStatus(200);
|
|
||||||
} else {
|
|
||||||
res.sendStatus(401);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
//
|
|
||||||
// SIWE Auth
|
|
||||||
//
|
|
||||||
router.post('/validate', async (req, res) => {
|
router.post('/validate', async (req, res) => {
|
||||||
const { message, signature } = req.body;
|
const { message, signature } = req.body;
|
||||||
const { success, data } = await new SiweMessage(message).verify({
|
const { success, data } = await new SiweMessage(message).verify({
|
||||||
signature,
|
signature
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!success) {
|
if (success) {
|
||||||
return res.send({ success });
|
req.session.address = data.address;
|
||||||
}
|
|
||||||
const service: Service = req.app.get('service');
|
|
||||||
const user = await service.getUserByEthAddress(data.address);
|
|
||||||
|
|
||||||
if (!user) {
|
|
||||||
const newUser = await service.createUser({
|
|
||||||
ethAddress: data.address,
|
|
||||||
email: `${data.address}@example.com`,
|
|
||||||
subOrgId: '',
|
|
||||||
turnkeyWalletId: '',
|
|
||||||
});
|
|
||||||
|
|
||||||
// SIWESession from the web3modal library requires both address and chain ID
|
|
||||||
req.session.address = newUser.id;
|
|
||||||
req.session.chainId = data.chainId;
|
|
||||||
} else {
|
|
||||||
req.session.address = user.id;
|
|
||||||
req.session.chainId = data.chainId;
|
req.session.chainId = data.chainId;
|
||||||
}
|
}
|
||||||
|
|
||||||
res.send({ success });
|
res.send({ success });
|
||||||
});
|
});
|
||||||
|
|
||||||
//
|
|
||||||
// General
|
|
||||||
//
|
|
||||||
router.get('/session', (req, res) => {
|
router.get('/session', (req, res) => {
|
||||||
if (req.session.address && req.session.chainId) {
|
if (req.session.address && req.session.chainId) {
|
||||||
res.send({
|
res.send({ address: req.session.address, chainId: req.session.chainId });
|
||||||
address: req.session.address,
|
|
||||||
chainId: req.session.chainId
|
|
||||||
});
|
|
||||||
} else {
|
} else {
|
||||||
res.status(401).send({ error: 'Unauthorized: No active session' });
|
res.status(401).send({ error: 'Unauthorized: No active session' });
|
||||||
}
|
}
|
||||||
|
@ -1,9 +0,0 @@
|
|||||||
import { Router } from 'express';
|
|
||||||
|
|
||||||
const router = Router();
|
|
||||||
|
|
||||||
router.get('/version', async (req, res) => {
|
|
||||||
return res.send({ version: '0.0.9' });
|
|
||||||
});
|
|
||||||
|
|
||||||
export default router;
|
|
@ -19,14 +19,6 @@ enum DeploymentStatus {
|
|||||||
Building
|
Building
|
||||||
Ready
|
Ready
|
||||||
Error
|
Error
|
||||||
Deleting
|
|
||||||
}
|
|
||||||
|
|
||||||
enum AuctionStatus {
|
|
||||||
completed
|
|
||||||
reveal
|
|
||||||
commit
|
|
||||||
expired
|
|
||||||
}
|
}
|
||||||
|
|
||||||
enum DomainStatus {
|
enum DomainStatus {
|
||||||
@ -72,13 +64,8 @@ type Project {
|
|||||||
repository: String!
|
repository: String!
|
||||||
prodBranch: String!
|
prodBranch: String!
|
||||||
description: String
|
description: String
|
||||||
deployers: [Deployer!]
|
|
||||||
auctionId: String
|
|
||||||
fundsReleased: Boolean
|
|
||||||
template: String
|
template: String
|
||||||
framework: String
|
framework: String
|
||||||
paymentAddress: String!
|
|
||||||
txHash: String!
|
|
||||||
webhooks: [String!]
|
webhooks: [String!]
|
||||||
members: [ProjectMember!]
|
members: [ProjectMember!]
|
||||||
environmentVariables: [EnvironmentVariable!]
|
environmentVariables: [EnvironmentVariable!]
|
||||||
@ -86,7 +73,7 @@ type Project {
|
|||||||
updatedAt: String!
|
updatedAt: String!
|
||||||
organization: Organization!
|
organization: Organization!
|
||||||
icon: String
|
icon: String
|
||||||
baseDomains: [String!]
|
subDomain: String
|
||||||
}
|
}
|
||||||
|
|
||||||
type ProjectMember {
|
type ProjectMember {
|
||||||
@ -106,10 +93,7 @@ type Deployment {
|
|||||||
commitMessage: String!
|
commitMessage: String!
|
||||||
url: String
|
url: String
|
||||||
environment: Environment!
|
environment: Environment!
|
||||||
deployer: Deployer
|
|
||||||
applicationDeploymentRequestId: String
|
|
||||||
isCurrent: Boolean!
|
isCurrent: Boolean!
|
||||||
baseDomain: String
|
|
||||||
status: DeploymentStatus!
|
status: DeploymentStatus!
|
||||||
createdAt: String!
|
createdAt: String!
|
||||||
updatedAt: String!
|
updatedAt: String!
|
||||||
@ -135,17 +119,6 @@ type EnvironmentVariable {
|
|||||||
updatedAt: String!
|
updatedAt: String!
|
||||||
}
|
}
|
||||||
|
|
||||||
type Deployer {
|
|
||||||
deployerLrn: String!
|
|
||||||
deployerId: String!
|
|
||||||
deployerApiUrl: String!
|
|
||||||
minimumPayment: String
|
|
||||||
paymentAddress: String
|
|
||||||
createdAt: String!
|
|
||||||
updatedAt: String!
|
|
||||||
baseDomain: String
|
|
||||||
}
|
|
||||||
|
|
||||||
type AuthResult {
|
type AuthResult {
|
||||||
token: String!
|
token: String!
|
||||||
}
|
}
|
||||||
@ -156,23 +129,11 @@ input AddEnvironmentVariableInput {
|
|||||||
value: String!
|
value: String!
|
||||||
}
|
}
|
||||||
|
|
||||||
input AddProjectFromTemplateInput {
|
|
||||||
templateOwner: String!
|
|
||||||
templateRepo: String!
|
|
||||||
owner: String!
|
|
||||||
name: String!
|
|
||||||
isPrivate: Boolean!
|
|
||||||
paymentAddress: String!
|
|
||||||
txHash: String!
|
|
||||||
}
|
|
||||||
|
|
||||||
input AddProjectInput {
|
input AddProjectInput {
|
||||||
name: String!
|
name: String!
|
||||||
repository: String!
|
repository: String!
|
||||||
prodBranch: String!
|
prodBranch: String!
|
||||||
template: String
|
template: String
|
||||||
paymentAddress: String!
|
|
||||||
txHash: String!
|
|
||||||
}
|
}
|
||||||
|
|
||||||
input UpdateProjectInput {
|
input UpdateProjectInput {
|
||||||
@ -212,48 +173,6 @@ input FilterDomainsInput {
|
|||||||
status: DomainStatus
|
status: DomainStatus
|
||||||
}
|
}
|
||||||
|
|
||||||
type Fee {
|
|
||||||
type: String!
|
|
||||||
quantity: String!
|
|
||||||
}
|
|
||||||
|
|
||||||
type Bid {
|
|
||||||
auctionId: String!
|
|
||||||
bidderAddress: String!
|
|
||||||
status: String!
|
|
||||||
commitHash: String!
|
|
||||||
commitTime: String
|
|
||||||
commitFee: Fee
|
|
||||||
revealTime: String
|
|
||||||
revealFee: Fee
|
|
||||||
bidAmount: Fee
|
|
||||||
}
|
|
||||||
|
|
||||||
type Auction {
|
|
||||||
id: String!
|
|
||||||
kind: String!
|
|
||||||
status: String!
|
|
||||||
ownerAddress: String!
|
|
||||||
createTime: String!
|
|
||||||
commitsEndTime: String!
|
|
||||||
revealsEndTime: String!
|
|
||||||
commitFee: Fee!
|
|
||||||
revealFee: Fee!
|
|
||||||
minimumBid: Fee
|
|
||||||
winnerAddresses: [String!]!
|
|
||||||
winnerBids: [Fee!]
|
|
||||||
winnerPrice: Fee
|
|
||||||
maxPrice: Fee
|
|
||||||
numProviders: Int!
|
|
||||||
fundsReleased: Boolean!
|
|
||||||
bids: [Bid!]!
|
|
||||||
}
|
|
||||||
|
|
||||||
input AuctionParams {
|
|
||||||
maxPrice: String,
|
|
||||||
numProviders: Int,
|
|
||||||
}
|
|
||||||
|
|
||||||
type Query {
|
type Query {
|
||||||
user: User!
|
user: User!
|
||||||
organizations: [Organization!]
|
organizations: [Organization!]
|
||||||
@ -264,11 +183,7 @@ type Query {
|
|||||||
environmentVariables(projectId: String!): [EnvironmentVariable!]
|
environmentVariables(projectId: String!): [EnvironmentVariable!]
|
||||||
projectMembers(projectId: String!): [ProjectMember!]
|
projectMembers(projectId: String!): [ProjectMember!]
|
||||||
searchProjects(searchText: String!): [Project!]
|
searchProjects(searchText: String!): [Project!]
|
||||||
getAuctionData(auctionId: String!): Auction!
|
|
||||||
domains(projectId: String!, filter: FilterDomainsInput): [Domain]
|
domains(projectId: String!, filter: FilterDomainsInput): [Domain]
|
||||||
deployers: [Deployer]
|
|
||||||
address: String!
|
|
||||||
verifyTx(txHash: String!, amount: String!, senderAddress: String!): Boolean!
|
|
||||||
}
|
}
|
||||||
|
|
||||||
type Mutation {
|
type Mutation {
|
||||||
@ -288,26 +203,12 @@ type Mutation {
|
|||||||
): Boolean!
|
): Boolean!
|
||||||
removeEnvironmentVariable(environmentVariableId: String!): Boolean!
|
removeEnvironmentVariable(environmentVariableId: String!): Boolean!
|
||||||
updateDeploymentToProd(deploymentId: String!): Boolean!
|
updateDeploymentToProd(deploymentId: String!): Boolean!
|
||||||
addProjectFromTemplate(
|
addProject(organizationSlug: String!, data: AddProjectInput): Project!
|
||||||
organizationSlug: String!
|
|
||||||
data: AddProjectFromTemplateInput
|
|
||||||
lrn: String
|
|
||||||
auctionParams: AuctionParams
|
|
||||||
environmentVariables: [AddEnvironmentVariableInput!]
|
|
||||||
): Project!
|
|
||||||
addProject(
|
|
||||||
organizationSlug: String!
|
|
||||||
data: AddProjectInput!
|
|
||||||
lrn: String
|
|
||||||
auctionParams: AuctionParams
|
|
||||||
environmentVariables: [AddEnvironmentVariableInput!]
|
|
||||||
): Project!
|
|
||||||
updateProject(projectId: String!, data: UpdateProjectInput): Boolean!
|
updateProject(projectId: String!, data: UpdateProjectInput): Boolean!
|
||||||
redeployToProd(deploymentId: String!): Boolean!
|
redeployToProd(deploymentId: String!): Boolean!
|
||||||
deleteProject(projectId: String!): Boolean!
|
deleteProject(projectId: String!): Boolean!
|
||||||
deleteDomain(domainId: String!): Boolean!
|
deleteDomain(domainId: String!): Boolean!
|
||||||
rollbackDeployment(projectId: String!, deploymentId: String!): Boolean!
|
rollbackDeployment(projectId: String!, deploymentId: String!): Boolean!
|
||||||
deleteDeployment(deploymentId: String!): Boolean!
|
|
||||||
addDomain(projectId: String!, data: AddDomainInput!): Boolean!
|
addDomain(projectId: String!, data: AddDomainInput!): Boolean!
|
||||||
updateDomain(domainId: String!, data: UpdateDomainInput!): Boolean!
|
updateDomain(domainId: String!, data: UpdateDomainInput!): Boolean!
|
||||||
authenticateGitHub(code: String!): AuthResult!
|
authenticateGitHub(code: String!): AuthResult!
|
||||||
|
@ -6,7 +6,7 @@ import { createServer } from 'http';
|
|||||||
import {
|
import {
|
||||||
ApolloServerPluginDrainHttpServer,
|
ApolloServerPluginDrainHttpServer,
|
||||||
ApolloServerPluginLandingPageLocalDefault,
|
ApolloServerPluginLandingPageLocalDefault,
|
||||||
AuthenticationError,
|
AuthenticationError
|
||||||
} from 'apollo-server-core';
|
} from 'apollo-server-core';
|
||||||
import session from 'express-session';
|
import session from 'express-session';
|
||||||
|
|
||||||
@ -17,14 +17,10 @@ import { ServerConfig } from './config';
|
|||||||
import { DEFAULT_GQL_PATH } from './constants';
|
import { DEFAULT_GQL_PATH } from './constants';
|
||||||
import githubRouter from './routes/github';
|
import githubRouter from './routes/github';
|
||||||
import authRouter from './routes/auth';
|
import authRouter from './routes/auth';
|
||||||
import stagingRouter from './routes/staging';
|
|
||||||
import { Service } from './service';
|
import { Service } from './service';
|
||||||
|
|
||||||
const log = debug('snowball:server');
|
const log = debug('snowball:server');
|
||||||
|
|
||||||
// Set cookie expiration to 1 month in milliseconds
|
|
||||||
const COOKIE_MAX_AGE = 30 * 24 * 60 * 60 * 1000;
|
|
||||||
|
|
||||||
declare module 'express-session' {
|
declare module 'express-session' {
|
||||||
interface SessionData {
|
interface SessionData {
|
||||||
address: string;
|
address: string;
|
||||||
@ -36,7 +32,7 @@ export const createAndStartServer = async (
|
|||||||
serverConfig: ServerConfig,
|
serverConfig: ServerConfig,
|
||||||
typeDefs: TypeSource,
|
typeDefs: TypeSource,
|
||||||
resolvers: any,
|
resolvers: any,
|
||||||
service: Service,
|
service: Service
|
||||||
): Promise<ApolloServer> => {
|
): Promise<ApolloServer> => {
|
||||||
const { host, port, gqlPath = DEFAULT_GQL_PATH } = serverConfig;
|
const { host, port, gqlPath = DEFAULT_GQL_PATH } = serverConfig;
|
||||||
const { appOriginUrl, secret, domain, trustProxy } = serverConfig.session;
|
const { appOriginUrl, secret, domain, trustProxy } = serverConfig.session;
|
||||||
@ -49,7 +45,7 @@ export const createAndStartServer = async (
|
|||||||
// Create the schema
|
// Create the schema
|
||||||
const schema = makeExecutableSchema({
|
const schema = makeExecutableSchema({
|
||||||
typeDefs,
|
typeDefs,
|
||||||
resolvers,
|
resolvers
|
||||||
});
|
});
|
||||||
|
|
||||||
const server = new ApolloServer({
|
const server = new ApolloServer({
|
||||||
@ -64,24 +60,24 @@ export const createAndStartServer = async (
|
|||||||
throw new AuthenticationError('Unauthorized: No active session');
|
throw new AuthenticationError('Unauthorized: No active session');
|
||||||
}
|
}
|
||||||
|
|
||||||
const user = await service.getUser(address);
|
// Find/create user from ETH address in request session
|
||||||
|
const user = await service.loadOrCreateUser(address);
|
||||||
|
|
||||||
return { user };
|
return { user };
|
||||||
},
|
},
|
||||||
plugins: [
|
plugins: [
|
||||||
// Proper shutdown for the HTTP server
|
// Proper shutdown for the HTTP server
|
||||||
ApolloServerPluginDrainHttpServer({ httpServer }),
|
ApolloServerPluginDrainHttpServer({ httpServer }),
|
||||||
ApolloServerPluginLandingPageLocalDefault({ embed: true }),
|
ApolloServerPluginLandingPageLocalDefault({ embed: true })
|
||||||
],
|
]
|
||||||
});
|
});
|
||||||
|
|
||||||
await server.start();
|
await server.start();
|
||||||
|
|
||||||
app.use(
|
app.use(cors({
|
||||||
cors({
|
origin: appOriginUrl,
|
||||||
origin: appOriginUrl,
|
credentials: true
|
||||||
credentials: true,
|
}));
|
||||||
}),
|
|
||||||
);
|
|
||||||
|
|
||||||
const sessionOptions: session.SessionOptions = {
|
const sessionOptions: session.SessionOptions = {
|
||||||
secret: secret,
|
secret: secret,
|
||||||
@ -89,12 +85,16 @@ export const createAndStartServer = async (
|
|||||||
saveUninitialized: true,
|
saveUninitialized: true,
|
||||||
cookie: {
|
cookie: {
|
||||||
secure: new URL(appOriginUrl).protocol === 'https:',
|
secure: new URL(appOriginUrl).protocol === 'https:',
|
||||||
maxAge: COOKIE_MAX_AGE,
|
// TODO: Set cookie maxAge and handle cookie expiry in frontend
|
||||||
domain: domain || undefined,
|
// maxAge: SESSION_COOKIE_MAX_AGE,
|
||||||
sameSite: new URL(appOriginUrl).protocol === 'https:' ? 'none' : 'lax',
|
sameSite: new URL(appOriginUrl).protocol === 'https:' ? 'none' : 'lax'
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
if (domain) {
|
||||||
|
sessionOptions.cookie!.domain = domain;
|
||||||
|
}
|
||||||
|
|
||||||
if (trustProxy) {
|
if (trustProxy) {
|
||||||
// trust first proxy
|
// trust first proxy
|
||||||
app.set('trust proxy', 1);
|
app.set('trust proxy', 1);
|
||||||
@ -109,8 +109,8 @@ export const createAndStartServer = async (
|
|||||||
path: gqlPath,
|
path: gqlPath,
|
||||||
cors: {
|
cors: {
|
||||||
origin: [appOriginUrl],
|
origin: [appOriginUrl],
|
||||||
credentials: true,
|
credentials: true
|
||||||
},
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
app.use(express.json());
|
app.use(express.json());
|
||||||
@ -118,12 +118,6 @@ export const createAndStartServer = async (
|
|||||||
app.set('service', service);
|
app.set('service', service);
|
||||||
app.use('/auth', authRouter);
|
app.use('/auth', authRouter);
|
||||||
app.use('/api/github', githubRouter);
|
app.use('/api/github', githubRouter);
|
||||||
app.use('/staging', stagingRouter);
|
|
||||||
|
|
||||||
app.use((err: any, req: any, res: any, next: any) => {
|
|
||||||
console.error(err);
|
|
||||||
res.status(500).json({ error: err.message });
|
|
||||||
});
|
|
||||||
|
|
||||||
httpServer.listen(port, host, () => {
|
httpServer.listen(port, host, () => {
|
||||||
log(`Server is listening on ${host}:${port}${server.graphqlPath}`);
|
log(`Server is listening on ${host}:${port}${server.graphqlPath}`);
|
||||||
|
File diff suppressed because it is too large
Load Diff
@ -1,130 +0,0 @@
|
|||||||
import { Turnkey, TurnkeyApiTypes } from '@turnkey/sdk-server';
|
|
||||||
|
|
||||||
// Default path for the first Ethereum address in a new HD wallet.
|
|
||||||
// See https://github.com/bitcoin/bips/blob/master/bip-0044.mediawiki, paths are in the form:
|
|
||||||
// m / purpose' / coin_type' / account' / change / address_index
|
|
||||||
// - Purpose is a constant set to 44' following the BIP43 recommendation.
|
|
||||||
// - Coin type is set to 60 (ETH) -- see https://github.com/satoshilabs/slips/blob/master/slip-0044.md
|
|
||||||
// - Account, Change, and Address Index are set to 0
|
|
||||||
import { DEFAULT_ETHEREUM_ACCOUNTS } from '@turnkey/sdk-server';
|
|
||||||
import { getConfig } from './utils';
|
|
||||||
import { Service } from './service';
|
|
||||||
|
|
||||||
type TAttestation = TurnkeyApiTypes['v1Attestation'];
|
|
||||||
|
|
||||||
type CreateUserParams = {
|
|
||||||
userName: string;
|
|
||||||
userEmail: string;
|
|
||||||
challenge: string;
|
|
||||||
attestation: TAttestation;
|
|
||||||
};
|
|
||||||
|
|
||||||
export async function createUser(
|
|
||||||
service: Service,
|
|
||||||
{ userName, userEmail, challenge, attestation }: CreateUserParams,
|
|
||||||
) {
|
|
||||||
try {
|
|
||||||
if (await service.getUserByEmail(userEmail)) {
|
|
||||||
throw new Error(`User already exists: ${userEmail}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const config = await getConfig();
|
|
||||||
const turnkey = new Turnkey(config.turnkey);
|
|
||||||
|
|
||||||
const apiClient = turnkey.api();
|
|
||||||
|
|
||||||
const walletName = `Default ETH Wallet`;
|
|
||||||
|
|
||||||
const createSubOrgResponse = await apiClient.createSubOrganization({
|
|
||||||
subOrganizationName: `Default SubOrg for ${userEmail}`,
|
|
||||||
rootQuorumThreshold: 1,
|
|
||||||
rootUsers: [
|
|
||||||
{
|
|
||||||
userName,
|
|
||||||
userEmail,
|
|
||||||
apiKeys: [],
|
|
||||||
authenticators: [
|
|
||||||
{
|
|
||||||
authenticatorName: 'Passkey',
|
|
||||||
challenge,
|
|
||||||
attestation,
|
|
||||||
},
|
|
||||||
],
|
|
||||||
},
|
|
||||||
],
|
|
||||||
wallet: {
|
|
||||||
walletName: walletName,
|
|
||||||
accounts: DEFAULT_ETHEREUM_ACCOUNTS,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const subOrgId = refineNonNull(createSubOrgResponse.subOrganizationId);
|
|
||||||
const wallet = refineNonNull(createSubOrgResponse.wallet);
|
|
||||||
|
|
||||||
const result = {
|
|
||||||
id: wallet.walletId,
|
|
||||||
address: wallet.addresses[0],
|
|
||||||
subOrgId: subOrgId,
|
|
||||||
};
|
|
||||||
console.log('Turnkey success', result);
|
|
||||||
|
|
||||||
const user = await service.createUser({
|
|
||||||
name: userName,
|
|
||||||
email: userEmail,
|
|
||||||
subOrgId,
|
|
||||||
ethAddress: wallet.addresses[0],
|
|
||||||
turnkeyWalletId: wallet.walletId,
|
|
||||||
});
|
|
||||||
console.log('New user', user);
|
|
||||||
|
|
||||||
return user;
|
|
||||||
} catch (e) {
|
|
||||||
console.error('Failed to create user:', e);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
export async function authenticateUser(
|
|
||||||
service: Service,
|
|
||||||
signedWhoamiRequest: {
|
|
||||||
url: string;
|
|
||||||
body: any;
|
|
||||||
stamp: {
|
|
||||||
stampHeaderName: string;
|
|
||||||
stampHeaderValue: string;
|
|
||||||
};
|
|
||||||
},
|
|
||||||
) {
|
|
||||||
try {
|
|
||||||
const tkRes = await fetch(signedWhoamiRequest.url, {
|
|
||||||
method: 'POST',
|
|
||||||
body: signedWhoamiRequest.body,
|
|
||||||
headers: {
|
|
||||||
[signedWhoamiRequest.stamp.stampHeaderName]:
|
|
||||||
signedWhoamiRequest.stamp.stampHeaderValue,
|
|
||||||
},
|
|
||||||
});
|
|
||||||
console.log('AUTH RESULT', tkRes.status);
|
|
||||||
if (tkRes.status !== 200) {
|
|
||||||
console.log(await tkRes.text());
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
const orgId = (await tkRes.json()).organizationId;
|
|
||||||
const user = await service.getUserBySubOrgId(orgId);
|
|
||||||
return user;
|
|
||||||
} catch (e) {
|
|
||||||
console.error('Failed to authenticate:', e);
|
|
||||||
throw e;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function refineNonNull<T>(
|
|
||||||
input: T | null | undefined,
|
|
||||||
errorMessage?: string,
|
|
||||||
): T {
|
|
||||||
if (input == null) {
|
|
||||||
throw new Error(errorMessage ?? `Unexpected ${JSON.stringify(input)}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
return input;
|
|
||||||
}
|
|
@ -24,13 +24,10 @@ export interface GitPushEventPayload {
|
|||||||
id: string;
|
id: string;
|
||||||
message: string;
|
message: string;
|
||||||
};
|
};
|
||||||
deleted: boolean;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AppDeploymentRecordAttributes {
|
export interface AppDeploymentRecordAttributes {
|
||||||
application: string;
|
application: string;
|
||||||
auction: string;
|
|
||||||
deployer: string;
|
|
||||||
dns: string;
|
dns: string;
|
||||||
meta: string;
|
meta: string;
|
||||||
name: string;
|
name: string;
|
||||||
@ -40,13 +37,6 @@ export interface AppDeploymentRecordAttributes {
|
|||||||
version: string;
|
version: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AppDeploymentRemovalRecordAttributes {
|
|
||||||
deployment: string;
|
|
||||||
request: string;
|
|
||||||
type: 'ApplicationDeploymentRemovalRecord';
|
|
||||||
version: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface RegistryRecord {
|
interface RegistryRecord {
|
||||||
id: string;
|
id: string;
|
||||||
names: string[] | null;
|
names: string[] | null;
|
||||||
@ -59,46 +49,3 @@ interface RegistryRecord {
|
|||||||
export interface AppDeploymentRecord extends RegistryRecord {
|
export interface AppDeploymentRecord extends RegistryRecord {
|
||||||
attributes: AppDeploymentRecordAttributes;
|
attributes: AppDeploymentRecordAttributes;
|
||||||
}
|
}
|
||||||
|
|
||||||
export interface AppDeploymentRemovalRecord extends RegistryRecord {
|
|
||||||
attributes: AppDeploymentRemovalRecordAttributes;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface AddProjectFromTemplateInput {
|
|
||||||
templateOwner: string;
|
|
||||||
templateRepo: string;
|
|
||||||
owner: string;
|
|
||||||
name: string;
|
|
||||||
isPrivate: boolean;
|
|
||||||
paymentAddress: string;
|
|
||||||
txHash: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface AuctionParams {
|
|
||||||
maxPrice: string,
|
|
||||||
numProviders: number,
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface EnvironmentVariables {
|
|
||||||
environments: string[],
|
|
||||||
key: string,
|
|
||||||
value: string,
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface DeployerRecord {
|
|
||||||
id: string;
|
|
||||||
names: string[];
|
|
||||||
owners: string[];
|
|
||||||
bondId: string;
|
|
||||||
createTime: string;
|
|
||||||
expiryTime: string;
|
|
||||||
attributes: {
|
|
||||||
apiUrl: string;
|
|
||||||
minimumPayment: string | null;
|
|
||||||
name: string;
|
|
||||||
paymentAddress: string;
|
|
||||||
publicKey: string;
|
|
||||||
type: string;
|
|
||||||
version: string;
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
@ -1,24 +1,13 @@
|
|||||||
import assert from 'assert';
|
|
||||||
import debug from 'debug';
|
|
||||||
import fs from 'fs-extra';
|
import fs from 'fs-extra';
|
||||||
import { Octokit } from 'octokit';
|
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
import toml from 'toml';
|
import toml from 'toml';
|
||||||
|
import debug from 'debug';
|
||||||
import { DataSource, DeepPartial, EntityTarget, ObjectLiteral } from 'typeorm';
|
import { DataSource, DeepPartial, EntityTarget, ObjectLiteral } from 'typeorm';
|
||||||
|
|
||||||
import { Config } from './config';
|
|
||||||
import { DEFAULT_CONFIG_FILE_PATH } from './constants';
|
|
||||||
import { PackageJSON } from './types';
|
|
||||||
|
|
||||||
const log = debug('snowball:utils');
|
const log = debug('snowball:utils');
|
||||||
|
|
||||||
export async function getConfig() {
|
export const getConfig = async <ConfigType>(
|
||||||
// TODO: get config path using cli
|
configFile: string
|
||||||
return await _getConfig<Config>(DEFAULT_CONFIG_FILE_PATH);
|
|
||||||
}
|
|
||||||
|
|
||||||
const _getConfig = async <ConfigType>(
|
|
||||||
configFile: string,
|
|
||||||
): Promise<ConfigType> => {
|
): Promise<ConfigType> => {
|
||||||
const configFilePath = path.resolve(configFile);
|
const configFilePath = path.resolve(configFile);
|
||||||
const fileExists = await fs.pathExists(configFilePath);
|
const fileExists = await fs.pathExists(configFilePath);
|
||||||
@ -52,7 +41,7 @@ export const loadAndSaveData = async <Entity extends ObjectLiteral>(
|
|||||||
entityType: EntityTarget<Entity>,
|
entityType: EntityTarget<Entity>,
|
||||||
dataSource: DataSource,
|
dataSource: DataSource,
|
||||||
entities: any,
|
entities: any,
|
||||||
relations?: any | undefined,
|
relations?: any | undefined
|
||||||
): Promise<Entity[]> => {
|
): Promise<Entity[]> => {
|
||||||
const entityRepository = dataSource.getRepository(entityType);
|
const entityRepository = dataSource.getRepository(entityType);
|
||||||
|
|
||||||
@ -67,7 +56,7 @@ export const loadAndSaveData = async <Entity extends ObjectLiteral>(
|
|||||||
|
|
||||||
entity = {
|
entity = {
|
||||||
...entity,
|
...entity,
|
||||||
[field]: relations[field][entityData[valueIndex]],
|
[field]: relations[field][entityData[valueIndex]]
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -77,67 +66,3 @@ export const loadAndSaveData = async <Entity extends ObjectLiteral>(
|
|||||||
|
|
||||||
return savedEntity;
|
return savedEntity;
|
||||||
};
|
};
|
||||||
|
|
||||||
export const sleep = async (ms: number): Promise<void> =>
|
|
||||||
new Promise((resolve) => setTimeout(resolve, ms));
|
|
||||||
|
|
||||||
export const getRepoDetails = async (
|
|
||||||
octokit: Octokit,
|
|
||||||
repository: string,
|
|
||||||
commitHash: string | undefined,
|
|
||||||
): Promise<{
|
|
||||||
repo: string;
|
|
||||||
packageJSON: PackageJSON;
|
|
||||||
repoUrl: string;
|
|
||||||
}> => {
|
|
||||||
const [owner, repo] = repository.split('/');
|
|
||||||
const { data: packageJSONData } = await octokit.rest.repos.getContent({
|
|
||||||
owner,
|
|
||||||
repo,
|
|
||||||
path: 'package.json',
|
|
||||||
ref: commitHash,
|
|
||||||
});
|
|
||||||
|
|
||||||
if (!packageJSONData) {
|
|
||||||
throw new Error('Package.json file not found');
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(!Array.isArray(packageJSONData) && packageJSONData.type === 'file');
|
|
||||||
const packageJSON: PackageJSON = JSON.parse(atob(packageJSONData.content));
|
|
||||||
|
|
||||||
assert(packageJSON.name, "name field doesn't exist in package.json");
|
|
||||||
|
|
||||||
const repoUrl = (
|
|
||||||
await octokit.rest.repos.get({
|
|
||||||
owner,
|
|
||||||
repo,
|
|
||||||
})
|
|
||||||
).data.html_url;
|
|
||||||
|
|
||||||
return {
|
|
||||||
repo,
|
|
||||||
packageJSON,
|
|
||||||
repoUrl
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
// Wrapper method for registry txs to retry once if 'account sequence mismatch' occurs
|
|
||||||
export const registryTransactionWithRetry = async (
|
|
||||||
txMethod: () => Promise<any>
|
|
||||||
): Promise<any> => {
|
|
||||||
try {
|
|
||||||
return await txMethod();
|
|
||||||
} catch (error: any) {
|
|
||||||
if (!error.message.includes('account sequence mismatch')) {
|
|
||||||
throw error;
|
|
||||||
}
|
|
||||||
|
|
||||||
console.error(`Transaction failed due to account sequence mismatch. Retrying...`);
|
|
||||||
|
|
||||||
try {
|
|
||||||
return await txMethod();
|
|
||||||
} catch (retryError: any) {
|
|
||||||
throw new Error(`Transaction failed again after retry: ${retryError.message}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -2,6 +2,8 @@ import * as fs from 'fs/promises';
|
|||||||
import debug from 'debug';
|
import debug from 'debug';
|
||||||
|
|
||||||
import { getConfig } from '../src/utils';
|
import { getConfig } from '../src/utils';
|
||||||
|
import { Config } from '../src/config';
|
||||||
|
import { DEFAULT_CONFIG_FILE_PATH } from '../src/constants';
|
||||||
|
|
||||||
const log = debug('snowball:delete-database');
|
const log = debug('snowball:delete-database');
|
||||||
|
|
||||||
@ -11,7 +13,7 @@ const deleteFile = async (filePath: string) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const main = async () => {
|
const main = async () => {
|
||||||
const config = await getConfig();
|
const config = await getConfig<Config>(DEFAULT_CONFIG_FILE_PATH);
|
||||||
|
|
||||||
deleteFile(config.database.dbPath);
|
deleteFile(config.database.dbPath);
|
||||||
};
|
};
|
||||||
|
@ -1,8 +1,8 @@
|
|||||||
[
|
[
|
||||||
{
|
{
|
||||||
"id": "2379cf1f-a232-4ad2-ae14-4d881131cc26",
|
"id": "2379cf1f-a232-4ad2-ae14-4d881131cc26",
|
||||||
"name": "Deploy Tools",
|
"name": "Snowball Tools",
|
||||||
"slug": "deploy-tools"
|
"slug": "snowball-tools-1"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"id": "7eb9b3eb-eb74-4b53-b59a-69884c82a7fb",
|
"id": "7eb9b3eb-eb74-4b53-b59a-69884c82a7fb",
|
||||||
|
@ -16,6 +16,8 @@ import {
|
|||||||
getEntities,
|
getEntities,
|
||||||
loadAndSaveData
|
loadAndSaveData
|
||||||
} from '../src/utils';
|
} from '../src/utils';
|
||||||
|
import { Config } from '../src/config';
|
||||||
|
import { DEFAULT_CONFIG_FILE_PATH } from '../src/constants';
|
||||||
|
|
||||||
const log = debug('snowball:initialize-database');
|
const log = debug('snowball:initialize-database');
|
||||||
|
|
||||||
@ -154,7 +156,7 @@ const generateTestData = async (dataSource: DataSource) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const main = async () => {
|
const main = async () => {
|
||||||
const config = await getConfig();
|
const config = await getConfig<Config>(DEFAULT_CONFIG_FILE_PATH);
|
||||||
const isDbPresent = await checkFileExists(config.database.dbPath);
|
const isDbPresent = await checkFileExists(config.database.dbPath);
|
||||||
|
|
||||||
if (!isDbPresent) {
|
if (!isDbPresent) {
|
||||||
|
@ -1,40 +1,39 @@
|
|||||||
import debug from 'debug';
|
import debug from 'debug';
|
||||||
|
|
||||||
import { parseGasAndFees, Registry } from '@cerc-io/registry-sdk';
|
import { Registry } from '@cerc-io/laconic-sdk';
|
||||||
|
|
||||||
|
import { DEFAULT_CONFIG_FILE_PATH } from '../src/constants';
|
||||||
|
import { Config } from '../src/config';
|
||||||
import { getConfig } from '../src/utils';
|
import { getConfig } from '../src/utils';
|
||||||
|
|
||||||
const log = debug('snowball:initialize-registry');
|
const log = debug('snowball:initialize-registry');
|
||||||
|
|
||||||
const DENOM = 'alnt';
|
const DENOM = 'aphoton';
|
||||||
const BOND_AMOUNT = '1000000000';
|
const BOND_AMOUNT = '1000000000';
|
||||||
|
|
||||||
async function main () {
|
async function main () {
|
||||||
const { registryConfig } = await getConfig();
|
const { registryConfig } = await getConfig<Config>(DEFAULT_CONFIG_FILE_PATH);
|
||||||
|
|
||||||
// TODO: Get authority names from args
|
// TODO: Get authority names from args
|
||||||
const authorityNames = ['snowballtools', registryConfig.authority];
|
const authorityNames = ['snowballtools', registryConfig.authority];
|
||||||
|
|
||||||
const registry = new Registry(registryConfig.gqlEndpoint, registryConfig.restEndpoint, {chainId: registryConfig.chainId});
|
const registry = new Registry(registryConfig.gqlEndpoint, registryConfig.restEndpoint, registryConfig.chainId);
|
||||||
|
|
||||||
const bondId = await registry.getNextBondId(registryConfig.privateKey);
|
const bondId = await registry.getNextBondId(registryConfig.privateKey);
|
||||||
log('bondId:', bondId);
|
log('bondId:', bondId);
|
||||||
|
|
||||||
const fee = parseGasAndFees(registryConfig.fee.gas, registryConfig.fee.fees);
|
|
||||||
|
|
||||||
await registry.createBond(
|
await registry.createBond(
|
||||||
{ denom: DENOM, amount: BOND_AMOUNT },
|
{ denom: DENOM, amount: BOND_AMOUNT },
|
||||||
registryConfig.privateKey,
|
registryConfig.privateKey,
|
||||||
fee
|
registryConfig.fee
|
||||||
);
|
);
|
||||||
|
|
||||||
for await (const name of authorityNames) {
|
for await (const name of authorityNames) {
|
||||||
await registry.reserveAuthority({ name }, registryConfig.privateKey, fee);
|
await registry.reserveAuthority({ name }, registryConfig.privateKey, registryConfig.fee);
|
||||||
log('Reserved authority name:', name);
|
log('Reserved authority name:', name);
|
||||||
await registry.setAuthorityBond(
|
await registry.setAuthorityBond(
|
||||||
{ name, bondId },
|
{ name, bondId },
|
||||||
registryConfig.privateKey,
|
registryConfig.privateKey,
|
||||||
fee
|
registryConfig.fee
|
||||||
);
|
);
|
||||||
log(`Bond ${bondId} set for authority ${name}`);
|
log(`Bond ${bondId} set for authority ${name}`);
|
||||||
}
|
}
|
||||||
|
@ -2,20 +2,22 @@ import debug from 'debug';
|
|||||||
import { DataSource } from 'typeorm';
|
import { DataSource } from 'typeorm';
|
||||||
import path from 'path';
|
import path from 'path';
|
||||||
|
|
||||||
import { parseGasAndFees, Registry } from '@cerc-io/registry-sdk';
|
import { Registry } from '@cerc-io/laconic-sdk';
|
||||||
|
|
||||||
|
import { Config } from '../src/config';
|
||||||
|
import { DEFAULT_CONFIG_FILE_PATH } from '../src/constants';
|
||||||
import { getConfig } from '../src/utils';
|
import { getConfig } from '../src/utils';
|
||||||
import { Deployment, DeploymentStatus, Environment } from '../src/entity/Deployment';
|
import { Deployment, DeploymentStatus } from '../src/entity/Deployment';
|
||||||
|
|
||||||
const log = debug('snowball:publish-deploy-records');
|
const log = debug('snowball:publish-deploy-records');
|
||||||
|
|
||||||
async function main() {
|
async function main () {
|
||||||
const { registryConfig, database, misc } = await getConfig();
|
const { registryConfig, database, misc } = await getConfig<Config>(DEFAULT_CONFIG_FILE_PATH);
|
||||||
|
|
||||||
const registry = new Registry(
|
const registry = new Registry(
|
||||||
registryConfig.gqlEndpoint,
|
registryConfig.gqlEndpoint,
|
||||||
registryConfig.restEndpoint,
|
registryConfig.restEndpoint,
|
||||||
{ chainId: registryConfig.chainId }
|
registryConfig.chainId
|
||||||
);
|
);
|
||||||
|
|
||||||
const dataSource = new DataSource({
|
const dataSource = new DataSource({
|
||||||
@ -38,7 +40,7 @@ async function main() {
|
|||||||
});
|
});
|
||||||
|
|
||||||
for await (const deployment of deployments) {
|
for await (const deployment of deployments) {
|
||||||
const url = `https://${(deployment.project.name).toLowerCase()}-${deployment.id}.${deployment.deployer.baseDomain}`;
|
const url = `${deployment.project.name}-${deployment.id}.${misc.projectDomain}`;
|
||||||
|
|
||||||
const applicationDeploymentRecord = {
|
const applicationDeploymentRecord = {
|
||||||
type: 'ApplicationDeploymentRecord',
|
type: 'ApplicationDeploymentRecord',
|
||||||
@ -59,8 +61,6 @@ async function main() {
|
|||||||
url
|
url
|
||||||
};
|
};
|
||||||
|
|
||||||
const fee = parseGasAndFees(registryConfig.fee.gas, registryConfig.fee.fees);
|
|
||||||
|
|
||||||
const result = await registry.setRecord(
|
const result = await registry.setRecord(
|
||||||
{
|
{
|
||||||
privateKey: registryConfig.privateKey,
|
privateKey: registryConfig.privateKey,
|
||||||
@ -68,26 +68,11 @@ async function main() {
|
|||||||
bondId: registryConfig.bondId
|
bondId: registryConfig.bondId
|
||||||
},
|
},
|
||||||
'',
|
'',
|
||||||
fee
|
registryConfig.fee
|
||||||
);
|
);
|
||||||
|
|
||||||
// Remove deployment for project subdomain if deployment is for production environment
|
|
||||||
if (deployment.environment === Environment.Production) {
|
|
||||||
applicationDeploymentRecord.url = `https://${deployment.project.name}.${deployment.deployer.baseDomain}`;
|
|
||||||
|
|
||||||
await registry.setRecord(
|
|
||||||
{
|
|
||||||
privateKey: registryConfig.privateKey,
|
|
||||||
record: applicationDeploymentRecord,
|
|
||||||
bondId: registryConfig.bondId
|
|
||||||
},
|
|
||||||
'',
|
|
||||||
fee
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
log('Application deployment record data:', applicationDeploymentRecord);
|
log('Application deployment record data:', applicationDeploymentRecord);
|
||||||
log(`Application deployment record published: ${result.id}`);
|
log(`Application deployment record published: ${result.data.id}`);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,67 +0,0 @@
|
|||||||
import debug from 'debug';
|
|
||||||
import { DataSource } from 'typeorm';
|
|
||||||
import path from 'path';
|
|
||||||
|
|
||||||
import { parseGasAndFees, Registry } from '@cerc-io/registry-sdk';
|
|
||||||
|
|
||||||
import { getConfig } from '../src/utils';
|
|
||||||
import { Deployment, DeploymentStatus } from '../src/entity/Deployment';
|
|
||||||
|
|
||||||
const log = debug('snowball:publish-deployment-removal-records');
|
|
||||||
|
|
||||||
async function main () {
|
|
||||||
const { registryConfig, database } = await getConfig();
|
|
||||||
|
|
||||||
const registry = new Registry(
|
|
||||||
registryConfig.gqlEndpoint,
|
|
||||||
registryConfig.restEndpoint,
|
|
||||||
{ chainId: registryConfig.chainId }
|
|
||||||
);
|
|
||||||
|
|
||||||
const dataSource = new DataSource({
|
|
||||||
type: 'better-sqlite3',
|
|
||||||
database: database.dbPath,
|
|
||||||
synchronize: true,
|
|
||||||
entities: [path.join(__dirname, '../src/entity/*')]
|
|
||||||
});
|
|
||||||
|
|
||||||
await dataSource.initialize();
|
|
||||||
|
|
||||||
const deploymentRepository = dataSource.getRepository(Deployment);
|
|
||||||
const deployments = await deploymentRepository.find({
|
|
||||||
relations: {
|
|
||||||
project: true
|
|
||||||
},
|
|
||||||
where: {
|
|
||||||
status: DeploymentStatus.Deleting
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
for await (const deployment of deployments) {
|
|
||||||
const applicationDeploymentRemovalRecord = {
|
|
||||||
type: "ApplicationDeploymentRemovalRecord",
|
|
||||||
version: "1.0.0",
|
|
||||||
deployment: deployment.applicationDeploymentRecordId,
|
|
||||||
request: deployment.applicationDeploymentRemovalRequestId,
|
|
||||||
}
|
|
||||||
|
|
||||||
const fee = parseGasAndFees(registryConfig.fee.gas, registryConfig.fee.fees);
|
|
||||||
|
|
||||||
const result = await registry.setRecord(
|
|
||||||
{
|
|
||||||
privateKey: registryConfig.privateKey,
|
|
||||||
record: applicationDeploymentRemovalRecord,
|
|
||||||
bondId: registryConfig.bondId
|
|
||||||
},
|
|
||||||
'',
|
|
||||||
fee
|
|
||||||
);
|
|
||||||
|
|
||||||
log('Application deployment removal record data:', applicationDeploymentRemovalRecord);
|
|
||||||
log(`Application deployment removal record published: ${result.id}`);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
main().catch((err) => {
|
|
||||||
log(err);
|
|
||||||
});
|
|
@ -1,3 +0,0 @@
|
|||||||
REGISTRY_BOND_ID=
|
|
||||||
DEPLOYER_LRN=
|
|
||||||
AUTHORITY=
|
|
@ -1,35 +1,74 @@
|
|||||||
# deployer
|
# deployer
|
||||||
|
|
||||||
- Install dependencies
|
- Install dependencies
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
yarn
|
yarn
|
||||||
```
|
```
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
brew install jq # if you do not have jq installed already
|
brew install jq # if you do not have jq installed already
|
||||||
```
|
```
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
Example of how to make the necessary deploy edits [here](https://github.com/snowball-tools/snowballtools-base/pull/131/files).
|
||||||
|
|
||||||
|
- Replace variables in the following files
|
||||||
|
- [records/application-deployment-request.yml](records/application-deployment-request.yml)
|
||||||
|
- update the name & application version numbers
|
||||||
|
- `<CURRENT_DATE_TIME>`: Replace with current time which can be generated by command `date -u`
|
||||||
|
```yml
|
||||||
|
# Example
|
||||||
|
record:
|
||||||
|
...
|
||||||
|
meta:
|
||||||
|
note: Added by Snowball @ Friday 23 February 2024 06:35:50 AM UTC
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Update record version in [records/application-record.yml](records/application-record.yml)
|
||||||
|
```yml
|
||||||
|
record:
|
||||||
|
type: ApplicationRecord
|
||||||
|
version: <NEW_VERSION>
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
|
- Update commit hash in the following places:
|
||||||
|
- [records/application-record.yml](records/application-record.yml)
|
||||||
|
```yml
|
||||||
|
record:
|
||||||
|
...
|
||||||
|
repository_ref: <COMMIT_HASH>
|
||||||
|
...
|
||||||
|
```
|
||||||
|
- [records/application-deployment-request.yml](records/application-deployment-request.yml)
|
||||||
|
```yml
|
||||||
|
record:
|
||||||
|
...
|
||||||
|
meta:
|
||||||
|
...
|
||||||
|
repository_ref: <COMMIT_HASH>
|
||||||
|
```
|
||||||
|
- [deploy-frontend.sh](deploy-frontend.sh)
|
||||||
|
Also be sure to update the app version
|
||||||
|
```bash
|
||||||
|
...
|
||||||
|
RCD_APP_VERSION="<NEW_VERSION>"
|
||||||
|
REPO_REF="<COMMIT_HASH>"
|
||||||
|
...
|
||||||
|
```
|
||||||
|
|
||||||
- Run script to deploy app
|
- Run script to deploy app
|
||||||
|
```
|
||||||
- To deploy frontend app to `dashboard.staging.apps.snowballtools.com`
|
./deploy-frontend.sh
|
||||||
|
```
|
||||||
```bash
|
|
||||||
./deploy-frontend.staging.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
- To deploy frontend app to `dashboard.apps.snowballtools.com`
|
|
||||||
|
|
||||||
```bash
|
|
||||||
./deploy-frontend.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
- Commit the updated [ApplicationRecord](records/application-record.yml) and [ApplicationDeploymentRequest](records/application-deployment-request.yml) files to the repository
|
- Commit the updated [ApplicationRecord](records/application-record.yml) and [ApplicationDeploymentRequest](records/application-deployment-request.yml) files to the repository
|
||||||
|
|
||||||
## Notes
|
## Notes
|
||||||
|
|
||||||
- Any config env can be updated in [records/application-deployment-request.yml](records/application-deployment-request.yml)
|
- Any config env can be updated in [records/application-deployment-request.yml](records/application-deployment-request.yml)
|
||||||
|
|
||||||
```yml
|
```yml
|
||||||
record:
|
record:
|
||||||
...
|
...
|
||||||
@ -38,27 +77,23 @@
|
|||||||
LACONIC_HOSTED_CONFIG_app_server_url: https://snowballtools-base-api-001.apps.snowballtools.com
|
LACONIC_HOSTED_CONFIG_app_server_url: https://snowballtools-base-api-001.apps.snowballtools.com
|
||||||
...
|
...
|
||||||
```
|
```
|
||||||
|
|
||||||
- On changing `LACONIC_HOSTED_CONFIG_app_github_clientid`, the GitHub client ID and secret need to be changed in backend config too
|
- On changing `LACONIC_HOSTED_CONFIG_app_github_clientid`, the GitHub client ID and secret need to be changed in backend config too
|
||||||
|
|
||||||
## Troubleshoot
|
## Troubleshoot
|
||||||
|
|
||||||
- Check deployment status in [web-app deployer](https://console.laconic.com/deployer).
|
- Check deployment status [here](https://console.laconic.com/deployer).
|
||||||
- Check records in [registry console app](https://console.laconic.com/#/registry).
|
- Check records [here](https://console.laconic.com/#/registry).
|
||||||
|
|
||||||
- If deployment fails due to low bond balance
|
- If deployment fails due to low bond balance
|
||||||
- Check balances
|
- Check balances
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Account balance
|
# Account balance
|
||||||
yarn laconic registry account get
|
yarn laconic cns account get
|
||||||
|
|
||||||
# Bond balance
|
# Bond balance
|
||||||
yarn laconic registry bond get --id 99c0e9aec0ac1b8187faa579be3b54f93fafb6060ac1fd29170b860df605be32
|
yarn laconic cns bond get --id 8fcf44b2f326b4b63ac57547777f1c78b7d494e5966e508f09001af53cb440ac
|
||||||
```
|
```
|
||||||
|
|
||||||
- Command to refill bond
|
- Command to refill bond
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
yarn laconic registry bond refill --id 99c0e9aec0ac1b8187faa579be3b54f93fafb6060ac1fd29170b860df605be32 --type alnt --quantity 10000000
|
yarn laconic cns bond refill --id 8fcf44b2f326b4b63ac57547777f1c78b7d494e5966e508f09001af53cb440ac --type aphoton --quantity 10000000
|
||||||
```
|
```
|
||||||
|
@ -1,10 +0,0 @@
|
|||||||
services:
|
|
||||||
registry:
|
|
||||||
restEndpoint: 'http://console.laconic.com:1317'
|
|
||||||
gqlEndpoint: 'http://console.laconic.com:9473/api'
|
|
||||||
userKey: 87d00f66a73e2ca428adeb49ba9164d0ad9a87edc60e33d46ad3031b9c5701fe
|
|
||||||
bondId: 89c75c7bc5759861d10285aff6f9e7227d6855e446b77ad5d8324822dfec7deb
|
|
||||||
chainId: laconic_9000-1
|
|
||||||
gas:
|
|
||||||
fees:
|
|
||||||
gasPrice: 1
|
|
@ -1,8 +1,9 @@
|
|||||||
services:
|
services:
|
||||||
registry:
|
cns:
|
||||||
rpcEndpoint: https://laconicd-sapo.laconic.com
|
restEndpoint: http://console.laconic.com:1317
|
||||||
gqlEndpoint: https://laconicd-sapo.laconic.com/api
|
gqlEndpoint: http://console.laconic.com:9473/api
|
||||||
userKey:
|
chainId: laconic_9000-1
|
||||||
bondId:
|
gas: 1000000
|
||||||
chainId: laconic_9000-2
|
fees: 200000aphoton
|
||||||
gasPrice: 1alnt
|
userKey: 0524fc22ea0a12e6c5cc4cfe08e73c95dffd0ab5ed72a59f459ed33134fa3b16
|
||||||
|
bondId: 8fcf44b2f326b4b63ac57547777f1c78b7d494e5966e508f09001af53cb440ac
|
||||||
|
@ -1,148 +1,35 @@
|
|||||||
#!/bin/bash
|
#!/bin/bash
|
||||||
|
|
||||||
source .env
|
|
||||||
echo "Using REGISTRY_BOND_ID: $REGISTRY_BOND_ID"
|
|
||||||
echo "Using DEPLOYER_LRN: $DEPLOYER_LRN"
|
|
||||||
echo "Using AUTHORITY: $AUTHORITY"
|
|
||||||
|
|
||||||
# Repository URL
|
|
||||||
REPO_URL="https://git.vdb.to/cerc-io/snowballtools-base"
|
|
||||||
|
|
||||||
# Get the latest commit hash from the repository
|
|
||||||
LATEST_HASH=$(git ls-remote $REPO_URL HEAD | awk '{print $1}')
|
|
||||||
|
|
||||||
# Extract version from ../frontend/package.json
|
|
||||||
PACKAGE_VERSION=$(jq -r '.version' ../frontend/package.json)
|
|
||||||
|
|
||||||
# Current date and time for note
|
|
||||||
CURRENT_DATE_TIME=$(date -u)
|
|
||||||
|
|
||||||
CONFIG_FILE=config.yml
|
|
||||||
|
|
||||||
# Reference: https://git.vdb.to/cerc-io/test-progressive-web-app/src/branch/main/scripts
|
# Reference: https://git.vdb.to/cerc-io/test-progressive-web-app/src/branch/main/scripts
|
||||||
|
|
||||||
# Get latest version from registry and increment application-record version
|
|
||||||
NEW_APPLICATION_VERSION=$(yarn --silent laconic -c $CONFIG_FILE registry record list --type ApplicationRecord --all --name "deploy-frontend" 2>/dev/null | jq -r -s ".[] | sort_by(.createTime) | reverse | [ .[] | select(.bondId == \"$REGISTRY_BOND_ID\") ] | .[0].attributes.version" | awk -F. -v OFS=. '{$NF += 1 ; print}')
|
|
||||||
|
|
||||||
if [ -z "$NEW_APPLICATION_VERSION" ] || [ "1" == "$NEW_APPLICATION_VERSION" ]; then
|
|
||||||
# Set application-record version if no previous records were found
|
|
||||||
NEW_APPLICATION_VERSION=0.0.1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Generate application-record.yml with incremented version
|
|
||||||
cat >./records/application-record.yml <<EOF
|
|
||||||
record:
|
|
||||||
type: ApplicationRecord
|
|
||||||
version: $NEW_APPLICATION_VERSION
|
|
||||||
repository_ref: $LATEST_HASH
|
|
||||||
repository: ["$REPO_URL"]
|
|
||||||
app_type: webapp
|
|
||||||
name: deploy-frontend
|
|
||||||
app_version: $PACKAGE_VERSION
|
|
||||||
EOF
|
|
||||||
|
|
||||||
echo "Files generated successfully"
|
|
||||||
|
|
||||||
RECORD_FILE=records/application-record.yml
|
RECORD_FILE=records/application-record.yml
|
||||||
|
CONFIG_FILE=config.yml
|
||||||
|
RCD_APP_VERSION="0.1.3"
|
||||||
|
REPO_REF="513ca69d01bee857cf207a0605483205b384e218"
|
||||||
|
|
||||||
# Publish ApplicationRecord
|
# Publish ApplicationRecord
|
||||||
publish_response=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $RECORD_FILE)
|
RECORD_ID=$(yarn --silent laconic -c $CONFIG_FILE cns record publish --filename $RECORD_FILE | jq -r '.id')
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to publish record"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
RECORD_ID=$(echo $publish_response | jq -r '.id')
|
|
||||||
echo "ApplicationRecord published"
|
echo "ApplicationRecord published"
|
||||||
echo $RECORD_ID
|
echo $RECORD_ID
|
||||||
|
|
||||||
# Set name to record
|
# Set name to record
|
||||||
REGISTRY_APP_LRN="lrn://$AUTHORITY/applications/deploy-frontend"
|
REGISTRY_APP_CRN="crn://snowballtools/applications/snowballtools-base-frontend"
|
||||||
|
|
||||||
sleep 2
|
yarn --silent laconic -c $CONFIG_FILE cns name set "$REGISTRY_APP_CRN@${RCD_APP_VERSION}" "$RECORD_ID"
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN@${PACKAGE_VERSION}" "$RECORD_ID"
|
yarn --silent laconic -c $CONFIG_FILE cns name set "$REGISTRY_APP_CRN@${REPO_REF}" "$RECORD_ID"
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to set name: $REGISTRY_APP_LRN@${PACKAGE_VERSION}"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
sleep 2
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN@${LATEST_HASH}" "$RECORD_ID"
|
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to set hash"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
sleep 2
|
|
||||||
# Set name if latest release
|
# Set name if latest release
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN" "$RECORD_ID"
|
yarn --silent laconic -c $CONFIG_FILE cns name set "$REGISTRY_APP_CRN" "$RECORD_ID"
|
||||||
rc=$?
|
echo "$REGISTRY_APP_CRN set for ApplicationRecord"
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to set release"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
echo "$REGISTRY_APP_LRN set for ApplicationRecord"
|
|
||||||
|
|
||||||
# Check if record found for REGISTRY_APP_LRN
|
# Check if record found for REGISTRY_APP_CRN
|
||||||
query_response=$(yarn --silent laconic -c $CONFIG_FILE registry name resolve "$REGISTRY_APP_LRN")
|
APP_RECORD=$(yarn --silent laconic -c $CONFIG_FILE cns name resolve "$REGISTRY_APP_CRN" | jq '.[0]')
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to query name"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
APP_RECORD=$(echo $query_response | jq '.[0]')
|
|
||||||
if [ -z "$APP_RECORD" ] || [ "null" == "$APP_RECORD" ]; then
|
if [ -z "$APP_RECORD" ] || [ "null" == "$APP_RECORD" ]; then
|
||||||
echo "No record found for $REGISTRY_APP_LRN."
|
echo "No record found for $REGISTRY_APP_CRN."
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Get payment address for deployer
|
|
||||||
paymentAddress=$(yarn --silent laconic -c config.yml registry name resolve "$DEPLOYER_LRN" | jq -r '.[0].attributes.paymentAddress')
|
|
||||||
paymentAmount=$(yarn --silent laconic -c config.yml registry name resolve "$DEPLOYER_LRN" | jq -r '.[0].attributes.minimumPayment' | sed 's/alnt//g')
|
|
||||||
# Pay deployer if paymentAmount is not null
|
|
||||||
if [[ -n "$paymentAmount" && "$paymentAmount" != "null" ]]; then
|
|
||||||
payment=$(yarn --silent laconic -c config.yml registry tokens send --address "$paymentAddress" --type alnt --quantity "$paymentAmount")
|
|
||||||
|
|
||||||
# Extract the transaction hash
|
|
||||||
txHash=$(echo "$payment" | jq -r '.tx.hash')
|
|
||||||
echo "Paid deployer with txHash as $txHash"
|
|
||||||
|
|
||||||
else
|
|
||||||
echo "Payment amount is null; skipping payment."
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Generate application-deployment-request.yml
|
|
||||||
cat >./records/application-deployment-request.yml <<EOF
|
|
||||||
record:
|
|
||||||
type: ApplicationDeploymentRequest
|
|
||||||
version: '1.0.0'
|
|
||||||
name: deploy-frontend@$PACKAGE_VERSION
|
|
||||||
application: lrn://$AUTHORITY/applications/deploy-frontend@$PACKAGE_VERSION
|
|
||||||
deployer: $DEPLOYER_LRN
|
|
||||||
dns: deploy
|
|
||||||
config:
|
|
||||||
env:
|
|
||||||
LACONIC_HOSTED_CONFIG_server_url: https://deploy-backend.apps.vaasl.io
|
|
||||||
LACONIC_HOSTED_CONFIG_github_clientid: Ov23liaet4yc0KX0iM1c
|
|
||||||
LACONIC_HOSTED_CONFIG_github_pwa_templaterepo: laconic-templates/test-progressive-web-app
|
|
||||||
LACONIC_HOSTED_CONFIG_github_image_upload_templaterepo: laconic-templates/image-upload-pwa-example
|
|
||||||
LACONIC_HOSTED_CONFIG_wallet_connect_id: 63cad7ba97391f63652161f484670e15
|
|
||||||
LACONIC_HOSTED_CONFIG_laconicd_chain_id: laconic-testnet-2
|
|
||||||
meta:
|
|
||||||
note: Added by Snowball @ $CURRENT_DATE_TIME
|
|
||||||
repository: "$REPO_URL"
|
|
||||||
repository_ref: $LATEST_HASH
|
|
||||||
payment: $txHash
|
|
||||||
EOF
|
|
||||||
|
|
||||||
RECORD_FILE=records/application-deployment-request.yml
|
RECORD_FILE=records/application-deployment-request.yml
|
||||||
|
|
||||||
sleep 2
|
DEPLOYMENT_REQUEST_ID=$(yarn --silent laconic -c $CONFIG_FILE cns record publish --filename $RECORD_FILE | jq -r '.id')
|
||||||
deployment_response=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $RECORD_FILE)
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to query deployment request"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
DEPLOYMENT_REQUEST_ID=$(echo $deployment_response | jq -r '.id')
|
|
||||||
echo "ApplicationDeploymentRequest published"
|
echo "ApplicationDeploymentRequest published"
|
||||||
echo $DEPLOYMENT_REQUEST_ID
|
echo $DEPLOYMENT_REQUEST_ID
|
||||||
|
@ -1,134 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Repository URL
|
|
||||||
REPO_URL="https://git.vdb.to/cerc-io/snowballtools-base"
|
|
||||||
|
|
||||||
# Get the latest commit hash from the repository
|
|
||||||
LATEST_HASH=$(git ls-remote $REPO_URL HEAD | awk '{print $1}')
|
|
||||||
|
|
||||||
# Extract version from ../frontend/package.json
|
|
||||||
PACKAGE_VERSION=$(jq -r '.version' ../frontend/package.json)
|
|
||||||
|
|
||||||
# Current date and time for note
|
|
||||||
CURRENT_DATE_TIME=$(date -u)
|
|
||||||
|
|
||||||
CONFIG_FILE=config.staging.yml
|
|
||||||
REGISTRY_BOND_ID="098c906850b87412f02200e41f449bc79e055eab77acfef32c0b22443bb46661"
|
|
||||||
|
|
||||||
# Reference: https://git.vdb.to/cerc-io/test-progressive-web-app/src/branch/main/scripts
|
|
||||||
|
|
||||||
# Get latest version from registry and increment application-record version
|
|
||||||
NEW_APPLICATION_VERSION=$(yarn --silent laconic -c $CONFIG_FILE registry record list --type ApplicationRecord --all --name "staging-snowballtools-base-frontend" 2>/dev/null | jq -r -s ".[] | sort_by(.createTime) | reverse | [ .[] | select(.bondId == \"$REGISTRY_BOND_ID\") ] | .[0].attributes.version" | awk -F. -v OFS=. '{$NF += 1 ; print}')
|
|
||||||
|
|
||||||
if [ -z "$NEW_APPLICATION_VERSION" ] || [ "1" == "$NEW_APPLICATION_VERSION" ]; then
|
|
||||||
# Set application-record version if no previous records were found
|
|
||||||
NEW_APPLICATION_VERSION=0.0.1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Generate application-deployment-request.yml
|
|
||||||
cat >./staging-records/application-deployment-request.yml <<EOF
|
|
||||||
record:
|
|
||||||
type: ApplicationDeploymentRequest
|
|
||||||
version: '1.0.0'
|
|
||||||
name: staging-snowballtools-base-frontend@$PACKAGE_VERSION
|
|
||||||
application: lrn://staging-snowballtools/applications/staging-snowballtools-base-frontend@$PACKAGE_VERSION
|
|
||||||
dns: dashboard.staging.apps.snowballtools.com
|
|
||||||
config:
|
|
||||||
env:
|
|
||||||
LACONIC_HOSTED_CONFIG_server_url: https://snowballtools-base-api.staging.apps.snowballtools.com
|
|
||||||
LACONIC_HOSTED_CONFIG_github_clientid: Ov23liOaoahRTYd4nSCV
|
|
||||||
LACONIC_HOSTED_CONFIG_github_templaterepo: snowball-tools/test-progressive-web-app
|
|
||||||
LACONIC_HOSTED_CONFIG_github_pwa_templaterepo: snowball-tools/test-progressive-web-app
|
|
||||||
LACONIC_HOSTED_CONFIG_github_image_upload_templaterepo: snowball-tools/image-upload-pwa-example
|
|
||||||
LACONIC_HOSTED_CONFIG_wallet_connect_id: eda9ba18042a5ea500f358194611ece2
|
|
||||||
LACONIC_HOSTED_CONFIG_laconicd_chain_id: laconic-testnet-2
|
|
||||||
LACONIC_HOSTED_CONFIG_lit_relay_api_key: 15DDD969-E75F-404D-AAD9-58A37C4FD354_snowball
|
|
||||||
LACONIC_HOSTED_CONFIG_aplchemy_api_key: THvPart_gqI5x02RNYSBntlmwA66I_qc
|
|
||||||
LACONIC_HOSTED_CONFIG_bugsnag_api_key: 8c480cd5386079f9dd44f9581264a073
|
|
||||||
LACONIC_HOSTED_CONFIG_passkey_wallet_rpid: dashboard.staging.apps.snowballtools.com
|
|
||||||
LACONIC_HOSTED_CONFIG_turnkey_api_base_url: https://api.turnkey.com
|
|
||||||
LACONIC_HOSTED_CONFIG_turnkey_organization_id: 5049ae99-5bca-40b3-8317-504384d4e591
|
|
||||||
meta:
|
|
||||||
note: Added by Snowball @ $CURRENT_DATE_TIME
|
|
||||||
repository: "$REPO_URL"
|
|
||||||
repository_ref: $LATEST_HASH
|
|
||||||
EOF
|
|
||||||
|
|
||||||
# Generate application-record.yml with incremented version
|
|
||||||
cat >./staging-records/application-record.yml <<EOF
|
|
||||||
record:
|
|
||||||
type: ApplicationRecord
|
|
||||||
version: $NEW_APPLICATION_VERSION
|
|
||||||
repository_ref: $LATEST_HASH
|
|
||||||
repository: ["$REPO_URL"]
|
|
||||||
app_type: webapp
|
|
||||||
name: staging-snowballtools-base-frontend
|
|
||||||
app_version: $PACKAGE_VERSION
|
|
||||||
EOF
|
|
||||||
|
|
||||||
echo "Files generated successfully."
|
|
||||||
|
|
||||||
RECORD_FILE=staging-records/application-record.yml
|
|
||||||
|
|
||||||
# Publish ApplicationRecord
|
|
||||||
publish_response=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $RECORD_FILE)
|
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to publish record"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
RECORD_ID=$(echo $publish_response | jq -r '.id')
|
|
||||||
echo "ApplicationRecord published"
|
|
||||||
echo $RECORD_ID
|
|
||||||
|
|
||||||
# Set name to record
|
|
||||||
REGISTRY_APP_LRN="lrn://staging-snowballtools/applications/staging-snowballtools-base-frontend"
|
|
||||||
|
|
||||||
sleep 2
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN@${PACKAGE_VERSION}" "$RECORD_ID"
|
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to set name: $REGISTRY_APP_LRN@${PACKAGE_VERSION}"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
sleep 2
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN@${LATEST_HASH}" "$RECORD_ID"
|
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to set hash"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
sleep 2
|
|
||||||
# Set name if latest release
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN" "$RECORD_ID"
|
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to set release"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
echo "$REGISTRY_APP_LRN set for ApplicationRecord"
|
|
||||||
|
|
||||||
# Check if record found for REGISTRY_APP_LRN
|
|
||||||
query_response=$(yarn --silent laconic -c $CONFIG_FILE registry name resolve "$REGISTRY_APP_LRN")
|
|
||||||
rc=$?
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to query name"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
APP_RECORD=$(echo $query_response | jq '.[0]')
|
|
||||||
if [ -z "$APP_RECORD" ] || [ "null" == "$APP_RECORD" ]; then
|
|
||||||
echo "No record found for $REGISTRY_APP_LRN."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
RECORD_FILE=staging-records/application-deployment-request.yml
|
|
||||||
|
|
||||||
sleep 2
|
|
||||||
deployment_response=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $RECORD_FILE)
|
|
||||||
if [ $rc -ne 0 ]; then
|
|
||||||
echo "FATAL: Failed to query deployment request"
|
|
||||||
exit $rc
|
|
||||||
fi
|
|
||||||
DEPLOYMENT_REQUEST_ID=$(echo $deployment_response | jq -r '.id')
|
|
||||||
echo "ApplicationDeploymentRequest published"
|
|
||||||
echo $DEPLOYMENT_REQUEST_ID
|
|
@ -4,6 +4,6 @@
|
|||||||
"main": "index.js",
|
"main": "index.js",
|
||||||
"private": true,
|
"private": true,
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@cerc-io/laconic-registry-cli": "^0.2.9"
|
"@cerc-io/laconic-registry-cli": "^0.1.10"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,17 +1,21 @@
|
|||||||
record:
|
record:
|
||||||
type: ApplicationDeploymentRequest
|
type: ApplicationDeploymentRequest
|
||||||
version: '1.0.0'
|
version: '1.0.0'
|
||||||
name: deploy-frontend@1.0.0
|
name: snowballtools-base-frontend@0.1.3
|
||||||
application: lrn://vaasl/applications/deploy-frontend@1.0.0
|
application: crn://snowballtools/applications/snowballtools-base-frontend@0.1.3
|
||||||
dns: deploy
|
dns: dashboard
|
||||||
config:
|
config:
|
||||||
env:
|
env:
|
||||||
LACONIC_HOSTED_CONFIG_server_url: https://deploy-backend.apps.vaasl.io
|
LACONIC_HOSTED_CONFIG_app_server_url: https://snowballtools-base-api-001.apps.snowballtools.com
|
||||||
LACONIC_HOSTED_CONFIG_github_clientid: Ov23liaet4yc0KX0iM1c
|
# If GitHub client ID is changed, same ID and corresponding secret has to be set in backend config
|
||||||
LACONIC_HOSTED_CONFIG_github_pwa_templaterepo: laconic-templates/test-progressive-web-app
|
LACONIC_HOSTED_CONFIG_app_github_clientid: b7c63b235ca1dd5639ab
|
||||||
LACONIC_HOSTED_CONFIG_github_image_upload_templaterepo: laconic-templates/image-upload-pwa-example
|
LACONIC_HOSTED_CONFIG_app_github_templaterepo: snowball-tools-platform/test-progressive-web-app
|
||||||
LACONIC_HOSTED_CONFIG_wallet_connect_id: 63cad7ba97391f63652161f484670e15
|
# New config env after changes for image upload PWA
|
||||||
|
LACONIC_HOSTED_CONFIG_app_github_pwa_templaterepo: snowball-tools-platform/test-progressive-web-app
|
||||||
|
LACONIC_HOSTED_CONFIG_app_github_image_upload_templaterepo: snowball-tools-platform/image-upload-pwa-example
|
||||||
|
LACONIC_HOSTED_CONFIG_app_wallet_connect_id: eda9ba18042a5ea500f358194611ece2
|
||||||
meta:
|
meta:
|
||||||
note: Added by Snowball @ Thu Apr 4 14:49:41 UTC 2024
|
# Set CURRENT_DATE_TIME; Use command date -u
|
||||||
|
note: Added by Snowball @ Tue Feb 27 17:24:06 UTC 2024
|
||||||
repository: "https://git.vdb.to/cerc-io/snowballtools-base"
|
repository: "https://git.vdb.to/cerc-io/snowballtools-base"
|
||||||
repository_ref: 351db16336eacc3e1f9119ceb8d1282b8e27a27e
|
repository_ref: 513ca69d01bee857cf207a0605483205b384e218
|
||||||
|
@ -1,8 +1,10 @@
|
|||||||
record:
|
record:
|
||||||
type: ApplicationRecord
|
type: ApplicationRecord
|
||||||
version: 0.0.2
|
version: 0.0.11
|
||||||
repository_ref: 351db16336eacc3e1f9119ceb8d1282b8e27a27e
|
repository_ref: 513ca69d01bee857cf207a0605483205b384e218
|
||||||
repository: ["https://git.vdb.to/cerc-io/snowballtools-base"]
|
repository: ["https://git.vdb.to/cerc-io/snowballtools-base"]
|
||||||
app_type: webapp
|
app_type: webapp
|
||||||
name: deploy-frontend
|
# name is set to repo name
|
||||||
app_version: 1.0.0
|
name: snowballtools-base-frontend
|
||||||
|
# app_version is set from package.json
|
||||||
|
app_version: 0.1.3
|
||||||
|
@ -1,24 +0,0 @@
|
|||||||
record:
|
|
||||||
type: ApplicationDeploymentRequest
|
|
||||||
version: '1.0.0'
|
|
||||||
name: staging-snowballtools-base-frontend@0.0.0
|
|
||||||
application: crn://staging-snowballtools/applications/staging-snowballtools-base-frontend@0.0.0
|
|
||||||
dns: dashboard.staging.apps.snowballtools.com
|
|
||||||
config:
|
|
||||||
env:
|
|
||||||
LACONIC_HOSTED_CONFIG_server_url: https://snowballtools-base-api.staging.apps.snowballtools.com
|
|
||||||
LACONIC_HOSTED_CONFIG_github_clientid: Ov23liOaoahRTYd4nSCV
|
|
||||||
LACONIC_HOSTED_CONFIG_github_templaterepo: snowball-tools/test-progressive-web-app
|
|
||||||
LACONIC_HOSTED_CONFIG_github_pwa_templaterepo: snowball-tools/test-progressive-web-app
|
|
||||||
LACONIC_HOSTED_CONFIG_github_image_upload_templaterepo: snowball-tools/image-upload-pwa-example
|
|
||||||
LACONIC_HOSTED_CONFIG_wallet_connect_id: eda9ba18042a5ea500f358194611ece2
|
|
||||||
LACONIC_HOSTED_CONFIG_lit_relay_api_key: 15DDD969-E75F-404D-AAD9-58A37C4FD354_snowball
|
|
||||||
LACONIC_HOSTED_CONFIG_aplchemy_api_key: THvPart_gqI5x02RNYSBntlmwA66I_qc
|
|
||||||
LACONIC_HOSTED_CONFIG_bugsnag_api_key: 8c480cd5386079f9dd44f9581264a073
|
|
||||||
LACONIC_HOSTED_CONFIG_passkey_wallet_rpid: dashboard.staging.apps.snowballtools.com
|
|
||||||
LACONIC_HOSTED_CONFIG_turnkey_api_base_url: https://api.turnkey.com
|
|
||||||
LACONIC_HOSTED_CONFIG_turnkey_organization_id: 5049ae99-5bca-40b3-8317-504384d4e591
|
|
||||||
meta:
|
|
||||||
note: Added by Snowball @ Mon Jun 24 23:51:48 UTC 2024
|
|
||||||
repository: "https://git.vdb.to/cerc-io/snowballtools-base"
|
|
||||||
repository_ref: 61e3e88a6c9d57e95441059369ee5a46f5c07601
|
|
@ -1,8 +0,0 @@
|
|||||||
record:
|
|
||||||
type: ApplicationRecord
|
|
||||||
version: 0.0.1
|
|
||||||
repository_ref: 61e3e88a6c9d57e95441059369ee5a46f5c07601
|
|
||||||
repository: ["https://git.vdb.to/cerc-io/snowballtools-base"]
|
|
||||||
app_type: webapp
|
|
||||||
name: staging-snowballtools-base-frontend
|
|
||||||
app_version: 0.0.0
|
|
@ -1,23 +0,0 @@
|
|||||||
# deployer test
|
|
||||||
|
|
||||||
Check if the live web app deployer is in a working state
|
|
||||||
|
|
||||||
- Web app repo used: <https://github.com/snowball-tools/test-progressive-web-app> (main branch)
|
|
||||||
- Config used: [../config.yml](../config.yml)
|
|
||||||
- The script [test-webapp-deployment-undeployment.sh](./test-webapp-deployment-undeployment.sh) performs the following:
|
|
||||||
- Create / update [`ApplicationRecord`](./records/application-record.yml) and [`ApplicationDeploymentRequest`](./records/application-deployment-request.yml) records with latest meta data from the repo
|
|
||||||
- Fetch the latest version of `deployment-test-app` from registry and increment `ApplicationRecord` version
|
|
||||||
- Publish the resulting `ApplicationRecord` record
|
|
||||||
- Set names to the record and check name resolution
|
|
||||||
- Publish the `ApplicationDeploymentRequest` record
|
|
||||||
- Check that the deployment occurs
|
|
||||||
- Check that a `ApplicationDeploymentRecord` is created
|
|
||||||
- Check that the deployment record has correct `ApplicationRecord` id
|
|
||||||
- Check that the URL present in deployment record is active
|
|
||||||
- Create and publish a [`ApplicationDeploymentRemovalRequest`](./records/application-deployment-removal-request.yml) record
|
|
||||||
- Check that the deployment is removed
|
|
||||||
- Check that a `ApplicationDeploymentRemovalRecord` is created
|
|
||||||
- Check that the deployment URL goes down
|
|
||||||
- The test script is run in a GitHub CI [workflow](../../../.github/workflows/test-app-deployment.yaml) that:
|
|
||||||
- Is scheduled to run everyday on the default (`main`) branch or can be triggered manually
|
|
||||||
- Sends Slack alerts to configured channels on failure
|
|
@ -1,4 +0,0 @@
|
|||||||
record:
|
|
||||||
deployment: <APPLICATION_DEPLOYMENT_RECORD_ID>
|
|
||||||
type: ApplicationDeploymentRemovalRequest
|
|
||||||
version: 1.0.0
|
|
@ -1,15 +0,0 @@
|
|||||||
record:
|
|
||||||
type: ApplicationDeploymentRequest
|
|
||||||
version: "1.0.0"
|
|
||||||
name: deployment-test-app@0.1.24
|
|
||||||
application: crn://snowballtools/applications/deployment-test-app@0.1.24
|
|
||||||
dns: deployment-ci-test
|
|
||||||
config:
|
|
||||||
env:
|
|
||||||
CERC_TEST_WEBAPP_CONFIG1: "deployment test config 1"
|
|
||||||
CERC_TEST_WEBAPP_CONFIG2: "deployment test config 2"
|
|
||||||
CERC_WEBAPP_DEBUG: 0
|
|
||||||
meta:
|
|
||||||
note: Deployment test @ Thu 11 Apr 2024 07:29:19 AM UTC
|
|
||||||
repository: "https://github.com/snowball-tools/test-progressive-web-app"
|
|
||||||
repository_ref: 05819619487a0d2dbc5453b6d1ccff3044c0dd26
|
|
@ -1,8 +0,0 @@
|
|||||||
record:
|
|
||||||
type: ApplicationRecord
|
|
||||||
version: 0.0.1
|
|
||||||
repository_ref: 05819619487a0d2dbc5453b6d1ccff3044c0dd26
|
|
||||||
repository: ["https://github.com/snowball-tools/test-progressive-web-app"]
|
|
||||||
app_type: webapp
|
|
||||||
name: deployment-test-app
|
|
||||||
app_version: 0.1.24
|
|
@ -1,225 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
# Repository URL
|
|
||||||
REPO_URL="https://github.com/snowball-tools/test-progressive-web-app"
|
|
||||||
|
|
||||||
# Get the latest commit hash from the repository
|
|
||||||
LATEST_HASH=$(git ls-remote $REPO_URL HEAD | awk '{print $1}')
|
|
||||||
|
|
||||||
# Fetch the package.json file content
|
|
||||||
# Extract version from package.json content
|
|
||||||
package_json=$(wget -qO- "$REPO_URL/raw/$LATEST_HASH/package.json")
|
|
||||||
PACKAGE_VERSION=$(echo "$package_json" | jq -r '.version')
|
|
||||||
|
|
||||||
# Current date and time for note
|
|
||||||
CURRENT_DATE_TIME=$(date -u)
|
|
||||||
|
|
||||||
CONFIG_FILE=packages/deployer/config.yml
|
|
||||||
REGISTRY_BOND_ID="99c0e9aec0ac1b8187faa579be3b54f93fafb6060ac1fd29170b860df605be32"
|
|
||||||
|
|
||||||
# Reference: https://git.vdb.to/cerc-io/test-progressive-web-app/src/branch/main/scripts
|
|
||||||
|
|
||||||
APP_NAME=deployment-test-app
|
|
||||||
|
|
||||||
# Get latest version from registry and increment application-record version
|
|
||||||
NEW_APPLICATION_VERSION=$(yarn --silent laconic -c $CONFIG_FILE registry record list --type ApplicationRecord --all --name "$APP_NAME" 2>/dev/null | jq -r -s ".[] | sort_by(.createTime) | reverse | [ .[] | select(.bondId == \"$REGISTRY_BOND_ID\") ] | .[0].attributes.version" | awk -F. -v OFS=. '{$NF += 1 ; print}')
|
|
||||||
|
|
||||||
if [ -z "$NEW_APPLICATION_VERSION" ] || [ "1" == "$NEW_APPLICATION_VERSION" ]; then
|
|
||||||
# Set application-record version if no previous records were found
|
|
||||||
NEW_APPLICATION_VERSION=0.0.1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Generate application-record.yml with incremented version
|
|
||||||
RECORD_FILE=packages/deployer/test/records/application-record.yml
|
|
||||||
|
|
||||||
cat >$RECORD_FILE <<EOF
|
|
||||||
record:
|
|
||||||
type: ApplicationRecord
|
|
||||||
version: $NEW_APPLICATION_VERSION
|
|
||||||
repository_ref: $LATEST_HASH
|
|
||||||
repository: ["$REPO_URL"]
|
|
||||||
app_type: webapp
|
|
||||||
name: $APP_NAME
|
|
||||||
app_version: $PACKAGE_VERSION
|
|
||||||
EOF
|
|
||||||
|
|
||||||
# Generate application-deployment-request.yml
|
|
||||||
REQUEST_RECORD_FILE=packages/deployer/test/records/application-deployment-request.yml
|
|
||||||
|
|
||||||
cat >$REQUEST_RECORD_FILE <<EOF
|
|
||||||
record:
|
|
||||||
type: ApplicationDeploymentRequest
|
|
||||||
version: '1.0.0'
|
|
||||||
name: $APP_NAME@$PACKAGE_VERSION
|
|
||||||
application: lrn://snowballtools/applications/$APP_NAME@$PACKAGE_VERSION
|
|
||||||
dns: deployment-ci-test
|
|
||||||
config:
|
|
||||||
env:
|
|
||||||
CERC_TEST_WEBAPP_CONFIG1: "deployment test config 1"
|
|
||||||
CERC_TEST_WEBAPP_CONFIG2: "deployment test config 2"
|
|
||||||
CERC_WEBAPP_DEBUG: 0
|
|
||||||
meta:
|
|
||||||
note: Deployment test @ $CURRENT_DATE_TIME
|
|
||||||
repository: "$REPO_URL"
|
|
||||||
repository_ref: $LATEST_HASH
|
|
||||||
EOF
|
|
||||||
|
|
||||||
echo "Record files generated successfully."
|
|
||||||
|
|
||||||
# Publish ApplicationRecord
|
|
||||||
RECORD_ID=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $RECORD_FILE | jq -r '.id')
|
|
||||||
echo "ApplicationRecord published"
|
|
||||||
echo $RECORD_ID
|
|
||||||
|
|
||||||
# Set name to record
|
|
||||||
REGISTRY_APP_LRN="lrn://snowballtools/applications/$APP_NAME"
|
|
||||||
|
|
||||||
sleep 2
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN@${PACKAGE_VERSION}" "$RECORD_ID"
|
|
||||||
sleep 2
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN@${LATEST_HASH}" "$RECORD_ID"
|
|
||||||
sleep 2
|
|
||||||
# Set name if latest release
|
|
||||||
yarn --silent laconic -c $CONFIG_FILE registry name set "$REGISTRY_APP_LRN" "$RECORD_ID"
|
|
||||||
echo "$REGISTRY_APP_LRN set for ApplicationRecord"
|
|
||||||
|
|
||||||
# Check if record exists for REGISTRY_APP_LRN
|
|
||||||
APP_RECORD=$(yarn --silent laconic -c $CONFIG_FILE registry name resolve "$REGISTRY_APP_LRN" | jq '.[0]')
|
|
||||||
if [ -z "$APP_RECORD" ] || [ "null" == "$APP_RECORD" ]; then
|
|
||||||
echo "No record found for $REGISTRY_APP_LRN."
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
sleep 2
|
|
||||||
DEPLOYMENT_REQUEST_ID=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $REQUEST_RECORD_FILE | jq -r '.id')
|
|
||||||
echo "ApplicationDeploymentRequest published"
|
|
||||||
echo $DEPLOYMENT_REQUEST_ID
|
|
||||||
|
|
||||||
# Deployment checks
|
|
||||||
RETRY_INTERVAL=30
|
|
||||||
MAX_RETRIES=20
|
|
||||||
|
|
||||||
# Check that a ApplicationDeploymentRecord is published
|
|
||||||
retry_count=0
|
|
||||||
while true; do
|
|
||||||
deployment_records_response=$(yarn --silent laconic -c $CONFIG_FILE registry record list --type ApplicationDeploymentRecord --all --name "$APP_NAME" request $DEPLOYMENT_REQUEST_ID)
|
|
||||||
len_deployment_records=$(echo $deployment_records_response | jq 'length')
|
|
||||||
|
|
||||||
# Check if number of records returned is 0
|
|
||||||
if [ $len_deployment_records -eq 0 ]; then
|
|
||||||
# Check if retries are exhausted
|
|
||||||
if [ $retry_count -eq $MAX_RETRIES ]; then
|
|
||||||
echo "Retries exhausted"
|
|
||||||
echo "ApplicationDeploymentRecord for deployment request $DEPLOYMENT_REQUEST_ID not found, exiting"
|
|
||||||
exit 1
|
|
||||||
else
|
|
||||||
echo "ApplicationDeploymentRecord not found, retrying in $RETRY_INTERVAL sec..."
|
|
||||||
sleep $RETRY_INTERVAL
|
|
||||||
retry_count=$((retry_count + 1))
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "ApplicationDeploymentRecord found"
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
DEPLOYMENT_RECORD_ID=$(echo $deployment_records_response | jq -r '.[0].id')
|
|
||||||
echo $DEPLOYMENT_RECORD_ID
|
|
||||||
|
|
||||||
# Check if ApplicationDeploymentRecord has the correct record id
|
|
||||||
fetched_application_record_id=$(echo $deployment_records_response | jq -r '.[0].attributes.application')
|
|
||||||
if [ "$fetched_application_record_id" = "$RECORD_ID" ]; then
|
|
||||||
echo "ApplicationRecord id matched"
|
|
||||||
else
|
|
||||||
echo "ApplicationRecord id does not match, expected: $RECORD_ID, received: $fetched_application_record_id"
|
|
||||||
exit 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Check if the url present in ApplicationDeploymentRecord is active
|
|
||||||
fetched_url=$(echo $deployment_records_response | jq -r '.[0].attributes.url')
|
|
||||||
|
|
||||||
retry_count=0
|
|
||||||
max_retries=10
|
|
||||||
retry_interval=10
|
|
||||||
while true; do
|
|
||||||
url_response=$(curl -s -o /dev/null -I -w "%{http_code}" $fetched_url)
|
|
||||||
if [ "$url_response" = "200" ]; then
|
|
||||||
echo "Deployment URL $fetched_url is active"
|
|
||||||
break
|
|
||||||
else
|
|
||||||
if [ $retry_count -eq $max_retries ]; then
|
|
||||||
echo "Retries exhausted"
|
|
||||||
echo "Deployment URL $fetched_url is not active, exiting"
|
|
||||||
exit 1
|
|
||||||
else
|
|
||||||
echo "Deployment URL $fetched_url is not active, received code $url_response, retrying in $retry_interval sec..."
|
|
||||||
sleep $retry_interval
|
|
||||||
retry_count=$((retry_count + 1))
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
# Generate application-deployment-removal-request.yml
|
|
||||||
REMOVAL_REQUEST_RECORD_FILE=packages/deployer/test/records/application-deployment-removal-request.yml
|
|
||||||
|
|
||||||
cat >$REMOVAL_REQUEST_RECORD_FILE <<EOF
|
|
||||||
record:
|
|
||||||
deployment: $DEPLOYMENT_RECORD_ID
|
|
||||||
type: ApplicationDeploymentRemovalRequest
|
|
||||||
version: 1.0.0
|
|
||||||
EOF
|
|
||||||
|
|
||||||
sleep 2
|
|
||||||
REMOVAL_REQUEST_ID=$(yarn --silent laconic -c $CONFIG_FILE registry record publish --filename $REMOVAL_REQUEST_RECORD_FILE | jq -r '.id')
|
|
||||||
echo "ApplicationDeploymentRemovalRequest published"
|
|
||||||
echo $REMOVAL_REQUEST_ID
|
|
||||||
|
|
||||||
# Check that an ApplicationDeploymentRemovalRecord is published
|
|
||||||
retry_count=0
|
|
||||||
while true; do
|
|
||||||
removal_records_response=$(yarn --silent laconic -c $CONFIG_FILE registry record list --type ApplicationDeploymentRemovalRecord --all request $REMOVAL_REQUEST_ID)
|
|
||||||
len_removal_records=$(echo $removal_records_response | jq 'length')
|
|
||||||
|
|
||||||
# Check if number of records returned is 0
|
|
||||||
if [ $len_removal_records -eq 0 ]; then
|
|
||||||
# Check if retries are exhausted
|
|
||||||
if [ $retry_count -eq $MAX_RETRIES ]; then
|
|
||||||
echo "Retries exhausted"
|
|
||||||
echo "ApplicationDeploymentRemovalRecord for deployment removal request $REMOVAL_REQUEST_ID not found"
|
|
||||||
exit 1
|
|
||||||
else
|
|
||||||
echo "ApplicationDeploymentRemovalRecord not found, retrying in $RETRY_INTERVAL sec..."
|
|
||||||
sleep $RETRY_INTERVAL
|
|
||||||
retry_count=$((retry_count + 1))
|
|
||||||
fi
|
|
||||||
else
|
|
||||||
echo "ApplicationDeploymentRemovalRecord found"
|
|
||||||
REMOVAL_RECORD_ID=$(echo $removal_records_response | jq -r '.[0].id')
|
|
||||||
echo $REMOVAL_RECORD_ID
|
|
||||||
break
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
# Check if the application url is down after deployment removal
|
|
||||||
retry_count=0
|
|
||||||
max_retries=10
|
|
||||||
retry_interval=5
|
|
||||||
while true; do
|
|
||||||
url_response=$(curl -s -o /dev/null -I -w "%{http_code}" $fetched_url)
|
|
||||||
if [ "$url_response" = "404" ]; then
|
|
||||||
echo "Deployment URL $fetched_url is down"
|
|
||||||
break
|
|
||||||
else
|
|
||||||
if [ $retry_count -eq $max_retries ]; then
|
|
||||||
echo "Retries exhausted"
|
|
||||||
echo "Deployment URL $fetched_url is still active, exiting"
|
|
||||||
exit 1
|
|
||||||
else
|
|
||||||
echo "Deployment URL $fetched_url is still active, received code $url_response, retrying in $retry_interval sec..."
|
|
||||||
sleep $retry_interval
|
|
||||||
retry_count=$((retry_count + 1))
|
|
||||||
fi
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
echo "Test successful"
|
|
@ -1,19 +1,7 @@
|
|||||||
VITE_SERVER_URL='http://localhost:8000'
|
REACT_APP_SERVER_URL = 'http://localhost:8000'
|
||||||
|
|
||||||
VITE_GITHUB_CLIENT_ID=
|
REACT_APP_GITHUB_CLIENT_ID =
|
||||||
VITE_GITHUB_PWA_TEMPLATE_REPO="snowball-tools/test-progressive-web-app"
|
REACT_APP_GITHUB_PWA_TEMPLATE_REPO =
|
||||||
VITE_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO="snowball-tools/image-upload-pwa-example"
|
REACT_APP_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO =
|
||||||
|
|
||||||
VITE_WALLET_CONNECT_ID=
|
REACT_APP_WALLET_CONNECT_ID =
|
||||||
|
|
||||||
VITE_LIT_RELAY_API_KEY=
|
|
||||||
|
|
||||||
VITE_ALCHEMY_API_KEY=
|
|
||||||
|
|
||||||
VITE_BUGSNAG_API_KEY=
|
|
||||||
|
|
||||||
VITE_PASSKEY_WALLET_RPID=
|
|
||||||
VITE_TURNKEY_API_BASE_URL=
|
|
||||||
VITE_TURNKEY_ORGANIZATION_ID=
|
|
||||||
|
|
||||||
VITE_LACONICD_CHAIN_ID=
|
|
1
packages/frontend/.eslintignore
Normal file
1
packages/frontend/.eslintignore
Normal file
@ -0,0 +1 @@
|
|||||||
|
build
|
@ -1,19 +0,0 @@
|
|||||||
module.exports = {
|
|
||||||
root: true,
|
|
||||||
env: { browser: true, es2020: true },
|
|
||||||
extends: [
|
|
||||||
'eslint:recommended',
|
|
||||||
'plugin:@typescript-eslint/recommended',
|
|
||||||
'plugin:react-hooks/recommended',
|
|
||||||
'plugin:storybook/recommended',
|
|
||||||
],
|
|
||||||
ignorePatterns: ['dist', '.eslintrc.cjs'],
|
|
||||||
parser: '@typescript-eslint/parser',
|
|
||||||
plugins: ['react-refresh'],
|
|
||||||
rules: {
|
|
||||||
'react-refresh/only-export-components': [
|
|
||||||
'warn',
|
|
||||||
{ allowConstantExport: true },
|
|
||||||
],
|
|
||||||
},
|
|
||||||
};
|
|
25
packages/frontend/.eslintrc.json
Normal file
25
packages/frontend/.eslintrc.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"parser": "@typescript-eslint/parser",
|
||||||
|
"parserOptions": {
|
||||||
|
"ecmaFeatures": {
|
||||||
|
"jsx": true
|
||||||
|
},
|
||||||
|
"ecmaVersion": 13,
|
||||||
|
"sourceType": "module"
|
||||||
|
},
|
||||||
|
"env": {
|
||||||
|
"browser": true,
|
||||||
|
"es2021": true
|
||||||
|
},
|
||||||
|
"plugins": ["react", "@typescript-eslint"],
|
||||||
|
"extends": [
|
||||||
|
"plugin:react/recommended",
|
||||||
|
"plugin:@typescript-eslint/recommended",
|
||||||
|
"plugin:prettier/recommended"
|
||||||
|
],
|
||||||
|
"settings": {
|
||||||
|
"react": {
|
||||||
|
"version": "detect"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
1
packages/frontend/.gitignore
vendored
1
packages/frontend/.gitignore
vendored
@ -22,4 +22,3 @@
|
|||||||
npm-debug.log*
|
npm-debug.log*
|
||||||
yarn-debug.log*
|
yarn-debug.log*
|
||||||
yarn-error.log*
|
yarn-error.log*
|
||||||
*storybook.log
|
|
@ -1 +0,0 @@
|
|||||||
v20.12.1
|
|
@ -1 +1,3 @@
|
|||||||
dist/
|
# artifacts
|
||||||
|
build
|
||||||
|
coverage
|
||||||
|
@ -1,33 +0,0 @@
|
|||||||
import type { StorybookConfig } from '@storybook/react-vite';
|
|
||||||
|
|
||||||
import { join, dirname } from 'path';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This function is used to resolve the absolute path of a package.
|
|
||||||
* It is needed in projects that use Yarn PnP or are set up within a monorepo.
|
|
||||||
*/
|
|
||||||
function getAbsolutePath(value: string): any {
|
|
||||||
return dirname(require.resolve(join(value, 'package.json')));
|
|
||||||
}
|
|
||||||
|
|
||||||
const config: StorybookConfig = {
|
|
||||||
stories: ['../src/**/*.mdx', '../src/**/*.stories.@(js|jsx|mjs|ts|tsx)'],
|
|
||||||
addons: [
|
|
||||||
getAbsolutePath('@storybook/addon-onboarding'),
|
|
||||||
getAbsolutePath('@storybook/addon-links'),
|
|
||||||
getAbsolutePath('@storybook/addon-essentials'),
|
|
||||||
getAbsolutePath('@chromatic-com/storybook'),
|
|
||||||
getAbsolutePath('@storybook/addon-interactions'),
|
|
||||||
getAbsolutePath('storybook-addon-remix-react-router'),
|
|
||||||
],
|
|
||||||
framework: {
|
|
||||||
name: getAbsolutePath('@storybook/react-vite'),
|
|
||||||
options: {},
|
|
||||||
},
|
|
||||||
docs: {
|
|
||||||
autodocs: 'tag',
|
|
||||||
},
|
|
||||||
staticDirs: ['../public'],
|
|
||||||
};
|
|
||||||
|
|
||||||
export default config;
|
|
@ -1,16 +0,0 @@
|
|||||||
import type { Preview } from '@storybook/react';
|
|
||||||
|
|
||||||
import '../src/index.css';
|
|
||||||
|
|
||||||
const preview: Preview = {
|
|
||||||
parameters: {
|
|
||||||
controls: {
|
|
||||||
matchers: {
|
|
||||||
color: /(background|color)$/i,
|
|
||||||
date: /Date$/i,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
export default preview;
|
|
@ -1,63 +1,46 @@
|
|||||||
# frontend
|
# Getting Started with Create React App
|
||||||
|
|
||||||
This is a [vite](https://vitejs.dev/) [react](https://reactjs.org/) [nextjs](https://nextjs.org/) project in a [yarn workspace](https://yarnpkg.com/features/workspaces).
|
This project was bootstrapped with [Create React App](https://github.com/facebook/create-react-app), using [typescript-tailwindcss-eslint-prettier](https://github.com/cufarvid/cra-templates) template.
|
||||||
|
|
||||||
## Getting Started
|
## Available Scripts
|
||||||
|
|
||||||
### Install dependencies
|
In the project directory, you can run:
|
||||||
|
|
||||||
In the root of the project, run:
|
### `yarn start`
|
||||||
|
|
||||||
```zsh
|
Runs the app in the development mode.\
|
||||||
yarn
|
Open [http://localhost:3000](http://localhost:3000) to view it in the browser.
|
||||||
```
|
|
||||||
|
|
||||||
### Build backend
|
The page will reload if you make edits.\
|
||||||
|
You will also see any lint errors in the console.
|
||||||
|
|
||||||
```zsh
|
### `yarn test`
|
||||||
yarn build --ignore frontend
|
|
||||||
```
|
|
||||||
|
|
||||||
### Environment variables
|
Launches the test runner in the interactive watch mode.\
|
||||||
|
See the section about [running tests](https://facebook.github.io/create-react-app/docs/running-tests) for more information.
|
||||||
|
|
||||||
#### Local
|
### `yarn build`
|
||||||
|
|
||||||
Copy the `.env.example` file to `.env`:
|
Builds the app for production to the `build` folder.\
|
||||||
|
It correctly bundles React in production mode and optimizes the build for the best performance.
|
||||||
|
|
||||||
```zsh
|
The build is minified and the filenames include the hashes.\
|
||||||
cp .env.example .env
|
Your app is ready to be deployed!
|
||||||
```
|
|
||||||
|
|
||||||
#### Staging environment variables
|
See the section about [deployment](https://facebook.github.io/create-react-app/docs/deployment) for more information.
|
||||||
|
|
||||||
Change in [deployer/deploy-frontend.staging.sh](/packages/deployer/deploy-frontend.staging.sh)
|
### `yarn eject`
|
||||||
|
|
||||||
#### Production environment variables
|
**Note: this is a one-way operation. Once you `eject`, you can’t go back!**
|
||||||
|
|
||||||
Change in [deployer/deploy-frontend.sh](/packages/deployer/deploy-frontend.sh)
|
If you aren’t satisfied with the build tool and configuration choices, you can `eject` at any time. This command will remove the single build dependency from your project.
|
||||||
|
|
||||||
### Run development server
|
Instead, it will copy all the configuration files and the transitive dependencies (webpack, Babel, ESLint, etc) right into your project so you have full control over them. All of the commands except `eject` will still work, but they will point to the copied scripts so you can tweak them. At this point you’re on your own.
|
||||||
|
|
||||||
```zsh
|
You don’t have to ever use `eject`. The curated feature set is suitable for small and middle deployments, and you shouldn’t feel obligated to use this feature. However we understand that this tool wouldn’t be useful if you couldn’t customize it when you are ready for it.
|
||||||
yarn dev
|
|
||||||
```
|
|
||||||
|
|
||||||
## Deployment
|
## Learn More
|
||||||
|
|
||||||
From the root of the project,
|
You can learn more in the [Create React App documentation](https://facebook.github.io/create-react-app/docs/getting-started).
|
||||||
|
|
||||||
### Staging
|
To learn React, check out the [React documentation](https://reactjs.org/).
|
||||||
|
|
||||||
```zsh
|
|
||||||
cd packages/deployer && ./deploy-frontend.staging.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
### Production
|
|
||||||
|
|
||||||
```zsh
|
|
||||||
cd packages/deployer && ./deploy-frontend.sh
|
|
||||||
```
|
|
||||||
|
|
||||||
### Deployment status
|
|
||||||
|
|
||||||
Check the status of the deployment [here](https://webapp-deployer.apps.snowballtools.com)
|
|
||||||
|
@ -1,4 +0,0 @@
|
|||||||
{
|
|
||||||
"projectId": "Project:663d04870db27ed66a48e466",
|
|
||||||
"zip": true
|
|
||||||
}
|
|
@ -1,22 +0,0 @@
|
|||||||
<!doctype html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="utf-8" />
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1" />
|
|
||||||
<meta name="description" content="snowball tools dashboard" />
|
|
||||||
<link rel="icon" href="/favicon.ico" />
|
|
||||||
<link rel="apple-touch-icon" href="/logo192.png" />
|
|
||||||
<link rel="apple-touch-icon" sizes="180x180" href="/apple-touch-icon.png" />
|
|
||||||
<link rel="icon" type="image/png" sizes="32x32" href="/favicon-32x32.png" />
|
|
||||||
<link rel="icon" type="image/png" sizes="16x16" href="/favicon-16x16.png" />
|
|
||||||
<link rel="manifest" href="/site.webmanifest" />
|
|
||||||
<meta name="msapplication-TileColor" content="#2d89ef" />
|
|
||||||
<meta name="theme-color" content="#ffffff" />
|
|
||||||
<link rel="manifest" href="/manifest.json" />
|
|
||||||
<title>Snowball</title>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<div id="root"></div>
|
|
||||||
<script type="module" src="/src/index.tsx"></script>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
7
packages/frontend/logo.svg
Normal file
7
packages/frontend/logo.svg
Normal file
@ -0,0 +1,7 @@
|
|||||||
|
<svg width="500" height="500" viewBox="0 0 500 500" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<rect width="500" height="500" fill="#0F86F5"/>
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M191.873 125.126C224.893 126.765 250.458 150.121 274.042 172.995C297.925 196.158 323.089 221.108 324.868 254.114C326.718 288.42 308.902 321.108 283.281 344.355C258.67 366.687 225.288 373.859 191.873 374.788C157.228 375.752 119.038 374.394 95.1648 349.588C71.6207 325.125 74.6696 287.843 75.7341 254.114C76.7518 221.865 79.2961 188.525 101.009 164.41C123.845 139.047 157.543 123.423 191.873 125.126Z" fill="#4BA4F7"/>
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M229.373 125.126C262.393 126.765 287.958 150.121 311.542 172.995C335.425 196.158 360.589 221.108 362.368 254.114C364.218 288.42 346.402 321.108 320.781 344.355C296.17 366.687 262.788 373.859 229.373 374.788C194.728 375.752 156.538 374.394 132.665 349.588C109.121 325.125 112.17 287.843 113.234 254.114C114.252 221.865 116.796 188.525 138.509 164.41C161.345 139.047 195.043 123.423 229.373 125.126Z" fill="#8AC4FA"/>
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M266.873 125.126C299.893 126.765 325.458 150.121 349.042 172.995C372.925 196.158 398.089 221.108 399.868 254.114C401.718 288.42 383.902 321.108 358.281 344.355C333.67 366.687 300.288 373.859 266.873 374.788C232.228 375.752 194.038 374.394 170.165 349.588C146.621 325.125 149.67 287.843 150.734 254.114C151.752 221.865 154.296 188.525 176.009 164.41C198.845 139.047 232.543 123.423 266.873 125.126Z" fill="#CAE4FD"/>
|
||||||
|
<path fill-rule="evenodd" clip-rule="evenodd" d="M304.373 125.126C337.393 126.765 362.958 150.121 386.542 172.995C410.425 196.158 435.589 221.108 437.368 254.114C439.218 288.42 421.402 321.108 395.781 344.355C371.17 366.687 337.788 373.859 304.373 374.788C269.728 375.752 231.538 374.394 207.665 349.588C184.121 325.125 187.17 287.843 188.234 254.114C189.252 221.865 191.796 188.525 213.509 164.41C236.345 139.047 270.043 123.423 304.373 125.126Z" fill="white"/>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 2.0 KiB |
@ -1,57 +1,36 @@
|
|||||||
{
|
{
|
||||||
"name": "frontend",
|
"name": "frontend",
|
||||||
|
"version": "0.1.3",
|
||||||
"private": true,
|
"private": true,
|
||||||
"version": "1.0.0",
|
|
||||||
"type": "module",
|
|
||||||
"scripts": {
|
|
||||||
"dev": "vite --port 3000",
|
|
||||||
"build": "vite build",
|
|
||||||
"lint": "tsc --noEmit",
|
|
||||||
"preview": "vite preview",
|
|
||||||
"format": "prettier --write .",
|
|
||||||
"storybook": "storybook dev -p 6006",
|
|
||||||
"build-storybook": "storybook build"
|
|
||||||
},
|
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@bugsnag/browser-performance": "^2.4.1",
|
|
||||||
"@bugsnag/js": "^7.22.7",
|
|
||||||
"@bugsnag/plugin-react": "^7.22.7",
|
|
||||||
"@emotion/react": "^11.13.3",
|
|
||||||
"@emotion/styled": "^11.13.0",
|
|
||||||
"@fontsource-variable/jetbrains-mono": "^5.0.19",
|
"@fontsource-variable/jetbrains-mono": "^5.0.19",
|
||||||
"@fontsource/inter": "^5.0.16",
|
"@fontsource/inter": "^5.0.16",
|
||||||
"@mui/material": "^6.1.3",
|
"@material-tailwind/react": "^2.1.7",
|
||||||
"@radix-ui/react-avatar": "^1.0.4",
|
"@radix-ui/react-avatar": "^1.0.4",
|
||||||
"@radix-ui/react-checkbox": "^1.0.4",
|
"@radix-ui/react-checkbox": "^1.0.4",
|
||||||
"@radix-ui/react-dialog": "^1.0.5",
|
|
||||||
"@radix-ui/react-popover": "^1.0.7",
|
"@radix-ui/react-popover": "^1.0.7",
|
||||||
"@radix-ui/react-radio-group": "^1.1.3",
|
"@radix-ui/react-radio-group": "^1.1.3",
|
||||||
"@radix-ui/react-switch": "^1.0.3",
|
"@radix-ui/react-switch": "^1.0.3",
|
||||||
"@radix-ui/react-tabs": "^1.0.4",
|
"@radix-ui/react-tabs": "^1.0.4",
|
||||||
"@radix-ui/react-toast": "^1.1.5",
|
"@radix-ui/react-toast": "^1.1.5",
|
||||||
"@radix-ui/react-tooltip": "^1.0.7",
|
"@radix-ui/react-tooltip": "^1.0.7",
|
||||||
"@snowballtools/material-tailwind-react-fork": "^2.1.10",
|
|
||||||
"@snowballtools/smartwallet-alchemy-light": "^0.2.0",
|
|
||||||
"@snowballtools/types": "^0.2.0",
|
|
||||||
"@snowballtools/utils": "^0.1.1",
|
|
||||||
"@tanstack/react-query": "^5.22.2",
|
"@tanstack/react-query": "^5.22.2",
|
||||||
"@testing-library/jest-dom": "^5.17.0",
|
"@testing-library/jest-dom": "^5.17.0",
|
||||||
"@testing-library/react": "^13.4.0",
|
"@testing-library/react": "^13.4.0",
|
||||||
"@testing-library/user-event": "^13.5.0",
|
"@testing-library/user-event": "^13.5.0",
|
||||||
"@turnkey/http": "^2.10.0",
|
"@types/jest": "^27.5.2",
|
||||||
"@turnkey/sdk-react": "^0.1.0",
|
"@types/node": "^16.18.68",
|
||||||
"@turnkey/webauthn-stamper": "^0.5.0",
|
"@types/react": "^18.2.42",
|
||||||
"@walletconnect/ethereum-provider": "^2.12.2",
|
"@types/react-dom": "^18.2.17",
|
||||||
"@web3modal/siwe": "4.0.5",
|
"@web3modal/siwe": "^4.0.5",
|
||||||
"@web3modal/wagmi": "4.0.5",
|
"@web3modal/wagmi": "^4.0.5",
|
||||||
"assert": "^2.1.0",
|
"assert": "^2.1.0",
|
||||||
"axios": "^1.6.7",
|
"axios": "^1.6.7",
|
||||||
"clsx": "^2.1.0",
|
"clsx": "^2.1.0",
|
||||||
"date-fns": "^3.3.1",
|
"date-fns": "^3.3.1",
|
||||||
"downshift": "^8.3.2",
|
"downshift": "^8.3.2",
|
||||||
"framer-motion": "^11.0.8",
|
"eslint-config-react-app": "^7.0.1",
|
||||||
"gql-client": "^1.0.0",
|
"gql-client": "^1.0.0",
|
||||||
"lottie-react": "^2.4.0",
|
|
||||||
"luxon": "^3.4.4",
|
"luxon": "^3.4.4",
|
||||||
"octokit": "^3.1.2",
|
"octokit": "^3.1.2",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
@ -61,44 +40,57 @@
|
|||||||
"react-dom": "^18.2.0",
|
"react-dom": "^18.2.0",
|
||||||
"react-dropdown": "^1.11.0",
|
"react-dropdown": "^1.11.0",
|
||||||
"react-hook-form": "^7.49.0",
|
"react-hook-form": "^7.49.0",
|
||||||
|
"react-hot-toast": "^2.4.1",
|
||||||
"react-oauth-popup": "^1.0.5",
|
"react-oauth-popup": "^1.0.5",
|
||||||
"react-router-dom": "^6.20.1",
|
"react-router-dom": "^6.20.1",
|
||||||
|
"react-scripts": "5.0.1",
|
||||||
"react-timer-hook": "^3.0.7",
|
"react-timer-hook": "^3.0.7",
|
||||||
"siwe": "2.1.4",
|
"siwe": "^2.1.4",
|
||||||
"tailwind-variants": "^0.2.0",
|
"tailwind-variants": "^0.2.0",
|
||||||
"usehooks-ts": "^2.15.1",
|
"typescript": "^4.9.5",
|
||||||
"uuid": "^9.0.1",
|
"usehooks-ts": "^2.10.0",
|
||||||
|
"vertical-stepper-nav": "^1.0.2",
|
||||||
"viem": "^2.7.11",
|
"viem": "^2.7.11",
|
||||||
"wagmi": "2.5.7",
|
"wagmi": "^2.5.7",
|
||||||
"web-vitals": "^2.1.4"
|
"web-vitals": "^2.1.4"
|
||||||
},
|
},
|
||||||
|
"scripts": {
|
||||||
|
"start": "react-scripts start",
|
||||||
|
"build": "react-scripts build",
|
||||||
|
"test": "react-scripts test",
|
||||||
|
"eject": "react-scripts eject",
|
||||||
|
"format": "prettier --write .",
|
||||||
|
"format:check": "prettier --check .",
|
||||||
|
"lint": "eslint ."
|
||||||
|
},
|
||||||
|
"eslintConfig": {
|
||||||
|
"extends": [
|
||||||
|
"react-app",
|
||||||
|
"react-app/jest"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
"browserslist": {
|
||||||
|
"production": [
|
||||||
|
">0.2%",
|
||||||
|
"not dead",
|
||||||
|
"not op_mini all"
|
||||||
|
],
|
||||||
|
"development": [
|
||||||
|
"last 1 chrome version",
|
||||||
|
"last 1 firefox version",
|
||||||
|
"last 1 safari version"
|
||||||
|
]
|
||||||
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@chromatic-com/storybook": "^1.3.3",
|
"@babel/plugin-proposal-private-property-in-object": "^7.21.11",
|
||||||
"@storybook/addon-essentials": "^8.0.10",
|
|
||||||
"@storybook/addon-interactions": "^8.0.10",
|
|
||||||
"@storybook/addon-links": "^8.0.10",
|
|
||||||
"@storybook/addon-onboarding": "^8.0.10",
|
|
||||||
"@storybook/blocks": "^8.0.10",
|
|
||||||
"@storybook/react": "^8.0.10",
|
|
||||||
"@storybook/react-vite": "^8.0.10",
|
|
||||||
"@storybook/test": "^8.0.10",
|
|
||||||
"@types/jest": "^27.5.2",
|
|
||||||
"@types/lodash": "^4.17.0",
|
|
||||||
"@types/luxon": "^3.3.7",
|
"@types/luxon": "^3.3.7",
|
||||||
"@types/node": "^16.18.68",
|
"@typescript-eslint/eslint-plugin": "^6.13.2",
|
||||||
"@types/react": "^18.2.66",
|
"@typescript-eslint/parser": "^6.13.2",
|
||||||
"@types/react-dom": "^18.2.22",
|
"eslint": "^8.55.0",
|
||||||
"@types/uuid": "^9.0.8",
|
"eslint-config-prettier": "^9.1.0",
|
||||||
"@vitejs/plugin-react": "^4.2.1",
|
"eslint-plugin-prettier": "^5.0.1",
|
||||||
"autoprefixer": "^10.4.19",
|
"eslint-plugin-react": "^7.33.2",
|
||||||
"chromatic": "^11.3.2",
|
|
||||||
"eslint-plugin-storybook": "^0.8.0",
|
|
||||||
"postcss": "^8.4.38",
|
|
||||||
"prettier": "^3.1.0",
|
"prettier": "^3.1.0",
|
||||||
"storybook": "^8.0.10",
|
"tailwindcss": "^3.4.1"
|
||||||
"storybook-addon-remix-react-router": "^3.0.0",
|
|
||||||
"tailwindcss": "^3.4.3",
|
|
||||||
"typescript": "^5.3.3",
|
|
||||||
"vite": "^5.2.0"
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,6 +0,0 @@
|
|||||||
export default {
|
|
||||||
plugins: {
|
|
||||||
tailwindcss: {},
|
|
||||||
autoprefixer: {},
|
|
||||||
},
|
|
||||||
};
|
|
@ -1 +0,0 @@
|
|||||||
350e9ac2-8b27-4a79-9a82-78cfdb68ef71=0eacb7ae462f82c8b0199d28193b0bfa5265973dbb1fe991eec2cab737dfc1ec
|
|
@ -1,3 +0,0 @@
|
|||||||
<svg width="197" height="2" viewBox="0 0 197 2" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<line x1="0.5" y1="1.19141" x2="197" y2="1.19141" stroke="#94A7B8" stroke-dasharray="1 12"/>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 196 B |
File diff suppressed because it is too large
Load Diff
10
packages/frontend/public/wave-border.svg
Normal file
10
packages/frontend/public/wave-border.svg
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
<svg width="333" height="5" viewBox="0 0 333 5" fill="none" xmlns="http://www.w3.org/2000/svg">
|
||||||
|
<path d="M0 4L6.39555 1.30714C8.38078 0.47125 10.6192 0.47125 12.6045 1.30714L15.8955 2.69286C17.8808 3.52875 20.1192 3.52875 22.1045 2.69286L25.3955 1.30714C27.3808 0.47125 29.6192 0.47125 31.6045 1.30714L34.8955 2.69286C36.8808 3.52875 39.1192 3.52875 41.1045 2.69286L44.3955 1.30714C46.3808 0.47125 48.6192 0.47125 50.6045 1.30714L53.8955 2.69286C55.8808 3.52875 58.1192 3.52875 60.1045 2.69286L63.3955 1.30714C65.3808 0.47125 67.6192 0.47125 69.6045 1.30714L72.8955 2.69286C74.8808 3.52875 77.1192 3.52875 79.1045 2.69286L82.3955 1.30714C84.3808 0.47125 86.6192 0.47125 88.6045 1.30714L91.8955 2.69286C93.8808 3.52875 96.1192 3.52875 98.1045 2.69286L101.396 1.30714C103.381 0.47125 105.619 0.47125 107.604 1.30714L110.896 2.69286C112.881 3.52875 115.119 3.52875 117.104 2.69286L120.396 1.30714C122.381 0.47125 124.619 0.47125 126.604 1.30714L129.896 2.69286C131.881 3.52875 134.119 3.52875 136.104 2.69286L139.396 1.30714C141.381 0.47125 143.619 0.47125 145.604 1.30714L148.896 2.69286C150.881 3.52875 153.119 3.52875 155.104 2.69286L158.396 1.30714C160.381 0.47125 162.619 0.47125 164.604 1.30714L167.896 2.69286C169.881 3.52875 172.119 3.52875 174.104 2.69286L177.396 1.30714C179.381 0.47125 181.619 0.47125 183.604 1.30714L186.896 2.69286C188.881 3.52875 191.119 3.52875 193.104 2.69286L196.396 1.30714C198.381 0.47125 200.619 0.47125 202.604 1.30714L205.896 2.69286C207.881 3.52875 210.119 3.52875 212.104 2.69286L215.396 1.30714C217.381 0.47125 219.619 0.47125 221.604 1.30714L224.896 2.69286C226.881 3.52875 229.119 3.52875 231.104 2.69286L234.396 1.30714C236.381 0.47125 238.619 0.47125 240.604 1.30714L243.896 2.69286C245.881 3.52875 248.119 3.52875 250.104 2.69286L253.396 1.30714C255.381 0.47125 257.619 0.47125 259.604 1.30714L262.896 2.69286C264.881 3.52875 267.119 3.52875 269.104 2.69286L272.396 1.30714C274.381 0.47125 276.619 0.47125 278.604 1.30714L281.896 2.69286C283.881 3.52875 286.119 3.52875 288.104 2.69286L291.396 1.30714C293.381 0.47125 295.619 0.47125 297.604 1.30714L300.896 2.69286C302.881 3.52875 305.119 3.52875 307.104 2.69286L310.396 1.30714C312.381 0.47125 314.619 0.47125 316.604 1.30714L319.973 2.72566C321.913 3.54216 324.095 3.56183 326.049 2.78039L330.029 1.18845C331.936 0.425535 334.064 0.425535 335.971 1.18845L343 4" stroke="#DBEBF9"/>
|
||||||
|
<path d="M6.39555 1.30714L0 4H342.5L336.027 1.27434C334.087 0.457837 331.905 0.438174 329.951 1.21961L326.049 2.78039C324.095 3.56183 321.913 3.54216 319.973 2.72566L316.604 1.30714C314.619 0.47125 312.381 0.47125 310.396 1.30714L307.104 2.69286C305.119 3.52875 302.881 3.52875 300.896 2.69286L297.604 1.30714C295.619 0.47125 293.381 0.47125 291.396 1.30714L288.104 2.69286C286.119 3.52875 283.881 3.52875 281.896 2.69286L278.604 1.30714C276.619 0.47125 274.381 0.47125 272.396 1.30714L269.104 2.69286C267.119 3.52875 264.881 3.52875 262.896 2.69286L259.604 1.30714C257.619 0.47125 255.381 0.47125 253.396 1.30714L250.104 2.69286C248.119 3.52875 245.881 3.52875 243.896 2.69286L240.604 1.30714C238.619 0.47125 236.381 0.47125 234.396 1.30714L231.104 2.69286C229.119 3.52875 226.881 3.52875 224.896 2.69286L221.604 1.30714C219.619 0.47125 217.381 0.47125 215.396 1.30714L212.104 2.69286C210.119 3.52875 207.881 3.52875 205.896 2.69286L202.604 1.30714C200.619 0.47125 198.381 0.47125 196.396 1.30714L193.104 2.69286C191.119 3.52875 188.881 3.52875 186.896 2.69286L183.604 1.30714C181.619 0.47125 179.381 0.47125 177.396 1.30714L174.104 2.69286C172.119 3.52875 169.881 3.52875 167.896 2.69286L164.604 1.30714C162.619 0.47125 160.381 0.47125 158.396 1.30714L155.104 2.69286C153.119 3.52875 150.881 3.52875 148.896 2.69286L145.604 1.30714C143.619 0.47125 141.381 0.47125 139.396 1.30714L136.104 2.69286C134.119 3.52875 131.881 3.52875 129.896 2.69286L126.604 1.30714C124.619 0.47125 122.381 0.47125 120.396 1.30714L117.104 2.69286C115.119 3.52875 112.881 3.52875 110.896 2.69286L107.604 1.30714C105.619 0.47125 103.381 0.47125 101.396 1.30714L98.1045 2.69286C96.1192 3.52875 93.8808 3.52875 91.8955 2.69286L88.6045 1.30714C86.6192 0.47125 84.3808 0.47125 82.3955 1.30714L79.1045 2.69286C77.1192 3.52875 74.8808 3.52875 72.8955 2.69286L69.6045 1.30714C67.6192 0.47125 65.3808 0.47125 63.3955 1.30714L60.1045 2.69286C58.1192 3.52875 55.8808 3.52875 53.8955 2.69286L50.6045 1.30714C48.6192 0.47125 46.3808 0.47125 44.3955 1.30714L41.1045 2.69286C39.1192 3.52875 36.8808 3.52875 34.8955 2.69286L31.6045 1.30714C29.6192 0.47125 27.3808 0.47125 25.3955 1.30714L22.1045 2.69286C20.1192 3.52875 17.8808 3.52875 15.8955 2.69286L12.6045 1.30714C10.6192 0.47125 8.38078 0.47125 6.39555 1.30714Z" fill="url(#paint0_linear_1729_11298)"/>
|
||||||
|
<defs>
|
||||||
|
<linearGradient id="paint0_linear_1729_11298" x1="171.25" y1="0" x2="171.25" y2="4" gradientUnits="userSpaceOnUse">
|
||||||
|
<stop stop-color="#E6F4FF"/>
|
||||||
|
<stop offset="1" stop-color="#F9FCFF"/>
|
||||||
|
</linearGradient>
|
||||||
|
</defs>
|
||||||
|
</svg>
|
After Width: | Height: | Size: 4.8 KiB |
@ -1,4 +0,0 @@
|
|||||||
<svg width="19" height="4" viewBox="0 0 19 4" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<path d="M-12.6045 1.27694L-19 3.9698H37.5L34.8955 2.66266L31.6045 1.27694C29.6192 0.441052 27.3808 0.441052 25.3955 1.27694L22.1045 2.66266C20.1192 3.49855 17.8808 3.49855 15.8955 2.66266L12.6045 1.27694C10.6192 0.441052 8.38078 0.441052 6.39555 1.27694L3.10446 2.66266C1.11922 3.49855 -1.11922 3.49855 -3.10445 2.66266L-6.39554 1.27694C-8.38078 0.441052 -10.6192 0.441052 -12.6045 1.27694Z" fill="currentColor"/>
|
|
||||||
<path d="M25.7836 1.58773L22.4925 2.97346C20.3832 3.86159 18.0049 3.86159 15.8955 2.97346L12.6045 1.58774C10.7433 0.804089 8.64476 0.804088 6.7836 1.58773L3.49251 2.97346C1.3832 3.86159 -0.99514 3.86159 -3.10445 2.97346L-6.39554 1.58774C-8.2567 0.804089 -10.3552 0.804088 -12.2164 1.58773L-18.6119 4.2806L-19 3.35896L-12.6045 0.666099C-10.4951 -0.222033 -8.1168 -0.222033 -6.00749 0.6661L-2.7164 2.05182C-0.855238 2.83547 1.2433 2.83547 3.10446 2.05182L6.39555 0.666099C8.50486 -0.222033 10.8832 -0.222033 12.9925 0.6661L16.2836 2.05182C18.1448 2.83547 20.2433 2.83547 22.1045 2.05182L25.3955 0.666099L25.7836 1.58773Z" fill="currentColor"/>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.1 KiB |
@ -1,15 +0,0 @@
|
|||||||
<svg width="19" height="4" viewBox="0 0 19 4" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<g clip-path="url(#clip0_1177_19315)">
|
|
||||||
<path d="M-12.6045 1.27694L-19 3.9698H37.5L34.8955 2.66266L31.6045 1.27694C29.6192 0.441052 27.3808 0.441052 25.3955 1.27694L22.1045 2.66266C20.1192 3.49855 17.8808 3.49855 15.8955 2.66266L12.6045 1.27694C10.6192 0.441052 8.38078 0.441052 6.39555 1.27694L3.10446 2.66266C1.11922 3.49855 -1.11922 3.49855 -3.10445 2.66266L-6.39554 1.27694C-8.38078 0.441052 -10.6192 0.441052 -12.6045 1.27694Z" fill="url(#paint0_linear_1177_19315)"/>
|
|
||||||
<path d="M25.7836 1.58773L22.4925 2.97346C20.3832 3.86159 18.0049 3.86159 15.8955 2.97346L12.6045 1.58774C10.7433 0.804089 8.64476 0.804088 6.7836 1.58773L3.49251 2.97346C1.3832 3.86159 -0.99514 3.86159 -3.10445 2.97346L-6.39554 1.58774C-8.2567 0.804089 -10.3552 0.804088 -12.2164 1.58773L-18.6119 4.2806L-19 3.35896L-12.6045 0.666099C-10.4951 -0.222033 -8.1168 -0.222033 -6.00749 0.6661L-2.7164 2.05182C-0.855238 2.83547 1.2433 2.83547 3.10446 2.05182L6.39555 0.666099C8.50486 -0.222033 10.8832 -0.222033 12.9925 0.6661L16.2836 2.05182C18.1448 2.83547 20.2433 2.83547 22.1045 2.05182L25.3955 0.666099L25.7836 1.58773Z" fill="#DBEBF9"/>
|
|
||||||
</g>
|
|
||||||
<defs>
|
|
||||||
<linearGradient id="paint0_linear_1177_19315" x1="9.25" y1="0.650024" x2="9.25" y2="3.9698" gradientUnits="userSpaceOnUse">
|
|
||||||
<stop stop-color="#EBF6FF"/>
|
|
||||||
<stop offset="1" stop-color="#F7FCFF" stop-opacity="0"/>
|
|
||||||
</linearGradient>
|
|
||||||
<clipPath id="clip0_1177_19315">
|
|
||||||
<rect width="19" height="4" fill="white"/>
|
|
||||||
</clipPath>
|
|
||||||
</defs>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 1.5 KiB |
@ -1,10 +0,0 @@
|
|||||||
<svg width="19" height="4" viewBox="0 0 19 4" fill="none" xmlns="http://www.w3.org/2000/svg">
|
|
||||||
<g clip-path="url(#clip0_1175_19292)">
|
|
||||||
<path d="M-9.5 0L-3.10445 2.69286C-1.11922 3.52875 1.11922 3.52875 3.10446 2.69286L6.39555 1.30714C8.38078 0.47125 10.6192 0.47125 12.6045 1.30714L15.8955 2.69286C17.8808 3.52875 20.1192 3.52875 22.1045 2.69286L28.5 0" stroke="#082F56" stroke-opacity="0.1"/>
|
|
||||||
</g>
|
|
||||||
<defs>
|
|
||||||
<clipPath id="clip0_1175_19292">
|
|
||||||
<rect width="19" height="4" fill="white"/>
|
|
||||||
</clipPath>
|
|
||||||
</defs>
|
|
||||||
</svg>
|
|
Before Width: | Height: | Size: 507 B |
@ -1,9 +0,0 @@
|
|||||||
#!/bin/bash
|
|
||||||
|
|
||||||
(cd /Users/rabbit-m2/p/snowball/js-sdk && NO_CLEAN=1 turbo build)
|
|
||||||
|
|
||||||
(cd ../.. && ./scripts/yarn-file-for-local-dev.sh)
|
|
||||||
|
|
||||||
rm -rf node_modules/.vite
|
|
||||||
|
|
||||||
yarn dev
|
|
@ -1,3 +1,4 @@
|
|||||||
|
import React from 'react';
|
||||||
import { render, screen } from '@testing-library/react';
|
import { render, screen } from '@testing-library/react';
|
||||||
import App from './App';
|
import App from './App';
|
||||||
|
|
||||||
|
@ -1,6 +1,7 @@
|
|||||||
import { useEffect } from 'react';
|
import React from 'react';
|
||||||
import { createBrowserRouter, RouterProvider } from 'react-router-dom';
|
import { createBrowserRouter, RouterProvider } from 'react-router-dom';
|
||||||
|
|
||||||
|
import OrgSlug from './pages/OrgSlug';
|
||||||
import Projects from './pages/org-slug';
|
import Projects from './pages/org-slug';
|
||||||
import Settings from './pages/org-slug/Settings';
|
import Settings from './pages/org-slug/Settings';
|
||||||
import {
|
import {
|
||||||
@ -9,15 +10,12 @@ import {
|
|||||||
} from './pages/org-slug/projects/routes';
|
} from './pages/org-slug/projects/routes';
|
||||||
import ProjectSearchLayout from './layouts/ProjectSearch';
|
import ProjectSearchLayout from './layouts/ProjectSearch';
|
||||||
import Index from './pages';
|
import Index from './pages';
|
||||||
import AuthPage from './pages/AuthPage';
|
import Login from './pages/Login';
|
||||||
import { DashboardLayout } from './pages/org-slug/layout';
|
|
||||||
import Web3Provider from 'context/Web3Provider';
|
|
||||||
import { BASE_URL } from 'utils/constants';
|
|
||||||
|
|
||||||
const router = createBrowserRouter([
|
const router = createBrowserRouter([
|
||||||
{
|
{
|
||||||
path: ':orgSlug',
|
path: ':orgSlug',
|
||||||
element: <DashboardLayout />,
|
element: <OrgSlug />,
|
||||||
children: [
|
children: [
|
||||||
{
|
{
|
||||||
element: <ProjectSearchLayout />,
|
element: <ProjectSearchLayout />,
|
||||||
@ -48,37 +46,12 @@ const router = createBrowserRouter([
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
path: '/login',
|
path: '/login',
|
||||||
element: <AuthPage />,
|
element: <Login />,
|
||||||
},
|
},
|
||||||
]);
|
]);
|
||||||
|
|
||||||
function App() {
|
function App() {
|
||||||
// Hacky way of checking session
|
return <RouterProvider router={router} />;
|
||||||
// TODO: Handle redirect backs
|
|
||||||
useEffect(() => {
|
|
||||||
fetch(`${BASE_URL}/auth/session`, {
|
|
||||||
credentials: 'include',
|
|
||||||
}).then((res) => {
|
|
||||||
const path = window.location.pathname;
|
|
||||||
if (res.status !== 200) {
|
|
||||||
localStorage.clear();
|
|
||||||
|
|
||||||
if (path !== '/login') {
|
|
||||||
window.location.pathname = '/login';
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (path === '/login') {
|
|
||||||
window.location.pathname = '/';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}, []);
|
|
||||||
|
|
||||||
return (
|
|
||||||
<Web3Provider>
|
|
||||||
<RouterProvider router={router} />
|
|
||||||
</Web3Provider>
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
export default App;
|
export default App;
|
||||||
|
@ -1,42 +1,32 @@
|
|||||||
import {
|
|
||||||
VITE_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO,
|
|
||||||
VITE_GITHUB_PWA_TEMPLATE_REPO,
|
|
||||||
} from 'utils/constants';
|
|
||||||
|
|
||||||
export default [
|
export default [
|
||||||
{
|
{
|
||||||
id: '1',
|
id: '1',
|
||||||
name: 'Progressive Web App (PWA)',
|
name: 'Progressive Web App (PWA)',
|
||||||
icon: 'pwa',
|
icon: '^',
|
||||||
repoFullName: `${VITE_GITHUB_PWA_TEMPLATE_REPO}`,
|
repoFullName: `${process.env.REACT_APP_GITHUB_PWA_TEMPLATE_REPO}`,
|
||||||
isComingSoon: false,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: '2',
|
id: '2',
|
||||||
name: 'Image Upload PWA',
|
name: 'Image Upload PWA',
|
||||||
icon: 'pwa',
|
icon: '^',
|
||||||
repoFullName: `${VITE_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO}`,
|
repoFullName: `${process.env.REACT_APP_GITHUB_IMAGE_UPLOAD_PWA_TEMPLATE_REPO}`,
|
||||||
isComingSoon: false,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: '3',
|
id: '3',
|
||||||
name: 'Kotlin',
|
name: 'Kotlin',
|
||||||
icon: 'kotlin',
|
icon: '^',
|
||||||
repoFullName: '',
|
repoFullName: '',
|
||||||
isComingSoon: true,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: '4',
|
id: '4',
|
||||||
name: 'React Native',
|
name: 'React Native',
|
||||||
icon: 'react-native',
|
icon: '^',
|
||||||
repoFullName: '',
|
repoFullName: '',
|
||||||
isComingSoon: true,
|
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: '5',
|
id: '5',
|
||||||
name: 'Swift',
|
name: 'Swift',
|
||||||
icon: 'swift',
|
icon: '^',
|
||||||
repoFullName: '',
|
repoFullName: '',
|
||||||
isComingSoon: true,
|
|
||||||
},
|
},
|
||||||
];
|
];
|
||||||
|
@ -1,219 +0,0 @@
|
|||||||
import React from 'react';
|
|
||||||
|
|
||||||
type Props = React.PropsWithChildren<{
|
|
||||||
className?: string;
|
|
||||||
snowZIndex?: number;
|
|
||||||
}>;
|
|
||||||
|
|
||||||
export const CloudyFlow = ({ className, children, snowZIndex }: Props) => {
|
|
||||||
return (
|
|
||||||
<div className={`bg-sky-100 relative ${className || ''}`}>
|
|
||||||
{children}
|
|
||||||
<div
|
|
||||||
className="absolute inset-0 overflow-hidden"
|
|
||||||
style={{ zIndex: snowZIndex || 0 }}
|
|
||||||
>
|
|
||||||
<div className="w-[3.72px] h-[3.72px] left-[587px] top-[147px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.72px] h-[4.72px] left-[742px] top-[336px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.49px] h-[3.49px] left-[36px] top-[68px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.25px] h-[3.25px] left-[55px] top-[114px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.60px] h-[5.60px] left-[1334px] top-[63px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.53px] h-[3.53px] left-[988px] top-[108px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.65px] h-[2.65px] left-[1380px] top-[16px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.60px] h-[3.60px] left-[1284px] top-[95px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-0.5 h-0.5 left-[1191px] top-[376px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[2.83px] h-[2.83px] left-[1182px] top-[257px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.41px] h-[2.41px] left-[627px] top-[26px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.71px] h-[5.71px] left-[30px] top-[33px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.09px] h-[4.09px] left-[425px] top-[386px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.38px] h-[3.38px] left-[394px] top-[29px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.70px] h-[4.70px] left-[817px] top-[113px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-1.5 h-1.5 left-[1194px] top-[332px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.89px] h-[4.89px] left-[811px] top-[76px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.25px] h-[4.25px] left-[458px] top-[366px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.82px] h-[4.82px] left-[936px] top-[46px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.74px] h-[3.74px] left-[64px] top-[132px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-1 h-1 left-[763px] top-[10px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.67px] h-[3.67px] left-[861px] top-[106px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.62px] h-[3.62px] left-[710px] top-[278px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.45px] h-[3.45px] left-[1069px] top-[329px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.92px] h-[2.92px] left-[1286px] top-[299px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.84px] h-[4.84px] left-[219px] top-[269px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.39px] h-[2.39px] left-[817px] top-[121px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.83px] h-[5.83px] left-[168px] top-[320px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.94px] h-[5.94px] left-[419px] top-[244px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.67px] h-[4.67px] left-[604px] top-[309px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.87px] h-[5.87px] left-[1098px] top-[379px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.85px] h-[5.85px] left-[644px] top-[352px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.19px] h-[4.19px] left-[1361px] top-[349px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[2.84px] h-[2.84px] left-[1299px] top-[194px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.51px] h-[4.51px] left-[468px] top-[319px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.73px] h-[2.73px] left-[1084px] top-[86px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.43px] h-[3.43px] left-[1271px] top-[28px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.25px] h-[2.25px] left-[106px] top-[197px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.82px] h-[2.82px] left-[122px] top-[173px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.89px] h-[2.89px] left-[343px] top-[345px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.82px] h-[2.82px] left-[433px] top-[40px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.11px] h-[4.11px] left-[904px] top-[350px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.42px] h-[4.42px] left-[1066px] top-[349px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.67px] h-[4.67px] left-[904px] top-[317px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.54px] h-[5.54px] left-[501px] top-[336px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.11px] h-[4.11px] left-[1149px] top-[206px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.55px] h-[3.55px] left-[235px] top-[362px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[2.60px] h-[2.60px] left-[1246px] top-[1px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.94px] h-[2.94px] left-[788px] top-[6px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.19px] h-[4.19px] left-[527px] top-[365px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.13px] h-[4.13px] left-[201px] top-[53px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.94px] h-[2.94px] left-[765px] top-[13px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.11px] h-[4.11px] left-[1254px] top-[30px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.85px] h-[3.85px] left-[107px] top-[316px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.72px] h-[5.72px] left-[1305px] top-[8px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.46px] h-[5.46px] left-[102px] top-[316px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.77px] h-[3.77px] left-[1322px] top-[334px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.84px] h-[4.84px] left-[1370px] top-[317px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.55px] h-[5.55px] left-[945px] top-[258px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.24px] h-[2.24px] left-[266px] top-[362px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.89px] h-[2.89px] left-[987px] top-[156px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.46px] h-[3.46px] left-[10px] top-[168px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.67px] h-[5.67px] left-[441px] top-[291px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.07px] h-[4.07px] left-[962px] top-[364px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.57px] h-[5.57px] left-[599px] top-[293px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.41px] h-[4.41px] left-[358px] top-[163px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.31px] h-[2.31px] left-[670px] top-[182px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.60px] h-[2.60px] left-[621px] top-[257px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[2.16px] h-[2.16px] left-[48px] top-[322px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.91px] h-[5.91px] left-[491px] top-[5px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.50px] h-[5.50px] left-[1139px] top-[274px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.74px] h-[3.74px] left-[24px] top-[177px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.57px] h-[5.57px] left-[1166px] top-[316px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5px] h-[5px] left-[445px] top-[326px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.01px] h-[3.01px] left-[438px] top-[252px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.14px] h-[4.14px] left-[554px] top-[131px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.30px] h-[5.30px] left-[1010px] top-[116px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.53px] h-[5.53px] left-[437px] top-[367px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.87px] h-[5.87px] left-[948px] top-[27px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[2.87px] h-[2.87px] left-[826px] top-[20px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.89px] h-[3.89px] left-[1222px] top-[112px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.77px] h-[3.77px] left-[796px] top-[395px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.09px] h-[2.09px] left-[272px] top-[103px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.12px] h-[4.12px] left-[76px] top-[2px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.51px] h-[3.51px] left-[226px] top-[276px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.03px] h-[3.03px] left-[723px] top-[197px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.14px] h-[2.14px] left-[1259px] top-[17px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.28px] h-[3.28px] left-[1244px] top-[293px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.45px] h-[4.45px] left-[118px] top-[128px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.15px] h-[4.15px] left-[490px] top-[204px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[4.93px] h-[4.93px] left-[552px] top-[38px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.56px] h-[5.56px] left-[115px] top-[303px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.35px] h-[2.35px] left-[509px] top-[278px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.24px] h-[5.24px] left-[804px] top-[389px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.44px] h-[2.44px] left-[1013px] top-[50px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.69px] h-[3.69px] left-[1183px] top-[95px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.83px] h-[2.83px] left-[278px] top-[181px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.22px] h-[3.22px] left-[1316px] top-[282px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.55px] h-[3.55px] left-[736px] top-[119px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.29px] h-[2.29px] left-[483px] top-[319px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.14px] h-[2.14px] left-[1135px] top-[19px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.64px] h-[3.64px] left-[39px] top-[126px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.30px] h-[5.30px] left-[237px] top-[369px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.57px] h-[5.57px] left-[1156px] top-[126px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.78px] h-[2.78px] left-[1295px] top-[74px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-0.5 h-0.5 left-[76px] top-[227px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.61px] h-[3.61px] left-[108px] top-[89px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.37px] h-[5.37px] left-[191px] top-[167px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.18px] h-[4.18px] left-[164px] top-[117px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.15px] h-[5.15px] left-[533px] top-[261px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-1.5 h-1.5 left-[327px] top-[157px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.74px] h-[5.74px] left-[1242px] top-[122px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.22px] h-[4.22px] left-[129px] top-[265px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.30px] h-[2.30px] left-[1305px] top-[86px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[2.70px] h-[2.70px] left-[1235px] top-[120px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[2.15px] h-[2.15px] left-[596px] top-[103px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.17px] h-[2.17px] left-[483px] top-[233px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.09px] h-[5.09px] left-[706px] top-[188px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.15px] h-[4.15px] left-[141px] top-[2px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.20px] h-[4.20px] left-[48px] top-[124px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.51px] h-[3.51px] left-[1095px] top-[201px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.21px] h-[3.21px] left-[730px] top-[185px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.61px] h-[2.61px] left-[722px] top-[319px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.28px] h-[2.28px] left-[444px] top-[26px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.49px] h-[4.49px] left-[355px] top-[212px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.69px] h-[3.69px] left-[1280px] top-[312px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.23px] h-[4.23px] left-[1114px] top-[113px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.48px] h-[3.48px] left-[729px] top-[117px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.11px] h-[4.11px] left-[647px] top-[276px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.16px] h-[4.16px] left-[365px] top-[116px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.35px] h-[5.35px] left-[94px] top-[194px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.84px] h-[5.84px] left-[2px] top-[84px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.43px] h-[4.43px] left-[1382px] top-[23px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.38px] h-[5.38px] left-[857px] top-[284px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[2.77px] h-[2.77px] left-[1228px] top-[385px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.65px] h-[4.65px] left-[165px] top-[184px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.53px] h-[5.53px] left-[568px] top-[354px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.59px] h-[3.59px] left-[1303px] top-[371px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.84px] h-[5.84px] left-[235px] top-[188px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.84px] h-[3.84px] left-[902px] top-[211px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.45px] h-[3.45px] left-[367px] top-[161px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.08px] h-[4.08px] left-[855px] top-[394px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.25px] h-[3.25px] left-[383px] top-[47px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.39px] h-[4.39px] left-[1313px] top-[165px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.60px] h-[5.60px] left-[697px] top-[327px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.09px] h-[2.09px] left-[646px] top-[370px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.13px] h-[3.13px] left-[728px] top-[122px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.53px] h-[5.53px] left-[203px] top-[293px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.83px] h-[5.83px] left-[424px] top-[121px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.82px] h-[4.82px] left-[1358px] top-[176px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.18px] h-[3.18px] left-[1212px] top-[24px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.23px] h-[5.23px] left-[260px] top-[217px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.29px] h-[5.29px] left-[1204px] top-[367px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.47px] h-[3.47px] left-[1163px] top-[159px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.77px] h-[5.77px] left-[1257px] top-[115px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.31px] h-[5.31px] left-[222px] top-[356px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.43px] h-[5.43px] left-[1141px] top-[349px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[5.62px] h-[5.62px] left-[683px] top-[81px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.91px] h-[3.91px] left-[269px] top-[3px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.51px] h-[3.51px] left-[305px] top-[310px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.41px] h-[5.41px] left-[530px] top-[94px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.64px] h-[4.64px] left-[730px] top-[301px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.59px] h-[3.59px] left-[716px] top-[14px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.77px] h-[4.77px] left-[544px] top-[13px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[2.29px] h-[2.29px] left-[357px] top-[281px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.42px] h-[2.42px] left-[1346px] top-[112px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.42px] h-[3.42px] left-[671px] top-[150px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.40px] h-[4.40px] left-[1324px] top-[268px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.21px] h-[5.21px] left-[1028px] top-[376px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.27px] h-[4.27px] left-[499px] top-[50px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[4.35px] h-[4.35px] left-[543px] top-[359px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.25px] h-[5.25px] left-[1245px] top-[296px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.52px] h-[5.52px] left-[360px] top-[98px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.46px] h-[4.46px] left-[741px] top-[358px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[3.90px] h-[3.90px] left-[1262px] top-[184px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[5.75px] h-[5.75px] left-[552px] top-[335px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[4.95px] h-[4.95px] left-[120px] top-[178px] absolute bg-white rounded-full" />
|
|
||||||
<div className="w-[3.28px] h-[3.28px] left-[1337px] top-[293px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[2.43px] h-[2.43px] left-[233px] top-[310px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-1 h-1 left-[218px] top-[322px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.68px] h-[3.68px] left-[984px] top-[8px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[2.44px] h-[2.44px] left-[832px] top-[55px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.93px] h-[3.93px] left-[1105px] top-[209px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.08px] h-[4.08px] left-[957px] top-[23px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[2.33px] h-[2.33px] left-[1066px] top-[390px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
<div className="w-[3.25px] h-[3.25px] left-[737px] top-[118px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.18px] h-[5.18px] left-[202px] top-[19px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.05px] h-[5.05px] left-[466px] top-[17px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.85px] h-[3.85px] left-[144px] top-[153px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.35px] h-[5.35px] left-[233px] top-[330px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-1 h-1 left-[730px] top-[179px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[4.46px] h-[4.46px] left-[1156px] top-[342px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.22px] h-[5.22px] left-[1275px] top-[204px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.50px] h-[5.50px] left-[38px] top-[343px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.14px] h-[5.14px] left-[867px] top-[113px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[2.19px] h-[2.19px] left-[1277px] top-[314px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[3.74px] h-[3.74px] left-[1136px] top-[197px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.37px] h-[5.37px] left-[34px] top-[226px] absolute bg-white bg-opacity-60 rounded-full" />
|
|
||||||
<div className="w-[5.93px] h-[5.93px] left-[727px] top-[272px] absolute bg-white bg-opacity-50 rounded-full" />
|
|
||||||
<div className="w-[5.29px] h-[5.29px] left-[277px] top-[43px] absolute bg-white bg-opacity-80 rounded-full" />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,4 +1,4 @@
|
|||||||
import { useCallback, useMemo, useState } from 'react';
|
import React, { useCallback, useMemo, useState } from 'react';
|
||||||
import { format } from 'date-fns';
|
import { format } from 'date-fns';
|
||||||
import {
|
import {
|
||||||
DayPicker,
|
DayPicker,
|
||||||
@ -12,7 +12,7 @@ import {
|
|||||||
Popover,
|
Popover,
|
||||||
PopoverContent,
|
PopoverContent,
|
||||||
PopoverHandler,
|
PopoverHandler,
|
||||||
} from '@snowballtools/material-tailwind-react-fork';
|
} from '@material-tailwind/react';
|
||||||
|
|
||||||
import HorizontalLine from './HorizontalLine';
|
import HorizontalLine from './HorizontalLine';
|
||||||
|
|
||||||
@ -120,11 +120,14 @@ const DatePicker = ({
|
|||||||
handler={(value) => setIsOpen(value)}
|
handler={(value) => setIsOpen(value)}
|
||||||
>
|
>
|
||||||
<PopoverHandler>
|
<PopoverHandler>
|
||||||
<Input onChange={() => null} value={inputValue} />
|
<Input
|
||||||
|
onChange={() => null}
|
||||||
|
value={inputValue}
|
||||||
|
crossOrigin={undefined}
|
||||||
|
/>
|
||||||
</PopoverHandler>
|
</PopoverHandler>
|
||||||
{/* TODO: Figure out what placeholder is for */}
|
{/* TODO: Figure out what placeholder is for */}
|
||||||
{/* @ts-ignore */}
|
<PopoverContent placeholder={''}>
|
||||||
<PopoverContent>
|
|
||||||
{mode === 'single' && (
|
{mode === 'single' && (
|
||||||
<DayPicker
|
<DayPicker
|
||||||
mode="single"
|
mode="single"
|
||||||
@ -149,6 +152,7 @@ const DatePicker = ({
|
|||||||
className="rounded-full mr-2"
|
className="rounded-full mr-2"
|
||||||
variant="outlined"
|
variant="outlined"
|
||||||
onClick={() => setIsOpen(false)}
|
onClick={() => setIsOpen(false)}
|
||||||
|
placeholder={''}
|
||||||
>
|
>
|
||||||
Cancel
|
Cancel
|
||||||
</Button>
|
</Button>
|
||||||
@ -158,6 +162,7 @@ const DatePicker = ({
|
|||||||
className="rounded-full"
|
className="rounded-full"
|
||||||
color="gray"
|
color="gray"
|
||||||
onClick={() => handleRangeSelect()}
|
onClick={() => handleRangeSelect()}
|
||||||
|
placeholder={''}
|
||||||
>
|
>
|
||||||
Select
|
Select
|
||||||
</Button>
|
</Button>
|
||||||
|
@ -1,3 +1,4 @@
|
|||||||
|
import React from 'react';
|
||||||
import {
|
import {
|
||||||
default as ReactDropdown,
|
default as ReactDropdown,
|
||||||
Option as ReactDropdownOption,
|
Option as ReactDropdownOption,
|
||||||
|
@ -1,22 +1,13 @@
|
|||||||
import { Duration } from 'luxon';
|
import { Duration } from 'luxon';
|
||||||
import { ComponentPropsWithoutRef } from 'react';
|
import React from 'react';
|
||||||
import { cn } from 'utils/classnames';
|
|
||||||
|
|
||||||
export interface FormatMilliSecondProps
|
const FormatMillisecond = ({ time }: { time: number }) => {
|
||||||
extends ComponentPropsWithoutRef<'div'> {
|
|
||||||
time: number;
|
|
||||||
}
|
|
||||||
|
|
||||||
const FormatMillisecond = ({ time, ...props }: FormatMilliSecondProps) => {
|
|
||||||
const formatTime = Duration.fromMillis(time)
|
const formatTime = Duration.fromMillis(time)
|
||||||
.shiftTo('days', 'hours', 'minutes', 'seconds')
|
.shiftTo('days', 'hours', 'minutes', 'seconds')
|
||||||
.toObject();
|
.toObject();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div>
|
||||||
{...props}
|
|
||||||
className={cn('text-sm text-elements-mid-em', props?.className)}
|
|
||||||
>
|
|
||||||
{formatTime.days !== 0 && <span>{formatTime.days}d </span>}
|
{formatTime.days !== 0 && <span>{formatTime.days}d </span>}
|
||||||
{formatTime.hours !== 0 && <span>{formatTime.hours}h </span>}
|
{formatTime.hours !== 0 && <span>{formatTime.hours}h </span>}
|
||||||
{formatTime.minutes !== 0 && <span>{formatTime.minutes}m </span>}
|
{formatTime.minutes !== 0 && <span>{formatTime.minutes}m </span>}
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
import React from 'react';
|
||||||
|
|
||||||
const HorizontalLine = () => {
|
const HorizontalLine = () => {
|
||||||
return <hr className="h-px bg-gray-100 border-0" />;
|
return <hr className="h-px bg-gray-100 border-0" />;
|
||||||
};
|
};
|
||||||
|
@ -1,23 +0,0 @@
|
|||||||
import { Link } from 'react-router-dom';
|
|
||||||
import { Heading } from './shared/Heading';
|
|
||||||
|
|
||||||
interface LogoProps {
|
|
||||||
orgSlug?: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const Logo = ({ orgSlug }: LogoProps) => {
|
|
||||||
return (
|
|
||||||
<Link to={`/${orgSlug}`}>
|
|
||||||
<div className="flex items-center gap-3 px-0 lg:px-2">
|
|
||||||
<img
|
|
||||||
src="/logo.svg"
|
|
||||||
alt="Snowball Logo"
|
|
||||||
className="lg:h-10 lg:w-10 h-8 w-8 rounded-lg"
|
|
||||||
/>
|
|
||||||
<Heading className="lg:text-[24px] text-[19px] font-semibold">
|
|
||||||
Snowball
|
|
||||||
</Heading>
|
|
||||||
</div>
|
|
||||||
</Link>
|
|
||||||
);
|
|
||||||
};
|
|
@ -6,7 +6,7 @@ import { Input, InputProps } from './shared/Input';
|
|||||||
const SearchBar: React.ForwardRefRenderFunction<
|
const SearchBar: React.ForwardRefRenderFunction<
|
||||||
HTMLInputElement,
|
HTMLInputElement,
|
||||||
InputProps & RefAttributes<HTMLInputElement>
|
InputProps & RefAttributes<HTMLInputElement>
|
||||||
> = ({ value, onChange, placeholder = 'Search', ...props }, ref) => {
|
> = ({ value, onChange, placeholder = 'Search', ...props }) => {
|
||||||
return (
|
return (
|
||||||
<div className="relative flex w-full">
|
<div className="relative flex w-full">
|
||||||
<Input
|
<Input
|
||||||
@ -15,10 +15,8 @@ const SearchBar: React.ForwardRefRenderFunction<
|
|||||||
value={value}
|
value={value}
|
||||||
type="search"
|
type="search"
|
||||||
placeholder={placeholder}
|
placeholder={placeholder}
|
||||||
appearance="borderless"
|
appearance={'borderless'}
|
||||||
className="w-full lg:w-[459px]"
|
|
||||||
{...props}
|
{...props}
|
||||||
ref={ref}
|
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
|
@ -1,4 +1,5 @@
|
|||||||
import { StepperNav } from './VerticalStepper';
|
import React from 'react';
|
||||||
|
import { StepperNav } from 'vertical-stepper-nav';
|
||||||
|
|
||||||
const COLOR_COMPLETED = '#059669';
|
const COLOR_COMPLETED = '#059669';
|
||||||
const COLOR_ACTIVE = '#CFE6FC';
|
const COLOR_ACTIVE = '#CFE6FC';
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import { useEffect } from 'react';
|
import React from 'react';
|
||||||
import { useStopwatch } from 'react-timer-hook';
|
import { useStopwatch } from 'react-timer-hook';
|
||||||
|
|
||||||
import FormatMillisecond, { FormatMilliSecondProps } from './FormatMilliSecond';
|
import FormatMillisecond from './FormatMilliSecond';
|
||||||
|
|
||||||
const setStopWatchOffset = (time: string) => {
|
const setStopWatchOffset = (time: string) => {
|
||||||
const providedTime = new Date(time);
|
const providedTime = new Date(time);
|
||||||
@ -11,22 +11,13 @@ const setStopWatchOffset = (time: string) => {
|
|||||||
return currentTime;
|
return currentTime;
|
||||||
};
|
};
|
||||||
|
|
||||||
interface StopwatchProps extends Omit<FormatMilliSecondProps, 'time'> {
|
const Stopwatch = ({ offsetTimestamp }: { offsetTimestamp: Date }) => {
|
||||||
offsetTimestamp: Date;
|
const { totalSeconds } = useStopwatch({
|
||||||
isPaused: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const Stopwatch = ({ offsetTimestamp, isPaused, ...props }: StopwatchProps) => {
|
|
||||||
const { totalSeconds, pause, start } = useStopwatch({
|
|
||||||
autoStart: true,
|
autoStart: true,
|
||||||
offsetTimestamp: offsetTimestamp,
|
offsetTimestamp: offsetTimestamp,
|
||||||
});
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
return <FormatMillisecond time={totalSeconds * 1000} />;
|
||||||
isPaused ? pause() : start();
|
|
||||||
}, [isPaused]);
|
|
||||||
|
|
||||||
return <FormatMillisecond time={totalSeconds * 1000} {...props} />;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export { Stopwatch, setStopWatchOffset };
|
export { Stopwatch, setStopWatchOffset };
|
||||||
|
@ -1,120 +0,0 @@
|
|||||||
import * as CSS from 'csstype';
|
|
||||||
|
|
||||||
//
|
|
||||||
// Nav
|
|
||||||
//
|
|
||||||
export interface IStepDescription {
|
|
||||||
stepContent: () => JSX.Element;
|
|
||||||
stepStateColor?: string;
|
|
||||||
stepStatusCircleSize?: number;
|
|
||||||
onClickHandler?: () => void | undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
export interface IStepperNavProps {
|
|
||||||
steps: IStepDescription[];
|
|
||||||
}
|
|
||||||
|
|
||||||
export const StepperNav = (props: IStepperNavProps): JSX.Element => {
|
|
||||||
return (
|
|
||||||
<nav>
|
|
||||||
{props.steps.map(
|
|
||||||
(
|
|
||||||
{ stepContent, stepStateColor, onClickHandler, stepStatusCircleSize },
|
|
||||||
index,
|
|
||||||
) => (
|
|
||||||
<div key={index}>
|
|
||||||
<Step
|
|
||||||
stepContent={stepContent}
|
|
||||||
statusColor={stepStateColor}
|
|
||||||
onClickHandler={onClickHandler}
|
|
||||||
statusCircleSize={stepStatusCircleSize}
|
|
||||||
/>
|
|
||||||
{index !== props.steps.length - 1 && (
|
|
||||||
<div
|
|
||||||
style={{
|
|
||||||
paddingLeft: `${(stepStatusCircleSize ?? 16) / 2 + 1}px`,
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<Separator />
|
|
||||||
</div>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
),
|
|
||||||
)}
|
|
||||||
</nav>
|
|
||||||
);
|
|
||||||
};
|
|
||||||
|
|
||||||
//
|
|
||||||
// Separator
|
|
||||||
//
|
|
||||||
const separatorStyles = {
|
|
||||||
height: '5vh',
|
|
||||||
width: 2,
|
|
||||||
border: '1px solid #E1E1E1',
|
|
||||||
background: '#E1E1E1',
|
|
||||||
};
|
|
||||||
|
|
||||||
export interface ISeparator {
|
|
||||||
height?: string | number;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const Separator = ({ height }: ISeparator): JSX.Element => {
|
|
||||||
return <div style={{ ...separatorStyles, height: height ?? '5vh' }} />;
|
|
||||||
};
|
|
||||||
|
|
||||||
//
|
|
||||||
// Step
|
|
||||||
//
|
|
||||||
export interface IStep {
|
|
||||||
stepContent: () => JSX.Element;
|
|
||||||
statusColor?: string;
|
|
||||||
statusCircleSize?: number;
|
|
||||||
onClickHandler?: (
|
|
||||||
event?: React.MouseEvent<HTMLDivElement>,
|
|
||||||
) => void | undefined;
|
|
||||||
}
|
|
||||||
|
|
||||||
const buttonContainerStyles: CSS.Properties = {
|
|
||||||
display: 'inline-flex',
|
|
||||||
flexWrap: 'wrap',
|
|
||||||
gap: '12px',
|
|
||||||
padding: '2px',
|
|
||||||
cursor: 'pointer',
|
|
||||||
};
|
|
||||||
|
|
||||||
export const Step = ({
|
|
||||||
stepContent,
|
|
||||||
statusColor,
|
|
||||||
statusCircleSize,
|
|
||||||
onClickHandler,
|
|
||||||
}: IStep): JSX.Element => {
|
|
||||||
const circleStyles = {
|
|
||||||
borderRadius: statusCircleSize ?? 16,
|
|
||||||
width: statusCircleSize ?? 16,
|
|
||||||
height: statusCircleSize ?? 16,
|
|
||||||
border: '2px solid #E1E1E1',
|
|
||||||
background: statusColor ?? 'white',
|
|
||||||
};
|
|
||||||
|
|
||||||
const keyDownHandler = (event: React.KeyboardEvent<HTMLDivElement>) => {
|
|
||||||
if (event.keyCode === 13 || event.keyCode === 32) {
|
|
||||||
onClickHandler?.();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<div
|
|
||||||
tabIndex={0}
|
|
||||||
onClick={onClickHandler}
|
|
||||||
onKeyDown={keyDownHandler}
|
|
||||||
role="button"
|
|
||||||
style={{ ...buttonContainerStyles }}
|
|
||||||
>
|
|
||||||
<div>
|
|
||||||
<div style={circleStyles} />
|
|
||||||
</div>
|
|
||||||
<div style={{ paddingBottom: 2 }}>{stepContent()}</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,29 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
interface CancelDeploymentDialogProps extends ConfirmDialogProps {}
|
|
||||||
|
|
||||||
export const CancelDeploymentDialog = ({
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: CancelDeploymentDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Cancel deployment?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, cancel deployment"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{ variant: 'danger' }}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-high-em tracking-[-0.006em]">
|
|
||||||
This will halt the deployment and you'll have to start the process
|
|
||||||
from scratch.
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,104 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
import { Deployment, Domain } from 'gql-client';
|
|
||||||
|
|
||||||
import DeploymentDialogBodyCard from 'components/projects/project/deployments/DeploymentDialogBodyCard';
|
|
||||||
import { Button } from 'components/shared/Button';
|
|
||||||
import {
|
|
||||||
ChevronDoubleDownIcon,
|
|
||||||
LinkChainIcon,
|
|
||||||
} from 'components/shared/CustomIcon';
|
|
||||||
import { TagProps } from 'components/shared/Tag';
|
|
||||||
import {
|
|
||||||
ArrowRightCircleFilledIcon,
|
|
||||||
LoadingIcon,
|
|
||||||
} from 'components/shared/CustomIcon';
|
|
||||||
|
|
||||||
interface ChangeStateToProductionDialogProps extends ConfirmDialogProps {
|
|
||||||
deployment: Deployment;
|
|
||||||
newDeployment?: Deployment;
|
|
||||||
domains: Domain[];
|
|
||||||
isConfirmButtonLoading?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const ChangeStateToProductionDialog = ({
|
|
||||||
deployment,
|
|
||||||
newDeployment,
|
|
||||||
domains,
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
isConfirmButtonLoading,
|
|
||||||
...props
|
|
||||||
}: ChangeStateToProductionDialogProps) => {
|
|
||||||
const currentChip = {
|
|
||||||
value: 'Live Deployment',
|
|
||||||
type: 'positive' as TagProps['type'],
|
|
||||||
};
|
|
||||||
const newChip = {
|
|
||||||
value: 'New Deployment',
|
|
||||||
type: 'attention' as TagProps['type'],
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{
|
|
||||||
disabled: isConfirmButtonLoading,
|
|
||||||
rightIcon: isConfirmButtonLoading ? (
|
|
||||||
<LoadingIcon className="animate-spin" />
|
|
||||||
) : (
|
|
||||||
<ArrowRightCircleFilledIcon />
|
|
||||||
),
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<div className="flex flex-col gap-7">
|
|
||||||
<div className="flex flex-col gap-3">
|
|
||||||
<p className="text-sm text-elements-high-em tracking-[-0.006em]">
|
|
||||||
Upon confirmation, this deployment will be changed to production.
|
|
||||||
</p>
|
|
||||||
<DeploymentDialogBodyCard
|
|
||||||
deployment={deployment}
|
|
||||||
chip={newDeployment ? currentChip : undefined}
|
|
||||||
/>
|
|
||||||
{newDeployment && (
|
|
||||||
<>
|
|
||||||
<div className="flex items-center justify-between w-full text-elements-info">
|
|
||||||
{Array.from({ length: 7 }).map((_, index) => (
|
|
||||||
<ChevronDoubleDownIcon key={index} />
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
<DeploymentDialogBodyCard
|
|
||||||
deployment={newDeployment}
|
|
||||||
chip={newChip}
|
|
||||||
/>
|
|
||||||
</>
|
|
||||||
)}
|
|
||||||
</div>
|
|
||||||
<div className="flex flex-col items-start gap-3">
|
|
||||||
<p className="text-sm text-elements-high-em tracking-[-0.006em]">
|
|
||||||
The new deployment will be associated with these domains:
|
|
||||||
</p>
|
|
||||||
{domains.length > 0 &&
|
|
||||||
domains.map((value) => {
|
|
||||||
return (
|
|
||||||
<Button
|
|
||||||
as="a"
|
|
||||||
href={value.name}
|
|
||||||
leftIcon={<LinkChainIcon size={18} />}
|
|
||||||
variant="link"
|
|
||||||
key={value.id}
|
|
||||||
>
|
|
||||||
{value.name}
|
|
||||||
</Button>
|
|
||||||
);
|
|
||||||
})}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,47 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
import {
|
|
||||||
ArrowRightCircleFilledIcon,
|
|
||||||
LoadingIcon,
|
|
||||||
} from 'components/shared/CustomIcon';
|
|
||||||
|
|
||||||
interface DeleteDeploymentDialogProps extends ConfirmDialogProps {
|
|
||||||
isConfirmButtonLoading?: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const DeleteDeploymentDialog = ({
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
isConfirmButtonLoading,
|
|
||||||
...props
|
|
||||||
}: DeleteDeploymentDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Delete deployment?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle={
|
|
||||||
isConfirmButtonLoading
|
|
||||||
? 'Deleting deployment'
|
|
||||||
: 'Yes, delete deployment'
|
|
||||||
}
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{
|
|
||||||
variant: 'danger',
|
|
||||||
disabled: isConfirmButtonLoading,
|
|
||||||
rightIcon: isConfirmButtonLoading ? (
|
|
||||||
<LoadingIcon className="animate-spin" />
|
|
||||||
) : (
|
|
||||||
<ArrowRightCircleFilledIcon />
|
|
||||||
),
|
|
||||||
}}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-high-em">
|
|
||||||
Once deleted, the deployment will not be accessible.
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,41 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
interface DeleteDomainDialogProps extends ConfirmDialogProps {
|
|
||||||
projectName: string;
|
|
||||||
domainName: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const DeleteDomainDialog = ({
|
|
||||||
projectName,
|
|
||||||
domainName,
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: DeleteDomainDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Delete domain?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, delete domain"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{ variant: 'danger' }}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-high-em">
|
|
||||||
Once deleted, the project{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-on-secondary bg-controls-secondary rounded px-0.5">
|
|
||||||
{projectName}
|
|
||||||
</span>{' '}
|
|
||||||
will not be accessible from the domain{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-on-secondary bg-controls-secondary rounded px-0.5">
|
|
||||||
{domainName}
|
|
||||||
</span>
|
|
||||||
.
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,35 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
interface DeleteVariableDialogProps extends ConfirmDialogProps {
|
|
||||||
variableKey: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const DeleteVariableDialog = ({
|
|
||||||
variableKey,
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: DeleteVariableDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Delete variable"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, confirm delete"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{ variant: 'danger' }}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-mid-em">
|
|
||||||
Are you sure you want to delete the variable{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-on-secondary bg-controls-secondary rounded px-0.5">
|
|
||||||
{variableKey}
|
|
||||||
</span>
|
|
||||||
?
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,35 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
interface DeleteWebhookDialogProps extends ConfirmDialogProps {
|
|
||||||
webhookUrl: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const DeleteWebhookDialog = ({
|
|
||||||
webhookUrl,
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: DeleteWebhookDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Delete webhook?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, confirm delete"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{ variant: 'danger' }}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-mid-em">
|
|
||||||
Are you sure you want to delete{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-high-em px-0.5">
|
|
||||||
{webhookUrl}
|
|
||||||
</span>
|
|
||||||
?
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,29 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
interface DisconnectRepositoryDialogProps extends ConfirmDialogProps {}
|
|
||||||
|
|
||||||
export const DisconnectRepositoryDialog = ({
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: DisconnectRepositoryDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Disconnect repository?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, confirm disconnect"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{ variant: 'danger' }}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-high-em">
|
|
||||||
Any data tied to your Git project may become misconfigured. Are you sure
|
|
||||||
you want to continue?
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,38 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
import { formatAddress } from 'utils/format';
|
|
||||||
|
|
||||||
interface RemoveMemberDialogProps extends ConfirmDialogProps {
|
|
||||||
memberName: string;
|
|
||||||
ethAddress: string;
|
|
||||||
emailDomain: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const RemoveMemberDialog = ({
|
|
||||||
memberName,
|
|
||||||
ethAddress,
|
|
||||||
emailDomain,
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: RemoveMemberDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Remove member?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, remove member"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
confirmButtonProps={{ variant: 'danger' }}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-high-em">
|
|
||||||
Once removed, {formatAddress(memberName)} ({formatAddress(ethAddress)}@
|
|
||||||
{emailDomain}) will not be able to access this project.
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -1,46 +0,0 @@
|
|||||||
import ConfirmDialog, {
|
|
||||||
ConfirmDialogProps,
|
|
||||||
} from 'components/shared/ConfirmDialog';
|
|
||||||
|
|
||||||
interface TransferProjectDialogProps extends ConfirmDialogProps {
|
|
||||||
projectName: string;
|
|
||||||
from: string;
|
|
||||||
to: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
export const TransferProjectDialog = ({
|
|
||||||
projectName,
|
|
||||||
from,
|
|
||||||
to,
|
|
||||||
open,
|
|
||||||
handleCancel,
|
|
||||||
handleConfirm,
|
|
||||||
...props
|
|
||||||
}: TransferProjectDialogProps) => {
|
|
||||||
return (
|
|
||||||
<ConfirmDialog
|
|
||||||
{...props}
|
|
||||||
dialogTitle="Transfer project?"
|
|
||||||
handleCancel={handleCancel}
|
|
||||||
open={open}
|
|
||||||
confirmButtonTitle="Yes, confirm transfer"
|
|
||||||
handleConfirm={handleConfirm}
|
|
||||||
>
|
|
||||||
<p className="text-sm text-elements-high-em">
|
|
||||||
Upon confirmation, your project{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-on-secondary bg-controls-secondary rounded px-0.5">
|
|
||||||
{projectName}
|
|
||||||
</span>{' '}
|
|
||||||
will be transferred from{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-on-secondary bg-controls-secondary rounded px-0.5">
|
|
||||||
{from}
|
|
||||||
</span>{' '}
|
|
||||||
to{' '}
|
|
||||||
<span className="text-sm font-mono text-elements-on-secondary bg-controls-secondary rounded px-0.5">
|
|
||||||
{to}
|
|
||||||
</span>
|
|
||||||
.
|
|
||||||
</p>
|
|
||||||
</ConfirmDialog>
|
|
||||||
);
|
|
||||||
};
|
|
@ -8,30 +8,25 @@ export const projectCardTheme = tv({
|
|||||||
'rounded-2xl',
|
'rounded-2xl',
|
||||||
'flex',
|
'flex',
|
||||||
'flex-col',
|
'flex-col',
|
||||||
'group',
|
|
||||||
'cursor-pointer',
|
|
||||||
],
|
],
|
||||||
upperContent: ['px-4', 'py-4', 'flex', 'items-start', 'gap-3', 'relative'],
|
upperContent: ['px-4', 'py-4', 'flex', 'items-start', 'gap-3', 'relative'],
|
||||||
content: ['flex', 'flex-col', 'gap-1', 'flex-1', 'overflow-hidden'],
|
content: ['flex', 'flex-col', 'gap-1', 'flex-1'],
|
||||||
title: [
|
title: [
|
||||||
'text-sm',
|
'text-sm',
|
||||||
'font-medium',
|
'font-medium',
|
||||||
'text-elements-high-em',
|
'text-elements-high-em',
|
||||||
'tracking-[-0.006em]',
|
'tracking-[-0.006em]',
|
||||||
'truncate',
|
|
||||||
],
|
],
|
||||||
description: ['text-xs', 'text-elements-low-em', 'truncate'],
|
description: ['text-xs', 'text-elements-low-em'],
|
||||||
icons: ['flex', 'items-center', 'gap-1'],
|
icons: ['flex', 'items-center', 'gap-1'],
|
||||||
lowerContent: [
|
lowerContent: [
|
||||||
'transition-colors',
|
'bg-surface-card-hovered',
|
||||||
'duration-150',
|
|
||||||
'px-4',
|
'px-4',
|
||||||
'py-4',
|
'py-4',
|
||||||
'flex',
|
'flex',
|
||||||
'flex-col',
|
'flex-col',
|
||||||
'gap-2',
|
'gap-2',
|
||||||
'rounded-b-2xl',
|
'rounded-b-2xl',
|
||||||
'group-hover:bg-surface-card-hovered',
|
|
||||||
],
|
],
|
||||||
latestDeployment: ['flex', 'items-center', 'gap-2'],
|
latestDeployment: ['flex', 'items-center', 'gap-2'],
|
||||||
deploymentStatusContainer: [
|
deploymentStatusContainer: [
|
||||||
@ -51,12 +46,6 @@ export const projectCardTheme = tv({
|
|||||||
'items-center',
|
'items-center',
|
||||||
'gap-2',
|
'gap-2',
|
||||||
],
|
],
|
||||||
wavyBorder: [
|
|
||||||
'bg-surface-card',
|
|
||||||
'transition-colors',
|
|
||||||
'duration-150',
|
|
||||||
'group-hover:bg-surface-card-hovered',
|
|
||||||
],
|
|
||||||
},
|
},
|
||||||
variants: {
|
variants: {
|
||||||
status: {
|
status: {
|
||||||
|
@ -1,26 +1,25 @@
|
|||||||
|
import React, { ComponentPropsWithoutRef, MouseEvent } from 'react';
|
||||||
|
import { ProjectCardTheme, projectCardTheme } from './ProjectCard.theme';
|
||||||
|
import { Project } from 'gql-client';
|
||||||
|
import { Button } from 'components/shared/Button';
|
||||||
|
import { WavyBorder } from 'components/shared/WavyBorder';
|
||||||
|
import {
|
||||||
|
BranchIcon,
|
||||||
|
ClockIcon,
|
||||||
|
GitHubLogo,
|
||||||
|
HorizontalDotIcon,
|
||||||
|
WarningDiamondIcon,
|
||||||
|
} from 'components/shared/CustomIcon';
|
||||||
|
import { relativeTimeMs } from 'utils/time';
|
||||||
|
import { Link } from 'react-router-dom';
|
||||||
|
import { Avatar } from 'components/shared/Avatar';
|
||||||
|
import { getInitials } from 'utils/geInitials';
|
||||||
import {
|
import {
|
||||||
Menu,
|
Menu,
|
||||||
MenuHandler,
|
MenuHandler,
|
||||||
MenuItem,
|
MenuItem,
|
||||||
MenuList,
|
MenuList,
|
||||||
} from '@snowballtools/material-tailwind-react-fork';
|
} from '@material-tailwind/react';
|
||||||
import { ComponentPropsWithoutRef, MouseEvent, useCallback } from 'react';
|
|
||||||
import { useNavigate } from 'react-router-dom';
|
|
||||||
import { Project } from 'gql-client';
|
|
||||||
import { Avatar } from 'components/shared/Avatar';
|
|
||||||
import { Button } from 'components/shared/Button';
|
|
||||||
import {
|
|
||||||
BranchIcon,
|
|
||||||
ClockIcon,
|
|
||||||
GithubLogoIcon,
|
|
||||||
HorizontalDotIcon,
|
|
||||||
WarningDiamondIcon,
|
|
||||||
} from 'components/shared/CustomIcon';
|
|
||||||
import { Tooltip } from 'components/shared/Tooltip';
|
|
||||||
import { WavyBorder } from 'components/shared/WavyBorder';
|
|
||||||
import { relativeTimeMs } from 'utils/time';
|
|
||||||
import { getInitials } from 'utils/geInitials';
|
|
||||||
import { ProjectCardTheme, projectCardTheme } from './ProjectCard.theme';
|
|
||||||
|
|
||||||
export interface ProjectCardProps
|
export interface ProjectCardProps
|
||||||
extends ComponentPropsWithoutRef<'div'>,
|
extends ComponentPropsWithoutRef<'div'>,
|
||||||
@ -28,8 +27,6 @@ export interface ProjectCardProps
|
|||||||
project: Project;
|
project: Project;
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Update the whole component to use `Link` from `react-router-dom` and remove the `useNavigate` hook,
|
|
||||||
// currently it's not possible to use `Link` because the dot menu is not a direct child of the `Link` component
|
|
||||||
export const ProjectCard = ({
|
export const ProjectCard = ({
|
||||||
className,
|
className,
|
||||||
project,
|
project,
|
||||||
@ -41,34 +38,14 @@ export const ProjectCard = ({
|
|||||||
// TODO: Update this to use the actual status from the API
|
// TODO: Update this to use the actual status from the API
|
||||||
const hasError = status === 'failure';
|
const hasError = status === 'failure';
|
||||||
|
|
||||||
const navigate = useNavigate();
|
|
||||||
|
|
||||||
const handleOptionsClick = (
|
const handleOptionsClick = (
|
||||||
e: MouseEvent<HTMLButtonElement, globalThis.MouseEvent>,
|
e: MouseEvent<HTMLButtonElement, globalThis.MouseEvent>,
|
||||||
) => {
|
) => {
|
||||||
e.stopPropagation();
|
e.stopPropagation();
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleClick = useCallback(() => {
|
|
||||||
navigate(`projects/${project.id}`);
|
|
||||||
}, [project.id, navigate]);
|
|
||||||
|
|
||||||
const navigateToSettingsOnClick = useCallback(
|
|
||||||
(
|
|
||||||
e: React.MouseEvent<HTMLLIElement> | React.MouseEvent<HTMLButtonElement>,
|
|
||||||
) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
navigate(`projects/${project.id}/settings`);
|
|
||||||
},
|
|
||||||
[project.id, navigate],
|
|
||||||
);
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div {...props} className={theme.wrapper({ className })}>
|
||||||
{...props}
|
|
||||||
className={theme.wrapper({ className })}
|
|
||||||
onClick={handleClick}
|
|
||||||
>
|
|
||||||
{/* Upper content */}
|
{/* Upper content */}
|
||||||
<div className={theme.upperContent()}>
|
<div className={theme.upperContent()}>
|
||||||
{/* Icon container */}
|
{/* Icon container */}
|
||||||
@ -77,15 +54,14 @@ export const ProjectCard = ({
|
|||||||
imageSrc={project.icon}
|
imageSrc={project.icon}
|
||||||
initials={getInitials(project.name)}
|
initials={getInitials(project.name)}
|
||||||
/>
|
/>
|
||||||
|
{/* </div> */}
|
||||||
{/* Title and website */}
|
{/* Title and website */}
|
||||||
<div className={theme.content()}>
|
<Link to={`projects/${project.id}`} className={theme.content()}>
|
||||||
<Tooltip content={project.name}>
|
<p className={theme.title()}>{project.name}</p>
|
||||||
<p className={theme.title()}>{project.name}</p>
|
|
||||||
</Tooltip>
|
|
||||||
<p className={theme.description()}>
|
<p className={theme.description()}>
|
||||||
{project.deployments[0]?.domain?.name ?? 'No domain'}
|
{project.deployments[0]?.domain?.name ?? 'No domain'}
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</Link>
|
||||||
{/* Icons */}
|
{/* Icons */}
|
||||||
<div className={theme.icons()}>
|
<div className={theme.icons()}>
|
||||||
{hasError && <WarningDiamondIcon className="text-elements-danger" />}
|
{hasError && <WarningDiamondIcon className="text-elements-danger" />}
|
||||||
@ -101,14 +77,9 @@ export const ProjectCard = ({
|
|||||||
<HorizontalDotIcon />
|
<HorizontalDotIcon />
|
||||||
</Button>
|
</Button>
|
||||||
</MenuHandler>
|
</MenuHandler>
|
||||||
<MenuList>
|
<MenuList placeholder={''}>
|
||||||
<MenuItem onClick={navigateToSettingsOnClick}>
|
<MenuItem placeholder={''}>Project settings</MenuItem>
|
||||||
Project settings
|
<MenuItem className="text-red-500" placeholder={''}>
|
||||||
</MenuItem>
|
|
||||||
<MenuItem
|
|
||||||
className="text-red-500"
|
|
||||||
onClick={navigateToSettingsOnClick}
|
|
||||||
>
|
|
||||||
Delete project
|
Delete project
|
||||||
</MenuItem>
|
</MenuItem>
|
||||||
</MenuList>
|
</MenuList>
|
||||||
@ -116,7 +87,7 @@ export const ProjectCard = ({
|
|||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
{/* Wave */}
|
{/* Wave */}
|
||||||
<WavyBorder variant="stroke-and-fill" className={theme.wavyBorder()} />
|
<WavyBorder />
|
||||||
{/* Lower content */}
|
{/* Lower content */}
|
||||||
<div className={theme.lowerContent()}>
|
<div className={theme.lowerContent()}>
|
||||||
{/* Latest deployment */}
|
{/* Latest deployment */}
|
||||||
@ -135,7 +106,7 @@ export const ProjectCard = ({
|
|||||||
<div className={theme.deploymentText()}>
|
<div className={theme.deploymentText()}>
|
||||||
{hasDeployment ? (
|
{hasDeployment ? (
|
||||||
<>
|
<>
|
||||||
<GithubLogoIcon />
|
<GitHubLogo />
|
||||||
<span>{relativeTimeMs(project.deployments[0].createdAt)} on</span>
|
<span>{relativeTimeMs(project.deployments[0].createdAt)} on</span>
|
||||||
<BranchIcon />
|
<BranchIcon />
|
||||||
<span>{project.deployments[0].branch}</span>
|
<span>{project.deployments[0].branch}</span>
|
||||||
|
127
packages/frontend/src/components/projects/ProjectSearchBar.tsx
Normal file
127
packages/frontend/src/components/projects/ProjectSearchBar.tsx
Normal file
@ -0,0 +1,127 @@
|
|||||||
|
import React, { useCallback, useEffect, useState } from 'react';
|
||||||
|
import { useCombobox } from 'downshift';
|
||||||
|
import { Project } from 'gql-client';
|
||||||
|
import { useDebounce } from 'usehooks-ts';
|
||||||
|
|
||||||
|
import {
|
||||||
|
List,
|
||||||
|
ListItem,
|
||||||
|
ListItemPrefix,
|
||||||
|
Card,
|
||||||
|
Typography,
|
||||||
|
Avatar,
|
||||||
|
} from '@material-tailwind/react';
|
||||||
|
|
||||||
|
import SearchBar from '../SearchBar';
|
||||||
|
import { useGQLClient } from '../../context/GQLClientContext';
|
||||||
|
|
||||||
|
interface ProjectsSearchProps {
|
||||||
|
onChange?: (data: Project) => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
const ProjectSearchBar = ({ onChange }: ProjectsSearchProps) => {
|
||||||
|
const [items, setItems] = useState<Project[]>([]);
|
||||||
|
const [selectedItem, setSelectedItem] = useState<Project | null>(null);
|
||||||
|
const client = useGQLClient();
|
||||||
|
|
||||||
|
const {
|
||||||
|
isOpen,
|
||||||
|
getMenuProps,
|
||||||
|
getInputProps,
|
||||||
|
getItemProps,
|
||||||
|
highlightedIndex,
|
||||||
|
inputValue,
|
||||||
|
} = useCombobox({
|
||||||
|
items,
|
||||||
|
itemToString(item) {
|
||||||
|
return item ? item.name : '';
|
||||||
|
},
|
||||||
|
selectedItem,
|
||||||
|
onSelectedItemChange: ({ selectedItem: newSelectedItem }) => {
|
||||||
|
if (newSelectedItem) {
|
||||||
|
setSelectedItem(newSelectedItem);
|
||||||
|
|
||||||
|
if (onChange) {
|
||||||
|
onChange(newSelectedItem);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const debouncedInputValue = useDebounce<string>(inputValue, 500);
|
||||||
|
|
||||||
|
const fetchProjects = useCallback(
|
||||||
|
async (inputValue: string) => {
|
||||||
|
const { searchProjects } = await client.searchProjects(inputValue);
|
||||||
|
setItems(searchProjects);
|
||||||
|
},
|
||||||
|
[client],
|
||||||
|
);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
if (debouncedInputValue) {
|
||||||
|
fetchProjects(debouncedInputValue);
|
||||||
|
}
|
||||||
|
}, [fetchProjects, debouncedInputValue]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="relative">
|
||||||
|
<SearchBar {...getInputProps()} />
|
||||||
|
<Card
|
||||||
|
className={`absolute w-1/2 max-h-52 -mt-1 overflow-y-auto ${
|
||||||
|
(!inputValue || !isOpen) && 'hidden'
|
||||||
|
}`}
|
||||||
|
placeholder={''}
|
||||||
|
>
|
||||||
|
<List {...getMenuProps()}>
|
||||||
|
{items.length ? (
|
||||||
|
<>
|
||||||
|
<div className="p-3">
|
||||||
|
<Typography variant="small" color="gray" placeholder={''}>
|
||||||
|
Suggestions
|
||||||
|
</Typography>
|
||||||
|
</div>
|
||||||
|
{items.map((item, index) => (
|
||||||
|
<ListItem
|
||||||
|
selected={highlightedIndex === index || selectedItem === item}
|
||||||
|
key={item.id}
|
||||||
|
placeholder={''}
|
||||||
|
{...getItemProps({ item, index })}
|
||||||
|
>
|
||||||
|
<ListItemPrefix placeholder={''}>
|
||||||
|
<Avatar
|
||||||
|
src={item.icon || '/gray.png'}
|
||||||
|
variant="rounded"
|
||||||
|
placeholder={''}
|
||||||
|
/>
|
||||||
|
</ListItemPrefix>
|
||||||
|
<div>
|
||||||
|
<Typography variant="h6" color="blue-gray" placeholder={''}>
|
||||||
|
{item.name}
|
||||||
|
</Typography>
|
||||||
|
<Typography
|
||||||
|
variant="small"
|
||||||
|
color="gray"
|
||||||
|
className="font-normal"
|
||||||
|
placeholder={''}
|
||||||
|
>
|
||||||
|
{item.organization.name}
|
||||||
|
</Typography>
|
||||||
|
</div>
|
||||||
|
</ListItem>
|
||||||
|
))}
|
||||||
|
</>
|
||||||
|
) : (
|
||||||
|
<div className="p-3">
|
||||||
|
<Typography placeholder={''}>
|
||||||
|
^ No projects matching this name
|
||||||
|
</Typography>
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</List>
|
||||||
|
</Card>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default ProjectSearchBar;
|
Some files were not shown because too many files have changed in this diff Show More
Loading…
Reference in New Issue
Block a user