├── .github
├── dependabot.yml
└── workflows
│ └── break_action.yml
├── .gitignore
├── .mergify.yml
├── LICENSE
├── Procfile
├── README.md
├── app.json
├── client
├── .eslintignore
├── .gitignore
├── .prettierignore
├── package-lock.json
├── package.json
├── public
│ ├── favicon.ico
│ ├── google.svg
│ ├── index.html
│ ├── manifest.json
│ ├── rainbow192.png
│ ├── rainbow512.png
│ └── robots.txt
├── setupJest.ts
├── src
│ ├── __tests__
│ │ └── bytes.ts
│ ├── app.tsx
│ ├── components
│ │ ├── ClusterModal.tsx
│ │ ├── ClusterStatusButton.tsx
│ │ ├── ConfigurationSidebar.tsx
│ │ ├── Header.tsx
│ │ ├── LoadingModal.tsx
│ │ ├── PaymentCard.tsx
│ │ ├── SlotTableRow.tsx
│ │ ├── TxContainer.tsx
│ │ ├── TxModal.tsx
│ │ ├── TxSquare.tsx
│ │ └── TxTableRow.tsx
│ ├── fonts
│ │ ├── cerebrisans
│ │ │ ├── cerebrisans-bold.eot
│ │ │ ├── cerebrisans-bold.svg
│ │ │ ├── cerebrisans-bold.ttf
│ │ │ ├── cerebrisans-bold.woff
│ │ │ ├── cerebrisans-book.eot
│ │ │ ├── cerebrisans-book.svg
│ │ │ ├── cerebrisans-book.ttf
│ │ │ ├── cerebrisans-book.woff
│ │ │ ├── cerebrisans-medium.eot
│ │ │ ├── cerebrisans-medium.svg
│ │ │ ├── cerebrisans-medium.ttf
│ │ │ ├── cerebrisans-medium.woff
│ │ │ ├── cerebrisans-regular.eot
│ │ │ ├── cerebrisans-regular.svg
│ │ │ ├── cerebrisans-regular.ttf
│ │ │ ├── cerebrisans-regular.woff
│ │ │ ├── cerebrisans-semibold.eot
│ │ │ ├── cerebrisans-semibold.svg
│ │ │ ├── cerebrisans-semibold.ttf
│ │ │ └── cerebrisans-semibold.woff
│ │ └── feather
│ │ │ ├── feather.css
│ │ │ └── fonts
│ │ │ ├── Feather.svg
│ │ │ ├── Feather.ttf
│ │ │ └── Feather.woff
│ ├── images
│ │ ├── SOLANA-preview.png
│ │ ├── break.svg
│ │ ├── dashkit
│ │ │ ├── covers
│ │ │ │ ├── auth-side-cover.jpg
│ │ │ │ ├── header-cover.jpg
│ │ │ │ ├── profile-cover-1.jpg
│ │ │ │ ├── profile-cover-2.jpg
│ │ │ │ ├── profile-cover-3.jpg
│ │ │ │ ├── profile-cover-4.jpg
│ │ │ │ ├── profile-cover-5.jpg
│ │ │ │ ├── profile-cover-6.jpg
│ │ │ │ ├── profile-cover-7.jpg
│ │ │ │ ├── profile-cover-8.jpg
│ │ │ │ ├── sidebar-cover.jpg
│ │ │ │ └── team-cover.jpg
│ │ │ └── masks
│ │ │ │ ├── avatar-group-hover-last.svg
│ │ │ │ ├── avatar-group-hover.svg
│ │ │ │ ├── avatar-group.svg
│ │ │ │ ├── avatar-status.svg
│ │ │ │ └── icon-status.svg
│ │ ├── graphic.svg
│ │ ├── hero.svg
│ │ ├── icons
│ │ │ ├── arrows.svg
│ │ │ ├── capacity.svg
│ │ │ ├── close.svg
│ │ │ ├── cup.svg
│ │ │ ├── loader.svg
│ │ │ ├── processing.svg
│ │ │ ├── table-cup.svg
│ │ │ ├── tap.svg
│ │ │ ├── timer.svg
│ │ │ └── user.svg
│ │ ├── logo.svg
│ │ ├── share-facebook-2.svg
│ │ ├── share-facebook.svg
│ │ ├── share-twitter.svg
│ │ └── solana.svg
│ ├── index.tsx
│ ├── pages
│ │ ├── GamePage.tsx
│ │ ├── HomePage.tsx
│ │ ├── ResultsPage.tsx
│ │ ├── SlotsPage.tsx
│ │ ├── StartPage.tsx
│ │ └── WalletPage.tsx
│ ├── providers
│ │ ├── accounts.tsx
│ │ ├── config.tsx
│ │ ├── game.tsx
│ │ ├── rpc
│ │ │ ├── balance.tsx
│ │ │ ├── blockhash.tsx
│ │ │ └── index.tsx
│ │ ├── server
│ │ │ ├── http
│ │ │ │ ├── config.ts
│ │ │ │ ├── index.tsx
│ │ │ │ └── request.tsx
│ │ │ ├── index.tsx
│ │ │ └── socket.tsx
│ │ ├── slot.tsx
│ │ ├── torus.tsx
│ │ ├── transactions
│ │ │ ├── confirmed.tsx
│ │ │ ├── create.tsx
│ │ │ ├── index.tsx
│ │ │ ├── selected.tsx
│ │ │ └── tps.tsx
│ │ └── wallet.tsx
│ ├── react-app-env.d.ts
│ ├── serviceWorker.ts
│ ├── setupTests.ts
│ ├── styles
│ │ ├── animate.scss
│ │ ├── custom-variables.scss
│ │ ├── custom.scss
│ │ ├── custom
│ │ │ ├── _app.scss
│ │ │ ├── _game.scss
│ │ │ ├── _header.scss
│ │ │ ├── _home.scss
│ │ │ ├── _payment-card.scss
│ │ │ ├── _results.scss
│ │ │ ├── _sidebar.scss
│ │ │ ├── _tx-container.scss
│ │ │ └── _tx-square.scss
│ │ ├── dashkit
│ │ │ ├── _alert.scss
│ │ │ ├── _avatar.scss
│ │ │ ├── _badge.scss
│ │ │ ├── _breadcrumb.scss
│ │ │ ├── _buttons.scss
│ │ │ ├── _card.scss
│ │ │ ├── _chart.scss
│ │ │ ├── _checklist.scss
│ │ │ ├── _close.scss
│ │ │ ├── _code.scss
│ │ │ ├── _comment.scss
│ │ │ ├── _custom-forms.scss
│ │ │ ├── _dashkit.scss
│ │ │ ├── _dropdowns.scss
│ │ │ ├── _forms.scss
│ │ │ ├── _header.scss
│ │ │ ├── _icon.scss
│ │ │ ├── _jumbotron.scss
│ │ │ ├── _kanban.scss
│ │ │ ├── _list-group.scss
│ │ │ ├── _main-content.scss
│ │ │ ├── _mixins.scss
│ │ │ ├── _modal.scss
│ │ │ ├── _nav.scss
│ │ │ ├── _navbar.scss
│ │ │ ├── _popover.scss
│ │ │ ├── _progress.scss
│ │ │ ├── _reboot.scss
│ │ │ ├── _root.scss
│ │ │ ├── _tables.scss
│ │ │ ├── _toasts.scss
│ │ │ ├── _type.scss
│ │ │ ├── _utilities.scss
│ │ │ ├── _variables.scss
│ │ │ ├── _vendors.scss
│ │ │ ├── dark
│ │ │ │ ├── _overrides-dark.scss
│ │ │ │ └── _variables-dark.scss
│ │ │ ├── mixins
│ │ │ │ ├── _badge.scss
│ │ │ │ └── _breakpoints.scss
│ │ │ ├── utilities
│ │ │ │ ├── _background.scss
│ │ │ │ ├── _borders.scss
│ │ │ │ ├── _lift.scss
│ │ │ │ ├── _sizing.scss
│ │ │ │ └── _type.scss
│ │ │ └── vendors
│ │ │ │ ├── _dropzone.scss
│ │ │ │ ├── _flatpickr.scss
│ │ │ │ ├── _list.scss
│ │ │ │ ├── _quill.scss
│ │ │ │ └── _select2.scss
│ │ ├── index.scss
│ │ └── tools
│ │ │ ├── shadows.scss
│ │ │ └── transitions.scss
│ ├── utils
│ │ ├── bytes.ts
│ │ └── index.ts
│ ├── worker-loader.d.ts
│ └── workers
│ │ ├── create-transaction-rpc.ts
│ │ └── create-transaction-worker-script.ts
├── svgTransform.js
└── tsconfig.json
├── package-lock.json
├── package.json
├── package.sh
├── program
├── .gitignore
├── Cargo.lock
├── Cargo.toml
├── Xargo.toml
├── dist
│ └── break_solana_program.so
└── src
│ └── lib.rs
└── server
├── .eslintignore
├── .eslintrc.json
├── .gitignore
├── .prettierignore
├── package-lock.json
├── package.json
├── src
├── api.ts
├── available_nodes.ts
├── index.ts
├── leader_schedule.ts
├── leader_tracker.ts
├── program.ts
├── tpu_proxy.ts
├── urls.ts
├── utils.ts
└── websocket.ts
├── tsconfig.json
└── webpack.config.js
/.github/dependabot.yml:
--------------------------------------------------------------------------------
1 | # To get started with Dependabot version updates, you'll need to specify which
2 | # package ecosystems to update and where the package manifests are located.
3 | # Please see the documentation for all configuration options:
4 | # https://help.github.com/github/administering-a-repository/configuration-options-for-dependency-updates
5 |
6 | version: 2
7 | updates:
8 | - package-ecosystem: npm
9 | directory: "/client"
10 | schedule:
11 | interval: daily
12 | time: "01:00"
13 | timezone: America/Los_Angeles
14 | labels:
15 | - "automerge"
16 | open-pull-requests-limit: 3
17 |
18 | - package-ecosystem: npm
19 | directory: "/server"
20 | schedule:
21 | interval: daily
22 | time: "01:00"
23 | timezone: America/Los_Angeles
24 | labels:
25 | - "automerge"
26 | open-pull-requests-limit: 3
27 |
--------------------------------------------------------------------------------
/.github/workflows/break_action.yml:
--------------------------------------------------------------------------------
1 | name: node_js
2 | on:
3 | push:
4 | branches: [main]
5 | pull_request:
6 | branches: [main]
7 |
8 | jobs:
9 | client:
10 | runs-on: ubuntu-latest
11 | env:
12 | PACKAGE_LOCATION: client
13 | steps:
14 | - name: Checkout repo
15 | uses: actions/checkout@v2
16 | - name: Use Node.js ${{ matrix.node-version }}
17 | uses: actions/setup-node@v2
18 | with:
19 | node-version: lts/*
20 |
21 | - name: Build
22 | run: |
23 | cd $PACKAGE_LOCATION
24 | npm ci
25 | npm run format
26 | npm run lint
27 | npm run build
28 | npm run test
29 | server:
30 | runs-on: ubuntu-latest
31 | env:
32 | PACKAGE_LOCATION: server
33 | steps:
34 | - name: Checkout repository
35 | uses: actions/checkout@v2
36 | - name: Use Node.js ${{ matrix.node-version }}
37 | uses: actions/setup-node@v2
38 | with:
39 | node-version: lts/*
40 | - name: Build
41 | run: |
42 | cd $PACKAGE_LOCATION
43 | npm ci
44 | npm run format
45 | npm run lint
46 | npm run build
47 | npm run test
48 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | .idea/
2 | build/
3 | *.iml
4 | *.log
5 | /dist/
6 | node_modules
7 |
--------------------------------------------------------------------------------
/.mergify.yml:
--------------------------------------------------------------------------------
1 | pull_request_rules:
2 | - name: automatic merge on CI success
3 | conditions:
4 | - status-success=client
5 | - status-success=server
6 | - label=automerge
7 | actions:
8 | merge:
9 | method: rebase
10 | - name: remove automerge label on CI failure
11 | conditions:
12 | - label=automerge
13 | - or:
14 | - status-failure=client
15 | - status-failure=server
16 | actions:
17 | label:
18 | remove:
19 | - automerge
20 | - name: remove outdated reviews
21 | conditions:
22 | - base=main
23 | actions:
24 | dismiss_reviews:
25 | changes_requested: true
26 |
--------------------------------------------------------------------------------
/LICENSE:
--------------------------------------------------------------------------------
1 | Copyright (c) 2019 Solana Labs, Inc
2 |
3 | Permission is hereby granted, free of charge, to any person obtaining
4 | a copy of this software and associated documentation files (the
5 | "Software"), to deal in the Software without restriction, including
6 | without limitation the rights to use, copy, modify, merge, publish,
7 | distribute, sublicense, and/or sell copies of the Software, and to
8 | permit persons to whom the Software is furnished to do so, subject to
9 | the following conditions:
10 |
11 | The above copyright notice and this permission notice shall be
12 | included in all copies or substantial portions of the Software.
13 |
14 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
15 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
17 | NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
18 | LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
19 | OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
20 | WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
21 |
--------------------------------------------------------------------------------
/Procfile:
--------------------------------------------------------------------------------
1 | web: npm run start
2 |
--------------------------------------------------------------------------------
/app.json:
--------------------------------------------------------------------------------
1 | {
2 | "buildpacks": [
3 | {
4 | "url": "heroku/nodejs"
5 | }
6 | ]
7 | }
8 |
--------------------------------------------------------------------------------
/client/.eslintignore:
--------------------------------------------------------------------------------
1 | dist
2 | build
3 | webpack
4 |
--------------------------------------------------------------------------------
/client/.gitignore:
--------------------------------------------------------------------------------
1 | dist
2 | build
3 | node_modules
--------------------------------------------------------------------------------
/client/.prettierignore:
--------------------------------------------------------------------------------
1 | dist
2 | build
3 | dashkit
4 | animate.scss
5 |
--------------------------------------------------------------------------------
/client/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@solana/break",
3 | "version": "1.0.0",
4 | "private": true,
5 | "scripts": {
6 | "start": "react-scripts start",
7 | "build": "react-scripts build",
8 | "test": "react-scripts test",
9 | "eject": "react-scripts eject",
10 | "lint": "echo \"lint is run during build\"",
11 | "format": "prettier -c \"**/*.+(js|jsx|ts|tsx|json|css|scss|md)\"",
12 | "format:fix": "prettier --write \"**/*.+(js|jsx|ts|tsx|json|css|scss|md)\""
13 | },
14 | "testnetDefaultChannel": "edge",
15 | "dependencies": {
16 | "@react-hook/debounce": "^4.0.0",
17 | "@react-hook/throttle": "^2.2.0",
18 | "@sentry/react": "^5.26.0",
19 | "@sentry/tracing": "^5.26.0",
20 | "@solana/web3.js": "^1.43.4",
21 | "@testing-library/jest-dom": "^5.16.4",
22 | "@testing-library/react": "^13.3.0",
23 | "@testing-library/user-event": "^14.2.0",
24 | "@toruslabs/fetch-node-details": "^6.0.1",
25 | "@toruslabs/torus.js": "^6.0.0",
26 | "@types/bs58": "^4.0.1",
27 | "@types/chai": "^4.3.1",
28 | "@types/chart.js": "^2.9.34",
29 | "@types/jest": "^27.5.1",
30 | "@types/node": "^17.0.35",
31 | "@types/qrcode.react": "^1.0.2",
32 | "@types/react": "^18.0.9",
33 | "@types/react-dom": "^17.0.14",
34 | "@types/react-router-dom": "^5.3.1",
35 | "@types/ua-parser-js": "^0.7.36",
36 | "bootstrap": "^4.6.0",
37 | "bs58": "^5.0.0",
38 | "buffer": "^6.0.3",
39 | "chai": "^4.3.6",
40 | "chart.js": "^2.9.4",
41 | "lodash.debounce": "^4.0.8",
42 | "node-sass": "^7.0.1",
43 | "prettier": "^2.6.2",
44 | "qrcode.react": "^3.0.2",
45 | "react": "^18.1.0",
46 | "react-chartjs-2": "^2.11.2",
47 | "react-dom": "^18.1.0",
48 | "react-google-login": "^5.2.2",
49 | "react-router-dom": "^5.3.0",
50 | "react-scripts": "^4.0.3",
51 | "tweetnacl": "^1.0.3",
52 | "typescript": "^4.7.2",
53 | "ua-parser-js": "^1.0.2",
54 | "worker-loader": "^3.0.8"
55 | },
56 | "eslintConfig": {
57 | "extends": "react-app"
58 | },
59 | "browserslist": {
60 | "production": [
61 | ">0.2%",
62 | "not dead",
63 | "not op_mini all"
64 | ],
65 | "development": [
66 | "last 1 chrome version",
67 | "last 1 firefox version",
68 | "last 1 safari version"
69 | ]
70 | }
71 | }
72 |
--------------------------------------------------------------------------------
/client/public/favicon.ico:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/public/favicon.ico
--------------------------------------------------------------------------------
/client/public/google.svg:
--------------------------------------------------------------------------------
1 | Google icon
--------------------------------------------------------------------------------
/client/public/index.html:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
17 |
21 |
22 |
26 |
27 |
36 | Break | Solana
37 |
41 |
42 |
43 | You need to enable JavaScript to run this app.
44 |
45 |
46 |
47 |
--------------------------------------------------------------------------------
/client/public/manifest.json:
--------------------------------------------------------------------------------
1 | {
2 | "short_name": "Solana Explorer",
3 | "name": "Explorer for Solana clusters",
4 | "icons": [
5 | {
6 | "src": "favicon.ico",
7 | "sizes": "64x64 32x32 24x24 16x16",
8 | "type": "image/x-icon"
9 | },
10 | {
11 | "src": "rainbow192.png",
12 | "type": "image/png",
13 | "sizes": "192x192"
14 | },
15 | {
16 | "src": "rainbow512.png",
17 | "type": "image/png",
18 | "sizes": "512x512"
19 | }
20 | ],
21 | "start_url": ".",
22 | "display": "standalone",
23 | "theme_color": "#75FBB4",
24 | "background_color": "#ffffff"
25 | }
26 |
--------------------------------------------------------------------------------
/client/public/rainbow192.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/public/rainbow192.png
--------------------------------------------------------------------------------
/client/public/rainbow512.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/public/rainbow512.png
--------------------------------------------------------------------------------
/client/public/robots.txt:
--------------------------------------------------------------------------------
1 | # https://www.robotstxt.org/robotstxt.html
2 | User-agent: *
3 | Disallow:
4 |
--------------------------------------------------------------------------------
/client/setupJest.ts:
--------------------------------------------------------------------------------
1 | import { GlobalWithFetchMock } from "jest-fetch-mock";
2 |
3 | const customGlobal: GlobalWithFetchMock = global as GlobalWithFetchMock;
4 | customGlobal.fetch = require("jest-fetch-mock");
5 | customGlobal.fetchMock = customGlobal.fetch;
6 |
--------------------------------------------------------------------------------
/client/src/__tests__/bytes.ts:
--------------------------------------------------------------------------------
1 | import * as Bytes from "../utils/bytes";
2 | import { expect } from "chai";
3 |
4 | describe("programDataToIds", () => {
5 | it("ids = []", () => {
6 | const ids = Bytes.programDataToIds(new Uint8Array([0]));
7 | expect(ids).to.eql([]);
8 | });
9 |
10 | it("ids = [0]", () => {
11 | const ids = Bytes.programDataToIds(new Uint8Array([128]));
12 | expect(ids).to.eql([0]);
13 | });
14 |
15 | it("ids = [8]", () => {
16 | const ids = Bytes.programDataToIds(new Uint8Array([0, 128]));
17 | expect(ids).to.eql([8]);
18 | });
19 |
20 | it("ids = [7, 8]", () => {
21 | const ids = Bytes.programDataToIds(new Uint8Array([1, 128]));
22 | expect(ids).to.eql([7, 8]);
23 | });
24 | });
25 |
26 | describe("instructionDataFromId", () => {
27 | it("id = 0", () => {
28 | const data = Bytes.instructionDataFromId(0);
29 | expect(data).to.eql(new Uint8Array([0, 0]));
30 | });
31 |
32 | it("id = 8", () => {
33 | const data = Bytes.instructionDataFromId(8);
34 | expect(data).to.eql(new Uint8Array([0, 8]));
35 | });
36 |
37 | it("id = 257", () => {
38 | const data = Bytes.instructionDataFromId(257);
39 | expect(data).to.eql(new Uint8Array([1, 1]));
40 | });
41 | });
42 |
--------------------------------------------------------------------------------
/client/src/app.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { Route, Switch, Redirect, useRouteMatch } from "react-router-dom";
3 |
4 | import { HomePage } from "pages/HomePage";
5 | import { GamePage } from "pages/GamePage";
6 | import { SlotsPage } from "pages/SlotsPage";
7 | import { WalletPage } from "pages/WalletPage";
8 | import { ResultsPage } from "pages/ResultsPage";
9 | import { StartPage } from "pages/StartPage";
10 |
11 | import { ClusterModal } from "components/ClusterModal";
12 | import { LoadingModal } from "components/LoadingModal";
13 | import { useGameState } from "providers/game";
14 | import { useClusterModal } from "providers/server";
15 | import { Header } from "components/Header";
16 | import { ConfigurationSidebar } from "components/ConfigurationSidebar";
17 |
18 | export default function App() {
19 | const isHomePage = !!useRouteMatch("/")?.isExact;
20 | const isWalletPage = !!useRouteMatch("/wallet")?.isExact;
21 | const isSlotsPage = !!useRouteMatch("/slots")?.isExact;
22 | const [showClusterModal] = useClusterModal();
23 | const gameState = useGameState();
24 |
25 | if (isHomePage) {
26 | return (
27 |
28 |
29 |
30 | );
31 | }
32 |
33 | const isLoading = gameState.loadingPhase !== "complete";
34 | const isInitializing =
35 | gameState.loadingPhase === "config" && !isWalletPage && !isSlotsPage;
36 | const showLoadingModal = isLoading && !isWalletPage && !isSlotsPage;
37 | const showConfigSidebar = !isHomePage && !isWalletPage;
38 | return (
39 |
40 |
41 |
42 |
43 | {showConfigSidebar && }
44 | {!isInitializing && (
45 |
46 |
47 |
48 |
49 |
50 |
51 |
52 |
53 | )}
54 |
55 |
56 |
57 |
58 |
59 |
60 | );
61 | }
62 |
63 | function Overlay({ show }: { show: boolean }) {
64 | if (show) return
;
65 | return
;
66 | }
67 |
--------------------------------------------------------------------------------
/client/src/components/ClusterModal.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Link } from "react-router-dom";
3 | import { useDebounceCallback } from "@react-hook/debounce";
4 | import { Location } from "history";
5 | import {
6 | useServer,
7 | serverName,
8 | useClusterModal,
9 | SERVERS,
10 | DEFAULT_SERVER,
11 | useCustomUrl,
12 | } from "../providers/server";
13 | import { useRpcUrlState } from "providers/rpc";
14 | import { useServerConfig } from "providers/server/http";
15 |
16 | export function ClusterModal() {
17 | const [show, setShow] = useClusterModal();
18 | const onClose = () => setShow(false);
19 | const { server } = useServer();
20 | return (
21 | <>
22 |
26 |
27 |
28 |
e.stopPropagation()}>
29 |
30 | ×
31 |
32 |
33 |
Choose a Cluster
34 |
35 |
36 |
37 |
38 | Override {serverName(server)} RPC
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 | >
47 | );
48 | }
49 |
50 | function CustomRpcInput() {
51 | const [rpcUrl, setRpcUrl] = useRpcUrlState();
52 | const [editing, setEditing] = React.useState(false);
53 | const configRpcUrl = useServerConfig()?.rpcUrl;
54 | const active = configRpcUrl !== rpcUrl;
55 |
56 | const customClass = (prefix: string) => (active ? `${prefix}-info` : "");
57 | const onUrlInput = useDebounceCallback((url: string) => {
58 | if (url.length > 0) {
59 | try {
60 | new URL(url);
61 | setRpcUrl(url);
62 | } catch (err) {
63 | // ignore bad url
64 | }
65 | } else if (configRpcUrl) {
66 | setRpcUrl(configRpcUrl);
67 | }
68 | }, 500);
69 |
70 | const defaultValue = active ? rpcUrl : "";
71 | const inputTextClass = editing ? "" : "text-muted";
72 | return (
73 | setEditing(true)}
81 | onBlur={() => setEditing(false)}
82 | onInput={(e) => onUrlInput(e.currentTarget.value)}
83 | />
84 | );
85 | }
86 |
87 | function CustomClusterInput() {
88 | const [customUrl, setCustomUrl] = useCustomUrl();
89 | const [editing, setEditing] = React.useState(false);
90 |
91 | const onUrlInput = useDebounceCallback((url: string) => {
92 | setCustomUrl(url);
93 | }, 500);
94 |
95 | const inputTextClass = editing ? "" : "text-muted";
96 | return (
97 | setEditing(true)}
102 | onBlur={() => setEditing(false)}
103 | onInput={(e) => onUrlInput(e.currentTarget.value)}
104 | />
105 | );
106 | }
107 |
108 | function ClusterToggle() {
109 | const { server } = useServer();
110 |
111 | return (
112 | <>
113 |
114 | {SERVERS.map((next) => {
115 | const active = next === server;
116 | const btnClass = active
117 | ? `active btn-dark border-info text-white`
118 | : "btn-dark";
119 |
120 | const clusterLocation = (location: Location) => {
121 | const params = new URLSearchParams(location.search);
122 | if (next !== DEFAULT_SERVER) {
123 | params.set("cluster", next);
124 | } else {
125 | params.delete("cluster");
126 | }
127 | return {
128 | ...location,
129 | search: params.toString(),
130 | };
131 | };
132 |
133 | return (
134 |
139 | {serverName(next)}
140 |
141 | );
142 | })}
143 |
144 | {server === "custom" && (
145 | <>
146 | Break Server URL
147 |
148 | >
149 | )}
150 | >
151 | );
152 | }
153 |
--------------------------------------------------------------------------------
/client/src/components/ClusterStatusButton.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { useServer, useClusterModal } from "../providers/server";
3 |
4 | function ClusterStatusButton() {
5 | const [, setShow] = useClusterModal();
6 | const { name } = useServer();
7 | return (
8 | setShow(true)}
11 | >
12 | {name}
13 |
14 | );
15 | }
16 |
17 | export default ClusterStatusButton;
18 |
--------------------------------------------------------------------------------
/client/src/components/Header.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import breakSvg from "images/break.svg";
4 | import solanaSvg from "images/solana.svg";
5 | import { useGameState } from "providers/game";
6 | import ClusterStatusButton from "./ClusterStatusButton";
7 | import { useClientConfig } from "providers/config";
8 |
9 | export function Header() {
10 | return (
11 |
12 |
13 |
14 |
15 |
16 |
17 |
18 |
19 |
36 |
37 |
42 |
43 |
44 |
45 | );
46 | }
47 |
48 | function HeaderCTA() {
49 | const gameState = useGameState();
50 | const [, setRefresh] = React.useState(false);
51 | const countdownStart = gameState.countdownStartTime;
52 | const gameStatus = gameState.status;
53 | const [{ countdownSeconds }] = useClientConfig();
54 |
55 | React.useEffect(() => {
56 | if (countdownStart !== undefined) {
57 | const timerId = setInterval(() => {
58 | setRefresh((r) => !r);
59 | }, 1000);
60 | return () => clearTimeout(timerId);
61 | }
62 | }, [countdownStart]);
63 |
64 | if (gameStatus === "loading" || gameStatus === "setup") {
65 | return null;
66 | }
67 |
68 | if (gameStatus === "finished") {
69 | return (
70 |
71 | Play Again
72 |
73 | );
74 | }
75 |
76 | let secondsRemaining = countdownSeconds;
77 | if (countdownStart !== undefined) {
78 | secondsRemaining = Math.max(
79 | 0,
80 | countdownSeconds - Math.floor((performance.now() - countdownStart) / 1000)
81 | );
82 | }
83 |
84 | return (
85 |
86 |
87 |
88 |
89 |
90 | {secondsRemaining}s
91 |
92 |
93 | );
94 | }
95 |
--------------------------------------------------------------------------------
/client/src/components/LoadingModal.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 |
3 | import { LoadingPhase } from "providers/game";
4 |
5 | export function LoadingModal({
6 | show,
7 | wallet,
8 | phase,
9 | }: {
10 | show: boolean;
11 | wallet?: boolean;
12 | phase?: LoadingPhase;
13 | }) {
14 | const renderContent = () => {
15 | if (!show) return null;
16 | let loadingText = "Loading";
17 | if (wallet) {
18 | loadingText = "Fetching wallet";
19 | } else if (phase) {
20 | switch (phase) {
21 | case "config":
22 | loadingText = "Initializing";
23 | break;
24 | case "blockhash":
25 | loadingText = "Connecting to cluster";
26 | break;
27 | case "costs":
28 | loadingText = "Calculating game cost";
29 | break;
30 | case "socket":
31 | loadingText = "Connecting to transaction forwarder";
32 | break;
33 | case "creating-accounts":
34 | loadingText = "Creating game accounts";
35 | break;
36 | }
37 | }
38 |
39 | return (
40 |
41 |
42 |
43 |
44 |
45 |
{loadingText}...
46 |
47 |
48 |
49 |
50 | );
51 | };
52 |
53 | return (
54 | {renderContent()}
55 | );
56 | }
57 |
--------------------------------------------------------------------------------
/client/src/components/SlotTableRow.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import { SlotTiming, useLeaderSchedule } from "providers/slot";
4 |
5 | interface Props {
6 | timing?: SlotTiming;
7 | slot: number;
8 | }
9 |
10 | export function timeElapsed(
11 | sentAt: number | undefined,
12 | receivedAt: number | undefined
13 | ): string | undefined {
14 | if (sentAt === undefined || receivedAt === undefined) return;
15 | return (Math.max(0, receivedAt - sentAt) / 1000).toFixed(3) + "s";
16 | }
17 |
18 | export function timestamp(sentAt: number | undefined): string | undefined {
19 | if (sentAt === undefined) return;
20 | const date = new Date(sentAt);
21 | const pad = (num: number, length: number) =>
22 | num.toString().padStart(length, "0");
23 | return `${pad(date.getHours(), 2)}:${pad(date.getMinutes(), 2)}:${pad(
24 | date.getSeconds(),
25 | 2
26 | )}.${pad(date.getMilliseconds(), 3)}`;
27 | }
28 |
29 | export function SlotTableRow({ slot, timing }: Props) {
30 | const schedule = useLeaderSchedule().current;
31 | function TdTimestamp({ time }: { time: number | undefined }) {
32 | return (
33 |
34 |
35 |
36 | {timeElapsed(timing?.firstShred, time)}
37 |
38 |
{timestamp(time) || "-"}
39 |
40 |
41 | );
42 | }
43 |
44 | const leader = React.useMemo(() => {
45 | if (schedule) {
46 | const [offset, leaderSchedule] = schedule;
47 | for (let [leader, slots] of Object.entries(leaderSchedule)) {
48 | if (slots.indexOf(slot - offset) >= 0) {
49 | return leader;
50 | }
51 | }
52 | }
53 | }, [slot, schedule]);
54 |
55 | let txCount, txSuccessRate, avgTpe;
56 | if (timing?.stats) {
57 | txCount =
58 | timing.stats.numSuccessfulTransactions +
59 | timing.stats.numFailedTransactions;
60 | const rawTxRate = timing.stats.numSuccessfulTransactions / txCount;
61 | txSuccessRate = `${(100 * rawTxRate).toFixed(1)}%`;
62 | avgTpe = (txCount / timing?.stats.numTransactionEntries).toFixed(1);
63 | }
64 |
65 | return (
66 |
67 | {leader ? leader.slice(0, 7) : "-"}
68 |
69 |
70 |
71 | {timing?.parent + "┐" || "-"}
72 |
73 |
{slot}
74 |
75 |
76 | {txCount || "-"}
77 | {txSuccessRate || "-"}
78 | {timing?.stats?.numTransactionEntries || "-"}
79 | {avgTpe || "-"}
80 | {timing?.stats?.maxTransactionsPerEntry || "-"}
81 |
82 |
83 |
84 | {timing?.err === undefined ? (
85 | <>
86 |
87 |
88 |
89 | >
90 | ) : (
91 | <>
92 |
93 | {timing?.err}
94 | >
95 | )}
96 |
97 | );
98 | }
99 |
--------------------------------------------------------------------------------
/client/src/components/TxSquare.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import "styles/animate.scss";
4 | import { TransactionState } from "providers/transactions";
5 | import { useSelectTransaction } from "providers/transactions/selected";
6 |
7 | interface Props {
8 | transaction: TransactionState;
9 | }
10 |
11 | export function TransactionSquare({ transaction }: Props) {
12 | const { status, details } = transaction;
13 | const selectTransaction = useSelectTransaction();
14 |
15 | let statusClass = "";
16 | if (transaction.status === "success") {
17 | statusClass = "primary";
18 | } else if (status === "failed") {
19 | statusClass = "danger";
20 | } else if (status === "timeout") {
21 | statusClass = "warning";
22 | } else {
23 | statusClass = "dark";
24 | }
25 |
26 | return (
27 | selectTransaction(details.signature)}
29 | className={`btn d-flex flex-column justify-content-center align-items-center square slideInRight btn-${statusClass}`}
30 | >
31 | );
32 | }
33 |
--------------------------------------------------------------------------------
/client/src/components/TxTableRow.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import "styles/animate.scss";
4 | import { TransactionState } from "providers/transactions";
5 | import { useSelectTransaction } from "providers/transactions/selected";
6 | import { useSlotTiming } from "providers/slot";
7 | import type { SlotTiming } from "providers/slot";
8 |
9 | interface Props {
10 | transaction: TransactionState;
11 | }
12 |
13 | export function timeElapsed(
14 | sentAt: number | undefined,
15 | receivedAt: number | undefined
16 | ): string | undefined {
17 | if (sentAt === undefined || receivedAt === undefined) return;
18 | return (Math.max(0, receivedAt - sentAt) / 1000).toFixed(3) + "s";
19 | }
20 |
21 | export function TxTableRow({ transaction }: Props) {
22 | const signature = transaction.details.signature;
23 | const selectTransaction = useSelectTransaction();
24 | const slotMetrics = useSlotTiming();
25 |
26 | let targetSlot;
27 | let landedSlot: number | undefined;
28 | let timing;
29 | let received;
30 | let targetSlotOverride: string | undefined;
31 | if (transaction.status === "success") {
32 | targetSlot = transaction.slot.target;
33 | landedSlot = transaction.slot.landed;
34 | timing = transaction.timing;
35 | received = transaction.received;
36 | } else if (
37 | transaction.status === "timeout" ||
38 | transaction.status === "failed"
39 | ) {
40 | targetSlotOverride = transaction.status.toUpperCase();
41 | } else {
42 | targetSlot = transaction.pending.targetSlot;
43 | timing = transaction.timing;
44 | received = transaction.received;
45 | }
46 |
47 | let slotTiming: SlotTiming | undefined;
48 | let landedTime: number | undefined;
49 | if (landedSlot !== undefined) {
50 | landedTime = received?.find((r) => r.slot === landedSlot)?.timestamp;
51 | slotTiming = slotMetrics.current.get(landedSlot);
52 | }
53 |
54 | let txCount, txSuccessRate, avgTpe;
55 | if (slotTiming?.stats) {
56 | txCount =
57 | slotTiming.stats.numSuccessfulTransactions +
58 | slotTiming.stats.numFailedTransactions;
59 | const rawTxRate = slotTiming.stats.numSuccessfulTransactions / txCount;
60 | txSuccessRate = `${(100 * rawTxRate).toFixed(1)}%`;
61 | avgTpe = (txCount / slotTiming?.stats.numTransactionEntries).toFixed(1);
62 | }
63 |
64 | return (
65 | selectTransaction(signature)}
68 | >
69 | {signature.slice(0, 7)}…
70 | {targetSlot || targetSlotOverride || "-"}
71 | {landedSlot || "-"}
72 | {txCount || "-"}
73 | {txSuccessRate || "-"}
74 | {slotTiming?.stats?.numTransactionEntries || "-"}
75 | {avgTpe || "-"}
76 | {slotTiming?.stats?.maxTransactionsPerEntry || "-"}
77 | {timeElapsed(timing?.subscribed, slotTiming?.firstShred) || "-"}
78 | {timeElapsed(timing?.subscribed, landedTime) || "-"}
79 | {timeElapsed(timing?.subscribed, slotTiming?.fullSlot) || "-"}
80 | {timeElapsed(timing?.subscribed, slotTiming?.createdBank) || "-"}
81 | {timeElapsed(timing?.subscribed, slotTiming?.frozen) || "-"}
82 | {timeElapsed(timing?.subscribed, slotTiming?.confirmed) || "-"}
83 | {timeElapsed(timing?.subscribed, slotTiming?.rooted) || "-"}
84 |
85 | );
86 | }
87 |
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-bold.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-bold.eot
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-bold.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-bold.ttf
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-bold.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-bold.woff
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-book.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-book.eot
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-book.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-book.ttf
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-book.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-book.woff
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-medium.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-medium.eot
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-medium.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-medium.ttf
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-medium.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-medium.woff
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-regular.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-regular.eot
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-regular.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-regular.ttf
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-regular.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-regular.woff
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-semibold.eot:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-semibold.eot
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-semibold.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-semibold.ttf
--------------------------------------------------------------------------------
/client/src/fonts/cerebrisans/cerebrisans-semibold.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/cerebrisans/cerebrisans-semibold.woff
--------------------------------------------------------------------------------
/client/src/fonts/feather/fonts/Feather.ttf:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/feather/fonts/Feather.ttf
--------------------------------------------------------------------------------
/client/src/fonts/feather/fonts/Feather.woff:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/fonts/feather/fonts/Feather.woff
--------------------------------------------------------------------------------
/client/src/images/SOLANA-preview.png:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/SOLANA-preview.png
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/auth-side-cover.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/auth-side-cover.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/header-cover.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/header-cover.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-1.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-1.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-2.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-2.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-3.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-3.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-4.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-4.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-5.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-5.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-6.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-6.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-7.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-7.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/profile-cover-8.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/profile-cover-8.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/sidebar-cover.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/sidebar-cover.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/covers/team-cover.jpg:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/client/src/images/dashkit/covers/team-cover.jpg
--------------------------------------------------------------------------------
/client/src/images/dashkit/masks/avatar-group-hover-last.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/client/src/images/dashkit/masks/avatar-group-hover.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
--------------------------------------------------------------------------------
/client/src/images/dashkit/masks/avatar-group.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/client/src/images/dashkit/masks/avatar-status.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/client/src/images/dashkit/masks/icon-status.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/client/src/images/icons/arrows.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
--------------------------------------------------------------------------------
/client/src/images/icons/capacity.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
12 |
13 |
--------------------------------------------------------------------------------
/client/src/images/icons/close.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/client/src/images/icons/cup.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
--------------------------------------------------------------------------------
/client/src/images/icons/loader.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/client/src/images/icons/processing.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
--------------------------------------------------------------------------------
/client/src/images/icons/table-cup.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/client/src/images/icons/tap.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
9 |
15 |
16 |
17 |
23 |
29 |
30 |
31 |
37 |
38 |
39 |
45 |
46 |
47 |
53 |
54 |
55 |
61 |
62 |
63 |
69 |
70 |
--------------------------------------------------------------------------------
/client/src/images/icons/timer.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
6 |
7 |
8 |
9 |
10 |
11 |
--------------------------------------------------------------------------------
/client/src/images/icons/user.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/client/src/images/share-facebook-2.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/client/src/images/share-facebook.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/client/src/images/share-twitter.svg:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
--------------------------------------------------------------------------------
/client/src/index.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import ReactDOM from "react-dom";
3 | import { BrowserRouter } from "react-router-dom";
4 | // import * as Sentry from "@sentry/react";
5 | // import { Integrations } from "@sentry/tracing";
6 |
7 | import "styles/index.scss";
8 |
9 | import App from "./app";
10 | import { WalletProvider } from "providers/wallet";
11 | import { TransactionsProvider } from "providers/transactions";
12 | import { GameStateProvider } from "providers/game";
13 | import { ServerProvider } from "providers/server";
14 | import { RpcProvider } from "providers/rpc";
15 | import { SlotProvider } from "providers/slot";
16 | import { AccountsProvider } from "providers/accounts";
17 | import { TorusProvider } from "providers/torus";
18 | import { ConfigProvider } from "providers/config";
19 |
20 | // if (process.env.NODE_ENV === "production") {
21 | // Sentry.init({
22 | // dsn:
23 | // "https://727cd3fff6f949449c1ce5030928e667@o434108.ingest.sentry.io/5411826",
24 | // integrations: [new Integrations.BrowserTracing()],
25 | // tracesSampleRate: 1.0,
26 | // });
27 | // }
28 |
29 | ReactDOM.render(
30 |
31 |
32 |
33 |
34 |
35 |
36 |
37 |
38 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
48 |
49 |
50 | ,
51 | document.getElementById("root")
52 | );
53 |
--------------------------------------------------------------------------------
/client/src/pages/GamePage.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { Redirect, useLocation } from "react-router-dom";
3 |
4 | import { TransactionContainer } from "components/TxContainer";
5 | import { TransactionModal } from "components/TxModal";
6 | import {
7 | useTps,
8 | useCreatedCount,
9 | useAvgConfirmationTime,
10 | } from "providers/transactions";
11 | import { useWalletState } from "providers/wallet";
12 | import { useActiveUsers } from "providers/server/socket";
13 | import { useGameState } from "providers/game";
14 |
15 | export function GamePage() {
16 | const gameState = useGameState();
17 | const gameStatus = gameState.status;
18 | const countdownStart = gameState.countdownStartTime;
19 | const showSetup = gameStatus === "setup";
20 | const showStats = gameStatus === "play" || countdownStart !== undefined;
21 | const payer = useWalletState().wallet;
22 | const location = useLocation();
23 |
24 | if (showSetup) {
25 | if (!payer) {
26 | return ;
27 | } else {
28 | return ;
29 | }
30 | }
31 |
32 | return (
33 |
34 | {showStats &&
}
35 |
40 |
41 |
42 | );
43 | }
44 |
45 | export function EmptyCard() {
46 | return (
47 |
51 | );
52 | }
53 |
54 | function Stats() {
55 | const createdCount = useCreatedCount();
56 | const avgConfTime = useAvgConfirmationTime().toFixed(2);
57 | const tps = useTps();
58 | const activeUsers = useActiveUsers();
59 |
60 | return (
61 |
62 |
63 |
68 |
69 |
70 |
71 | );
72 | }
73 |
74 | type StatProps = {
75 | label: React.ReactNode;
76 | value: React.ReactNode;
77 | icon: string;
78 | };
79 | function StatCard({ label, value, icon }: StatProps) {
80 | return (
81 |
82 |
83 |
84 |
85 |
86 |
{label}
87 | {value}
88 |
89 |
90 |
91 |
92 |
93 |
94 |
95 |
96 | );
97 | }
98 |
--------------------------------------------------------------------------------
/client/src/pages/HomePage.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import graphic from "images/graphic.svg";
4 | import breakSvg from "images/break.svg";
5 | import solanaSvg from "images/solana.svg";
6 | import { Link } from "react-router-dom";
7 |
8 | export class HomePage extends React.Component {
9 | render() {
10 | return (
11 |
12 |
13 |
14 |
15 |
20 |
21 |
22 |
23 |
24 |
25 | Solana is the world’s most performant blockchain — currently, it
26 | can handle 50,000 transactions per second with 400
27 | millisecond block times.
28 |
29 |
30 | To see just how fast it is, you're invited to come and Break
31 | Solana. During the next fifteen seconds, send as many
32 | transactions as you can by mashing your keyboard.
33 |
34 |
35 |
36 |
40 | Play the game
41 |
42 |
43 |
44 |
45 |
46 |
47 | );
48 | }
49 | }
50 |
--------------------------------------------------------------------------------
/client/src/pages/SlotsPage.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import { useSlotTiming } from "providers/slot";
4 | import { SlotTableRow } from "components/SlotTableRow";
5 |
6 | export function SlotsPage() {
7 | return (
8 |
9 |
10 |
11 |
12 |
13 |
14 | Live Slot Stats
15 |
16 |
17 |
18 |
19 |
20 |
21 |
22 | );
23 | }
24 |
25 | function SlotTable() {
26 | const slotTiming = useSlotTiming();
27 | let min = Number.MAX_SAFE_INTEGER,
28 | max = 0;
29 | for (const slot of slotTiming.current.keys()) {
30 | min = Math.min(min, slot);
31 | max = Math.max(max, slot);
32 | }
33 |
34 | const slots = [];
35 | for (let i = min; i <= max; i++) {
36 | slots.push(i);
37 | }
38 |
39 | return (
40 |
41 |
42 |
43 |
44 |
45 |
46 | Leader
47 | Slot + Parent
48 | Tx Count
49 | Tx Success %
50 | Tx Entries
51 | Avg Tx Per Entry
52 | Max Tx Per Entry
53 | First Shred
54 | Shreds Full
55 | Bank Created
56 | Bank Frozen / Dead
57 | Confirmed
58 | Rooted
59 |
60 |
61 |
62 | {slots.map((slot) => (
63 |
68 | ))}
69 |
70 |
71 |
72 |
73 |
74 | );
75 | }
76 |
--------------------------------------------------------------------------------
/client/src/pages/StartPage.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 |
3 | import { useWalletState } from "providers/wallet";
4 | import { Redirect, useLocation } from "react-router-dom";
5 | import { PaymentCard } from "components/PaymentCard";
6 |
7 | export function StartPage() {
8 | const payer = useWalletState().wallet;
9 | const location = useLocation();
10 |
11 | if (!payer) {
12 | return ;
13 | }
14 |
15 | return (
16 |
23 | );
24 | }
25 |
--------------------------------------------------------------------------------
/client/src/providers/config.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { TrackedCommitment } from "./transactions";
3 |
4 | export interface ClientConfig {
5 | computeUnitPrice?: number;
6 | parallelization: number;
7 | trackedCommitment: TrackedCommitment;
8 | showDebugTable: boolean;
9 | countdownSeconds: number;
10 | retryTransactionEnabled: boolean;
11 | autoSendTransactions: boolean;
12 | useTpu: boolean;
13 | rpcUrl?: string;
14 | extraWriteAccount?: string;
15 | }
16 |
17 | const DEFAULT_CONFIG: ClientConfig = {
18 | parallelization: 4,
19 | trackedCommitment: "confirmed",
20 | showDebugTable: false,
21 | retryTransactionEnabled: false,
22 | autoSendTransactions: false,
23 | countdownSeconds: 15,
24 | useTpu: false,
25 | };
26 |
27 | type SetConfig = React.Dispatch>;
28 | const ConfigContext = React.createContext<
29 | [ClientConfig, SetConfig] | undefined
30 | >(undefined);
31 |
32 | type Props = { children: React.ReactNode };
33 | export function ConfigProvider({ children }: Props) {
34 | const stateHook = React.useState(DEFAULT_CONFIG);
35 | return (
36 |
37 | {children}
38 |
39 | );
40 | }
41 |
42 | export function useClientConfig() {
43 | const context = React.useContext(ConfigContext);
44 | if (!context) {
45 | throw new Error(`useClientConfig must be used within a ConfigProvider`);
46 | }
47 | return context;
48 | }
49 |
--------------------------------------------------------------------------------
/client/src/providers/game.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { useHistory, useLocation } from "react-router-dom";
3 |
4 | import { useServerConfig } from "providers/server/http";
5 | import { useSocket } from "providers/server/socket";
6 | import { useBlockhash } from "providers/rpc/blockhash";
7 | import { useDispatch as useTransactionsDispatch } from "providers/transactions";
8 | import { useAccountsState } from "./accounts";
9 | import { useConnection } from "./rpc";
10 | import { useClientConfig } from "./config";
11 |
12 | type GameStatus = "loading" | "setup" | "play" | "finished";
13 | export type LoadingPhase =
14 | | "blockhash"
15 | | "socket"
16 | | "config"
17 | | "costs"
18 | | "creating-accounts"
19 | | "complete";
20 |
21 | export interface GameState {
22 | status: GameStatus;
23 | loadingPhase: LoadingPhase;
24 | countdownStartTime: number | undefined;
25 | prepareGame: () => void;
26 | resetGame: () => void;
27 | startGame: () => void;
28 | }
29 |
30 | const GameStateContext = React.createContext(undefined);
31 |
32 | type Props = { children: React.ReactNode };
33 | export function GameStateProvider({ children }: Props) {
34 | const [status, setGameStatus] = React.useState("loading");
35 | const [countdownStartTime, setCountdownStart] = React.useState();
36 | const connection = useConnection();
37 | const blockhash = useBlockhash();
38 | const serverConfig = useServerConfig();
39 | const [clientConfig] = useClientConfig();
40 | const socket = useSocket();
41 | const accountsState = useAccountsState();
42 | const loadingPhase: LoadingPhase = React.useMemo(() => {
43 | if (!serverConfig) return "config";
44 | if (!blockhash) return "blockhash";
45 | if (!accountsState.creationCost) return "costs";
46 | if (!socket) return "socket";
47 | if (accountsState.status === "creating") return "creating-accounts";
48 | return "complete";
49 | }, [blockhash, serverConfig, socket, accountsState]);
50 |
51 | React.useEffect(() => {
52 | setGameStatus("loading");
53 | setCountdownStart(undefined);
54 | }, [
55 | connection,
56 | clientConfig.showDebugTable,
57 | clientConfig.parallelization,
58 | clientConfig.trackedCommitment,
59 | ]);
60 |
61 | React.useEffect(() => {
62 | if (status === "loading" && loadingPhase === "complete") {
63 | setGameStatus("setup");
64 | }
65 | }, [status, loadingPhase]);
66 |
67 | const history = useHistory();
68 | const location = useLocation();
69 | const resultsTimerRef = React.useRef();
70 | React.useEffect(() => {
71 | if (countdownStartTime !== undefined) {
72 | if (!resultsTimerRef.current) {
73 | resultsTimerRef.current = setTimeout(() => {
74 | setGameStatus("finished");
75 | history.push({ ...location, pathname: "/results" });
76 | }, clientConfig.countdownSeconds * 1000);
77 | }
78 | } else if (resultsTimerRef.current) {
79 | clearTimeout(resultsTimerRef.current);
80 | resultsTimerRef.current = undefined;
81 | }
82 | }, [countdownStartTime, history, location, clientConfig.countdownSeconds]);
83 |
84 | const startGame = React.useCallback(() => {
85 | setCountdownStart(performance.now());
86 | }, []);
87 |
88 | const transactionsDispatch = useTransactionsDispatch();
89 | const resetGame = React.useCallback(() => {
90 | setGameStatus("setup");
91 | setCountdownStart(undefined);
92 | transactionsDispatch({ type: "reset" });
93 | accountsState.deactivate();
94 | history.push({ ...location, pathname: "/start" });
95 | }, [accountsState, history, location, transactionsDispatch]);
96 |
97 | const prepareGame = React.useCallback(() => {
98 | setGameStatus("play");
99 | accountsState.createAccounts();
100 | history.push({ ...location, pathname: "/game" });
101 | }, [accountsState, history, location]);
102 |
103 | const accountsStatus = accountsState.status;
104 | React.useEffect(() => {
105 | const shouldReset = status === "play" && accountsStatus === "inactive";
106 | if (shouldReset) {
107 | resetGame();
108 | }
109 | }, [status, accountsStatus, resetGame]);
110 |
111 | const gameState: GameState = React.useMemo(
112 | () => ({
113 | status,
114 | loadingPhase,
115 | countdownStartTime,
116 | resetGame,
117 | prepareGame,
118 | startGame,
119 | }),
120 | [
121 | status,
122 | loadingPhase,
123 | countdownStartTime,
124 | prepareGame,
125 | startGame,
126 | resetGame,
127 | ]
128 | );
129 |
130 | return (
131 |
132 | {children}
133 |
134 | );
135 | }
136 |
137 | export function useGameState() {
138 | const context = React.useContext(GameStateContext);
139 | if (!context) {
140 | throw new Error(`useGameState must be used within a GameStateProvider`);
141 | }
142 | return context;
143 | }
144 |
--------------------------------------------------------------------------------
/client/src/providers/rpc/balance.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { AccountInfo } from "@solana/web3.js";
3 |
4 | import { useWalletState } from "providers/wallet";
5 | import { useConnection } from "providers/rpc";
6 | import { getFeePayers, reportError } from "utils";
7 | import { useClientConfig } from "providers/config";
8 |
9 | type Balance = number | "loading";
10 |
11 | interface State {
12 | payer: Balance;
13 | feePayers: Array;
14 | }
15 |
16 | const StateContext = React.createContext(undefined);
17 |
18 | type Props = { children: React.ReactNode };
19 | export function BalanceProvider({ children }: Props) {
20 | const [balance, setBalance] = React.useState("loading");
21 | const [{ parallelization }] = useClientConfig();
22 | const [feePayerBalances, setFeePayerBalances] = React.useState(
23 | Array(parallelization).fill(0)
24 | );
25 | const payer = useWalletState().wallet;
26 | const connection = useConnection();
27 |
28 | const refreshBalance = React.useCallback(() => {
29 | if (payer === undefined || connection === undefined) {
30 | setBalance("loading");
31 | return;
32 | }
33 |
34 | (async () => {
35 | try {
36 | const balance = await connection.getBalance(payer.publicKey);
37 | setBalance(balance);
38 | } catch (err) {
39 | reportError(err, "Failed to refresh balance");
40 | }
41 | })();
42 | }, [payer, connection]);
43 |
44 | React.useEffect(() => {
45 | refreshBalance();
46 | const onChange = () => {
47 | if (document.visibilityState !== "visible") return;
48 | refreshBalance();
49 | };
50 |
51 | document.addEventListener("visibilitychange", onChange);
52 | return () => document.removeEventListener("visibilitychange", onChange);
53 | }, [refreshBalance]);
54 |
55 | React.useEffect(() => {
56 | if (!payer || !connection) return;
57 | const subscription = connection.onAccountChange(
58 | payer.publicKey,
59 | (accountInfo: AccountInfo) => setBalance(accountInfo.lamports)
60 | );
61 |
62 | return () => {
63 | connection.removeAccountChangeListener(subscription);
64 | };
65 | }, [payer, connection]);
66 |
67 | const feePayers = React.useMemo(
68 | () => getFeePayers(parallelization),
69 | [parallelization]
70 | );
71 | const feePayerCounter = React.useRef(0);
72 | React.useEffect(() => {
73 | if (!connection) return;
74 | feePayerCounter.current++;
75 | const currentCounter = feePayerCounter.current;
76 |
77 | (async () => {
78 | const balances = await Promise.all(
79 | feePayers.map((feePayer) => {
80 | return connection.getBalance(feePayer.publicKey);
81 | })
82 | );
83 | if (feePayerCounter.current === currentCounter) {
84 | setFeePayerBalances(balances);
85 | }
86 | })();
87 |
88 | const subscriptions = feePayers.map((feePayer, index) => {
89 | return connection.onAccountChange(
90 | feePayer.publicKey,
91 | (accountInfo: AccountInfo) => {
92 | setFeePayerBalances((balances) => {
93 | const copy = [...balances];
94 | copy[index] = accountInfo.lamports;
95 | return copy;
96 | });
97 | }
98 | );
99 | });
100 |
101 | return () => {
102 | subscriptions.forEach((subscription) => {
103 | connection.removeAccountChangeListener(subscription);
104 | });
105 | };
106 | }, [connection, feePayers]);
107 |
108 | const state = React.useMemo(
109 | () => ({
110 | payer: balance,
111 | feePayers: feePayerBalances,
112 | }),
113 | [balance, feePayerBalances]
114 | );
115 |
116 | return (
117 | {children}
118 | );
119 | }
120 |
121 | export function useBalanceState(): State {
122 | const state = React.useContext(StateContext);
123 | if (state === undefined) {
124 | throw new Error(`useBalanceState must be used within a BalanceProvider`);
125 | }
126 | return state;
127 | }
128 |
--------------------------------------------------------------------------------
/client/src/providers/rpc/blockhash.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { Blockhash, Commitment, Connection } from "@solana/web3.js";
3 | import { sleep, reportError } from "utils";
4 | import { useConnection } from ".";
5 |
6 | const POLL_INTERVAL_MS = 2000;
7 |
8 | export enum ActionType {
9 | Start,
10 | Stop,
11 | Update,
12 | }
13 |
14 | interface Stop {
15 | type: ActionType.Stop;
16 | }
17 |
18 | interface Update {
19 | type: ActionType.Update;
20 | blockhash: Blockhash;
21 | }
22 |
23 | interface State {
24 | blockhash?: Blockhash;
25 | }
26 |
27 | type Action = Stop | Update;
28 | type Dispatch = (action: Action) => void;
29 |
30 | function reducer(state: State, action: Action): State {
31 | switch (action.type) {
32 | case ActionType.Stop: {
33 | return {};
34 | }
35 | case ActionType.Update: {
36 | return Object.assign({}, state, {
37 | blockhash: action.blockhash,
38 | });
39 | }
40 | }
41 | }
42 |
43 | const StateContext = React.createContext(undefined);
44 | const DispatchContext = React.createContext(undefined);
45 |
46 | type BlockhashProviderProps = { children: React.ReactNode };
47 | export function BlockhashProvider({ children }: BlockhashProviderProps) {
48 | const [state, dispatch] = React.useReducer(reducer, {});
49 | const connection = useConnection();
50 | const connectionRef = React.useRef(connection);
51 | const refreshingRef = React.useRef(false);
52 |
53 | React.useEffect(() => {
54 | if (connection === undefined) return;
55 |
56 | connectionRef.current = connection;
57 | refresh(dispatch, connectionRef, refreshingRef);
58 | const timerId = window.setInterval(
59 | () => refresh(dispatch, connectionRef, refreshingRef),
60 | POLL_INTERVAL_MS
61 | );
62 |
63 | return () => {
64 | clearInterval(timerId);
65 | dispatch({ type: ActionType.Stop });
66 | };
67 | }, [connection]);
68 |
69 | return (
70 |
71 |
72 | {children}
73 |
74 |
75 | );
76 | }
77 |
78 | export function useBlockhash() {
79 | const state = React.useContext(StateContext);
80 | if (!state) {
81 | throw new Error(`useBlockhash must be used within a BlockhashProvider`);
82 | }
83 |
84 | return state.blockhash;
85 | }
86 |
87 | async function nodeProgress(
88 | connection: Connection,
89 | commitment: Commitment
90 | ): Promise<{ blockhash: Blockhash; slot: number }> {
91 | const [{ blockhash }, slot] = await Promise.all([
92 | connection.getLatestBlockhash(commitment),
93 | connection.getSlot(commitment),
94 | ]);
95 |
96 | return { blockhash, slot };
97 | }
98 |
99 | async function refresh(
100 | dispatch: Dispatch,
101 | connectionRef: React.MutableRefObject,
102 | refreshingRef: React.MutableRefObject
103 | ) {
104 | let blockhash = undefined;
105 | const connection = connectionRef.current;
106 | if (connection === undefined) return;
107 |
108 | if (refreshingRef.current) return;
109 | refreshingRef.current = true;
110 |
111 | let reported = false;
112 | while (blockhash === undefined && connection === connectionRef.current) {
113 | try {
114 | const processedProgress = await nodeProgress(connection, "processed");
115 | const confirmedProgress = await nodeProgress(connection, "confirmed");
116 | const finalizedProgress = await nodeProgress(connection, "finalized");
117 | console.log(
118 | `[${processedProgress.slot}, ${confirmedProgress.slot}, ${
119 | finalizedProgress.slot
120 | }], [${processedProgress.blockhash.slice(
121 | 0,
122 | 5
123 | )}, ${confirmedProgress.blockhash.slice(
124 | 0,
125 | 5
126 | )}, ${finalizedProgress.blockhash.slice(0, 5)}]`
127 | );
128 | blockhash = finalizedProgress.blockhash;
129 | dispatch({ type: ActionType.Update, blockhash });
130 | } catch (err) {
131 | if (!reported) reportError(err, "Failed to refresh blockhash");
132 | reported = true;
133 | await sleep(1000);
134 | }
135 | }
136 |
137 | refreshingRef.current = false;
138 | }
139 |
--------------------------------------------------------------------------------
/client/src/providers/rpc/index.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Connection } from "@solana/web3.js";
3 | import { useServerConfig } from "providers/server/http";
4 | import { BlockhashProvider } from "./blockhash";
5 | import { BalanceProvider } from "./balance";
6 |
7 | type SetUrl = (url: string) => void;
8 | type State = [string | undefined, SetUrl];
9 |
10 | type ConnectionState = {
11 | connection?: Connection;
12 | };
13 |
14 | const StateContext = React.createContext(undefined);
15 | const ConnectionContext = React.createContext(
16 | undefined
17 | );
18 |
19 | type ProviderProps = { children: React.ReactNode };
20 | export function RpcProvider({ children }: ProviderProps) {
21 | const state = React.useState();
22 | const [rpcUrl, setRpcUrl] = state;
23 |
24 | // Reset rpc url whenever config is fetched
25 | const configRpcUrl = useServerConfig()?.rpcUrl;
26 | React.useEffect(() => {
27 | setRpcUrl(configRpcUrl);
28 | }, [configRpcUrl, setRpcUrl]);
29 |
30 | const connection: ConnectionState = React.useMemo(() => {
31 | if (rpcUrl === undefined) return {};
32 | try {
33 | const url = new URL(rpcUrl).toString();
34 | return { connection: new Connection(url, "confirmed") };
35 | } catch (err) {
36 | console.error(err);
37 | return {};
38 | }
39 | }, [rpcUrl]);
40 |
41 | return (
42 |
43 |
44 |
45 | {children}
46 |
47 |
48 |
49 | );
50 | }
51 |
52 | export function useRpcUrlState(): State {
53 | const state = React.useContext(StateContext);
54 | if (state === undefined) {
55 | throw new Error(`useRpcUrlState must be used within a RpcProvider`);
56 | }
57 | return state;
58 | }
59 |
60 | export function useConnection(): Connection | undefined {
61 | const state = React.useContext(ConnectionContext);
62 | if (state === undefined) {
63 | throw new Error(`useConnection must be used within a RpcProvider`);
64 | }
65 | return state.connection;
66 | }
67 |
--------------------------------------------------------------------------------
/client/src/providers/server/http/config.ts:
--------------------------------------------------------------------------------
1 | import { PublicKey, Cluster } from "@solana/web3.js";
2 |
3 | export interface Config {
4 | cluster: Cluster | undefined;
5 | rpcUrl: string;
6 | programId: PublicKey;
7 | airdropEnabled: boolean;
8 | }
9 |
10 | function stringToCluster(str: string | undefined): Cluster | undefined {
11 | switch (str) {
12 | case "devnet":
13 | case "testnet":
14 | case "mainnet-beta": {
15 | return str;
16 | }
17 | default:
18 | return undefined;
19 | }
20 | }
21 |
22 | export function configFromInit(response: any): Config {
23 | const cluster = stringToCluster(response.cluster);
24 | return {
25 | cluster,
26 | rpcUrl: response.clusterUrl,
27 | programId: new PublicKey(response.programId),
28 | airdropEnabled: !response.paymentRequired,
29 | };
30 | }
31 |
--------------------------------------------------------------------------------
/client/src/providers/server/http/index.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Config as ServerConfig } from "./config";
3 | import { useServer } from "providers/server";
4 | import { fetchWithRetry } from "./request";
5 | import { Connection } from "@solana/web3.js";
6 |
7 | export enum ConfigStatus {
8 | Initialized,
9 | Fetching,
10 | Ready,
11 | Failure,
12 | }
13 |
14 | interface State {
15 | status: ConfigStatus;
16 | config?: ServerConfig;
17 | connection?: Connection;
18 | }
19 |
20 | interface Initialized {
21 | status: ConfigStatus.Initialized;
22 | config: ServerConfig;
23 | }
24 |
25 | interface Fetching {
26 | status: ConfigStatus.Fetching;
27 | }
28 |
29 | interface Ready {
30 | status: ConfigStatus.Ready;
31 | }
32 |
33 | interface Failure {
34 | status: ConfigStatus.Failure;
35 | config?: undefined;
36 | }
37 |
38 | export type Action = Initialized | Fetching | Ready | Failure;
39 | export type Dispatch = (action: Action) => void;
40 |
41 | function configReducer(state: State, action: Action): State {
42 | switch (action.status) {
43 | case ConfigStatus.Ready:
44 | case ConfigStatus.Initialized: {
45 | return { ...state, ...action };
46 | }
47 | case ConfigStatus.Failure: {
48 | if (state.status === ConfigStatus.Fetching) {
49 | return { ...state, ...action };
50 | } else {
51 | return state;
52 | }
53 | }
54 | case ConfigStatus.Fetching: {
55 | return {
56 | ...state,
57 | ...action,
58 | };
59 | }
60 | }
61 | }
62 |
63 | const StateContext = React.createContext(undefined);
64 | const RefContext = React.createContext<
65 | React.MutableRefObject | undefined
66 | >(undefined);
67 | const DispatchContext = React.createContext(undefined);
68 |
69 | type ApiProviderProps = { children: React.ReactNode };
70 | export function HttpProvider({ children }: ApiProviderProps) {
71 | const [state, dispatch] = React.useReducer(configReducer, {
72 | status: ConfigStatus.Fetching,
73 | });
74 |
75 | const { httpUrl } = useServer();
76 | const httpUrlRef = React.useRef(httpUrl);
77 | React.useEffect(() => {
78 | httpUrlRef.current = httpUrl;
79 | initConfig(dispatch, httpUrlRef);
80 | }, [httpUrl]);
81 |
82 | React.useEffect(() => {
83 | httpUrlRef.current = httpUrl;
84 | }, [httpUrl]);
85 |
86 | return (
87 |
88 |
89 | {children}
90 |
91 |
92 | );
93 | }
94 |
95 | async function initConfig(
96 | dispatch: Dispatch,
97 | httpUrlRef: React.MutableRefObject
98 | ): Promise {
99 | return fetchWithRetry(dispatch, httpUrlRef);
100 | }
101 |
102 | export function useServerConfig() {
103 | const context = React.useContext(StateContext);
104 | if (!context) {
105 | throw new Error(`useServerConfig must be used within a ApiProvider`);
106 | }
107 | return context.config;
108 | }
109 |
110 | export function useIsFetching() {
111 | const context = React.useContext(StateContext);
112 | if (!context) {
113 | throw new Error(`useIsFetching must be used within a ApiProvider`);
114 | }
115 | return context.status === ConfigStatus.Fetching;
116 | }
117 |
118 | export function useClusterParam(): string {
119 | const context = React.useContext(StateContext);
120 | if (!context) {
121 | throw new Error(`useClusterParam must be used within a ApiProvider`);
122 | }
123 | const cluster = context.config?.cluster;
124 | const rpcUrl = context.config?.rpcUrl;
125 | if (!cluster && rpcUrl) {
126 | return `cluster=custom&customUrl=${rpcUrl}`;
127 | } else if (cluster && cluster !== "mainnet-beta") {
128 | return `cluster=${cluster}`;
129 | } else {
130 | return "";
131 | }
132 | }
133 |
--------------------------------------------------------------------------------
/client/src/providers/server/http/request.tsx:
--------------------------------------------------------------------------------
1 | import { configFromInit } from "./config";
2 | import { sleep, reportError } from "utils";
3 | import { Action, Dispatch, ConfigStatus } from "./index";
4 |
5 | export async function fetchWithRetry(
6 | dispatch: Dispatch,
7 | httpUrlRef: React.MutableRefObject
8 | ) {
9 | dispatch({
10 | status: ConfigStatus.Fetching,
11 | });
12 |
13 | const httpUrl = httpUrlRef.current;
14 | while (httpUrl === httpUrlRef.current) {
15 | let response: Action | "retry" = await fetchInit(httpUrl);
16 | if (httpUrl !== httpUrlRef.current) break;
17 | if (response === "retry") {
18 | await sleep(2000);
19 | } else {
20 | dispatch(response);
21 | break;
22 | }
23 | }
24 | }
25 |
26 | async function fetchInit(httpUrl: string): Promise {
27 | try {
28 | const body = JSON.stringify({});
29 | const response = await fetch(
30 | new Request(httpUrl + "/init", {
31 | method: "POST",
32 | headers: {
33 | "Content-Type": "application/json",
34 | },
35 | body,
36 | })
37 | );
38 | const data = await response.json();
39 | if (!("clusterUrl" in data) || !("programId" in data)) {
40 | throw new Error("Received invalid response");
41 | }
42 |
43 | return {
44 | status: ConfigStatus.Initialized,
45 | config: configFromInit(data),
46 | };
47 | } catch (err) {
48 | reportError(err, "/init failed");
49 | return "retry";
50 | }
51 | }
52 |
--------------------------------------------------------------------------------
/client/src/providers/server/index.tsx:
--------------------------------------------------------------------------------
1 | import React from "react";
2 | import { Cluster } from "@solana/web3.js";
3 | import { useLocation } from "react-router-dom";
4 | import { isLocalHost } from "../../utils";
5 | import { HttpProvider } from "./http";
6 | import { SocketProvider } from "./socket";
7 |
8 | type Server = Cluster | "custom";
9 | export const DEFAULT_SERVER = isLocalHost() ? "custom" : "mainnet-beta";
10 | export const SERVERS: Server[] = isLocalHost()
11 | ? ["custom"]
12 | : ["mainnet-beta", "testnet", "devnet", "custom"];
13 |
14 | const DEFAULT_CUSTOM_URL = `http://${window.location.hostname}:${
15 | process.env.PORT || 8080
16 | }`;
17 |
18 | export function serverName(server: Server): string {
19 | switch (server) {
20 | case "mainnet-beta":
21 | return "Mainnet Beta";
22 | case "testnet":
23 | return "Testnet";
24 | case "devnet":
25 | return "Devnet";
26 | case "custom":
27 | return "Custom";
28 | }
29 | }
30 |
31 | function parseQuery(query: URLSearchParams): Server {
32 | const clusterParam = query.get("cluster");
33 | switch (clusterParam) {
34 | case "devnet":
35 | return "devnet";
36 | case "testnet":
37 | return "testnet";
38 | case "mainnet-beta":
39 | return "mainnet-beta";
40 | case "custom":
41 | return "custom";
42 | default:
43 | return DEFAULT_SERVER;
44 | }
45 | }
46 |
47 | type SetShowModal = React.Dispatch>;
48 | type ModalState = [boolean, SetShowModal];
49 | const ModalContext = React.createContext(undefined);
50 | type SetCustomUrl = React.Dispatch>;
51 | type SetServer = React.Dispatch>;
52 | type ServerState = {
53 | server: Server;
54 | setServer: SetServer;
55 | customUrl: string;
56 | setCustomUrl: SetCustomUrl;
57 | };
58 | const ServerContext = React.createContext(undefined);
59 |
60 | type ProviderProps = { children: React.ReactNode };
61 | export function ServerProvider({ children }: ProviderProps) {
62 | const query = new URLSearchParams(useLocation().search);
63 | const serverParam = parseQuery(query);
64 | const [server, setServer] = React.useState(serverParam);
65 | const [customUrl, setCustomUrl] = React.useState(DEFAULT_CUSTOM_URL);
66 | const [showModal, setShowModal] = React.useState(false);
67 |
68 | // Update state when query params change
69 | React.useEffect(() => {
70 | setServer(serverParam);
71 | }, [serverParam]);
72 |
73 | const modalState: ModalState = React.useMemo(() => {
74 | return [showModal, setShowModal];
75 | }, [showModal]);
76 |
77 | return (
78 |
81 |
82 |
83 | {children}
84 |
85 |
86 |
87 | );
88 | }
89 |
90 | function getServerUrl(server: Server, customUrl: string) {
91 | switch (server) {
92 | case "custom": {
93 | return customUrl;
94 | }
95 | default: {
96 | const useHttp = isLocalHost();
97 | let slug: string = server;
98 | if (server === "mainnet-beta") {
99 | slug = "mainnet";
100 | }
101 | return `${
102 | useHttp ? "http" : "https"
103 | }://break-solana-${slug}.herokuapp.com`;
104 | }
105 | }
106 | }
107 |
108 | export function useServer() {
109 | const context = React.useContext(ServerContext);
110 | if (!context) {
111 | throw new Error(`useServer must be used within a ServerProvider`);
112 | }
113 | const { server, customUrl } = context;
114 | const httpUrl = getServerUrl(server, customUrl);
115 | const webSocketUrl = httpUrl.replace("http", "ws");
116 |
117 | return {
118 | server,
119 | httpUrl,
120 | webSocketUrl,
121 | name: serverName(server),
122 | };
123 | }
124 |
125 | export function useCustomUrl(): [string, SetCustomUrl] {
126 | const context = React.useContext(ServerContext);
127 | if (!context) {
128 | throw new Error(`useCustomUrl must be used within a ServerProvider`);
129 | }
130 | return [context.customUrl, context.setCustomUrl];
131 | }
132 |
133 | export function useClusterModal() {
134 | const context = React.useContext(ModalContext);
135 | if (!context) {
136 | throw new Error(`useClusterModal must be used within a ServerProvider`);
137 | }
138 | return context;
139 | }
140 |
--------------------------------------------------------------------------------
/client/src/providers/server/socket.tsx:
--------------------------------------------------------------------------------
1 | import { useClientConfig } from "providers/config";
2 | import * as React from "react";
3 | import { useServer } from ".";
4 |
5 | type SetSocket = React.Dispatch>;
6 | const SocketContext = React.createContext(undefined);
7 |
8 | const FailureCallbackContext = React.createContext<
9 | React.MutableRefObject | undefined
10 | >(undefined);
11 |
12 | type SetActiveUsers = React.Dispatch>;
13 | const ActiveUsersContext = React.createContext(undefined);
14 |
15 | const SWITCH_URL_CODE = 4444;
16 |
17 | type ServerSocket = {
18 | socket: WebSocket;
19 | id: number;
20 | };
21 |
22 | let socketCounter = 0;
23 |
24 | type FailureCallback = (signature: string, reason: string) => void;
25 |
26 | type SocketProviderProps = { children: React.ReactNode };
27 | export function SocketProvider({ children }: SocketProviderProps) {
28 | let [socket, setSocket] = React.useState(undefined);
29 | let failureCallbackRef = React.useRef(() => {});
30 | let [activeUsers, setActiveUsers] = React.useState(1);
31 | let [{ useTpu, rpcUrl }] = useClientConfig();
32 |
33 | const { webSocketUrl } = useServer();
34 | React.useEffect(() => {
35 | newSocket(webSocketUrl, setSocket, setActiveUsers, failureCallbackRef);
36 | }, [webSocketUrl]);
37 |
38 | React.useEffect(() => {
39 | if (socket) {
40 | socket.socket.send(useTpu ? "tpu" : "rpc");
41 | }
42 | }, [socket, useTpu]);
43 |
44 | React.useEffect(() => {
45 | if (socket && rpcUrl) {
46 | socket.socket.send(rpcUrl);
47 | }
48 | }, [socket, rpcUrl]);
49 |
50 | return (
51 |
52 |
53 |
54 | {children}
55 |
56 |
57 |
58 | );
59 | }
60 |
61 | function newSocket(
62 | webSocketUrl: string,
63 | setSocket: SetSocket,
64 | setActiveUsers: SetActiveUsers,
65 | failureCallbackRef: React.MutableRefObject
66 | ): WebSocket | undefined {
67 | socketCounter++;
68 | const id = socketCounter;
69 |
70 | let socket: WebSocket;
71 | try {
72 | socket = new WebSocket(webSocketUrl);
73 | } catch (err) {
74 | return;
75 | }
76 |
77 | socket.onopen = () =>
78 | setSocket((serverSocket) => {
79 | if (!serverSocket || serverSocket.id <= id) {
80 | if (serverSocket && serverSocket.socket.readyState === WebSocket.OPEN) {
81 | serverSocket.socket.close(SWITCH_URL_CODE);
82 | }
83 | return { socket, id };
84 | } else {
85 | socket.close(SWITCH_URL_CODE);
86 | return serverSocket;
87 | }
88 | });
89 |
90 | socket.onmessage = (e) => {
91 | const data = JSON.parse(e.data);
92 | if ("activeUsers" in data) {
93 | setActiveUsers(data.activeUsers);
94 | }
95 |
96 | if (data?.type === "failure") {
97 | let signature = data?.signature;
98 | let reason = data?.reason;
99 | if (typeof signature === "string" && typeof reason === "string") {
100 | failureCallbackRef.current(signature, reason);
101 | }
102 | }
103 | };
104 |
105 | socket.onclose = async (event) => {
106 | setSocket((serverSocket) => {
107 | // Socket may have been updated already
108 | if (!serverSocket || serverSocket.id === id) {
109 | // Reconnect if close was not explicit
110 | if (event.code !== SWITCH_URL_CODE) {
111 | console.error("Socket closed, reconnecting...");
112 | // TODO: Re-enable
113 | // reportError(new Error("Socket was closed"), "Socket closed");
114 | setTimeout(() => {
115 | newSocket(
116 | webSocketUrl,
117 | setSocket,
118 | setActiveUsers,
119 | failureCallbackRef
120 | );
121 | }, 5000);
122 | }
123 | return undefined;
124 | }
125 | return serverSocket;
126 | });
127 | };
128 |
129 | socket.onerror = async () => {
130 | socket.close();
131 | };
132 |
133 | return socket;
134 | }
135 |
136 | export function useSocket() {
137 | return React.useContext(SocketContext);
138 | }
139 |
140 | export function useActiveUsers() {
141 | const context = React.useContext(ActiveUsersContext);
142 | if (!context) {
143 | throw new Error(`useActiveUsers must be used within a SocketProvider`);
144 | }
145 |
146 | return context;
147 | }
148 |
149 | export function useFailureCallback() {
150 | const context = React.useContext(FailureCallbackContext);
151 | if (!context) {
152 | throw new Error(`useFailureCallback must be used within a SocketProvider`);
153 | }
154 |
155 | return context;
156 | }
157 |
--------------------------------------------------------------------------------
/client/src/providers/transactions/confirmed.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 |
3 | import { useDispatch, TrackedCommitment } from "./index";
4 | import { useConnection } from "providers/rpc";
5 | import * as Bytes from "utils/bytes";
6 | import { AccountInfo } from "@solana/web3.js";
7 | import { useAccountsState } from "providers/accounts";
8 | import { useClientConfig } from "providers/config";
9 |
10 | // Determine commitment levels to subscribe to. "singleGossip" is used
11 | // to stop tx send retries so it must be returned
12 | export const subscribedCommitments = (
13 | trackedCommitment: TrackedCommitment,
14 | showDebugTable: boolean
15 | ): TrackedCommitment[] => {
16 | if (showDebugTable) return ["confirmed"];
17 | switch (trackedCommitment) {
18 | case "processed": {
19 | return ["processed", "confirmed"];
20 | }
21 | default: {
22 | return ["confirmed"];
23 | }
24 | }
25 | };
26 |
27 | type Props = { children: React.ReactNode };
28 | export function ConfirmedHelper({ children }: Props) {
29 | const dispatch = useDispatch();
30 | const connection = useConnection();
31 | const accounts = useAccountsState().accounts;
32 | const [{ showDebugTable, trackedCommitment }] = useClientConfig();
33 |
34 | React.useEffect(() => {
35 | if (connection === undefined || accounts === undefined) return;
36 | if (showDebugTable) return;
37 |
38 | const commitments = subscribedCommitments(
39 | trackedCommitment,
40 | showDebugTable
41 | );
42 | const partitionCount = accounts.programAccounts.length;
43 |
44 | const accountSubscriptions = accounts.programAccounts.map(
45 | (account, partition) =>
46 | commitments.map((commitment) =>
47 | connection.onAccountChange(
48 | account,
49 | (accountInfo: AccountInfo, { slot }) => {
50 | const ids = new Set(Bytes.programDataToIds(accountInfo.data));
51 | const activeIdPartition = {
52 | ids,
53 | partition,
54 | partitionCount,
55 | };
56 | dispatch({
57 | type: "update",
58 | activeIdPartition,
59 | commitment,
60 | estimatedSlot: slot,
61 | receivedAt: performance.now(),
62 | });
63 | },
64 | commitment
65 | )
66 | )
67 | );
68 |
69 | return () => {
70 | accountSubscriptions.forEach((listeners) => {
71 | listeners.forEach((listener: any) => {
72 | connection.removeAccountChangeListener(listener);
73 | });
74 | });
75 | };
76 | }, [dispatch, connection, accounts, showDebugTable, trackedCommitment]);
77 |
78 | return <>{children}>;
79 | }
80 |
--------------------------------------------------------------------------------
/client/src/providers/transactions/selected.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { useTransactions, TransactionState } from "./index";
3 |
4 | type SetSelected = React.Dispatch>;
5 | type SelectedState = [TransactionState | undefined, SetSelected];
6 | export const SelectedContext = React.createContext(
7 | undefined
8 | );
9 | type ProviderProps = { children: React.ReactNode };
10 | export function SelectedTxProvider({ children }: ProviderProps) {
11 | const transactions = useTransactions();
12 | const [signature, selectSignature] = React.useState(
13 | undefined
14 | );
15 | const [transaction, selectTransaction] = React.useState<
16 | TransactionState | undefined
17 | >(undefined);
18 |
19 | React.useEffect(() => {
20 | selectTransaction(
21 | transactions.find((tx) => tx.details.signature === signature)
22 | );
23 | }, [transactions, signature]);
24 |
25 | const selectedState: SelectedState = React.useMemo(() => {
26 | return [transaction, selectSignature];
27 | }, [transaction]);
28 |
29 | return (
30 |
31 | {children}
32 |
33 | );
34 | }
35 |
36 | export function useSelectedTransaction() {
37 | const state = React.useContext(SelectedContext);
38 | if (!state) {
39 | throw new Error(`useSelectedTx must be used within a TransactionsProvider`);
40 | }
41 | return state[0];
42 | }
43 |
44 | export function useSelectTransaction() {
45 | const state = React.useContext(SelectedContext);
46 | if (!state) {
47 | throw new Error(`useSelectTx must be used within a TransactionsProvider`);
48 | }
49 | return state[1];
50 | }
51 |
--------------------------------------------------------------------------------
/client/src/providers/transactions/tps.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { useCreatedCount } from "./index";
3 |
4 | const TPS_REFRESH_MS = 100;
5 | const TPS_LOOK_BACK = 10;
6 |
7 | export const TpsContext = React.createContext(undefined);
8 | type ProviderProps = { children: React.ReactNode };
9 | export function TpsProvider({ children }: ProviderProps) {
10 | const [tps, setTps] = React.useState(0);
11 | const createdCount = useCreatedCount();
12 | const createdCountRef = React.useRef(0);
13 | createdCountRef.current = createdCount;
14 |
15 | React.useEffect(() => {
16 | const recentCounts: number[] = [];
17 | const timerId = setInterval(() => {
18 | if (createdCountRef.current === 0) {
19 | recentCounts.splice(0);
20 | setTps(0);
21 | return;
22 | }
23 |
24 | recentCounts.push(createdCountRef.current);
25 | while (recentCounts.length - 1 > TPS_LOOK_BACK) {
26 | recentCounts.shift();
27 | }
28 |
29 | const ticksElapsed = recentCounts.length - 1;
30 | if (ticksElapsed <= 0) return;
31 |
32 | const oldTxCount = recentCounts[0];
33 | const latestTxCount = recentCounts[ticksElapsed];
34 | const tps =
35 | (latestTxCount - oldTxCount) / ((TPS_REFRESH_MS / 1000) * ticksElapsed);
36 | setTps(Math.floor(tps));
37 | }, TPS_REFRESH_MS);
38 | return () => {
39 | clearInterval(timerId);
40 | };
41 | }, []);
42 |
43 | return {children} ;
44 | }
45 |
--------------------------------------------------------------------------------
/client/src/providers/wallet.tsx:
--------------------------------------------------------------------------------
1 | import * as React from "react";
2 | import { Keypair } from "@solana/web3.js";
3 | import { getLocalStorageKeypair } from "utils";
4 | import { useHistory, useLocation } from "react-router";
5 |
6 | interface State {
7 | wallet?: Keypair;
8 | selectWallet: (wallet: Keypair | undefined) => void;
9 | }
10 |
11 | const StateContext = React.createContext(undefined);
12 |
13 | type Props = { children: React.ReactNode };
14 | export function WalletProvider({ children }: Props) {
15 | const [wallet, setWallet] = React.useState();
16 |
17 | const history = useHistory();
18 | const location = useLocation();
19 | const selectWallet = React.useCallback(
20 | (keypair: Keypair | undefined) => {
21 | setWallet(keypair);
22 | if (keypair === undefined) {
23 | history.push({ ...location, pathname: "/wallet" });
24 | } else {
25 | history.push({ ...location, pathname: "/start" });
26 | }
27 | },
28 | [history, location]
29 | );
30 |
31 | const state = React.useMemo(
32 | () => ({
33 | wallet,
34 | selectWallet,
35 | }),
36 | [wallet, selectWallet]
37 | );
38 |
39 | return (
40 | {children}
41 | );
42 | }
43 |
44 | export const LOCAL_WALLET = (() => {
45 | return getLocalStorageKeypair("paymentKey");
46 | })();
47 |
48 | export function useWalletState(): State {
49 | const state = React.useContext(StateContext);
50 | if (state === undefined) {
51 | throw new Error(`usePayerState must be used within a WalletProvider`);
52 | }
53 | return state;
54 | }
55 |
--------------------------------------------------------------------------------
/client/src/react-app-env.d.ts:
--------------------------------------------------------------------------------
1 | ///
2 |
--------------------------------------------------------------------------------
/client/src/setupTests.ts:
--------------------------------------------------------------------------------
1 | // jest-dom adds custom jest matchers for asserting on DOM nodes.
2 | // allows you to do things like:
3 | // expect(element).toHaveTextContent(/react/i)
4 | // learn more: https://github.com/testing-library/jest-dom
5 | import "@testing-library/jest-dom/extend-expect";
6 |
--------------------------------------------------------------------------------
/client/src/styles/custom-variables.scss:
--------------------------------------------------------------------------------
1 | $path-to-img: "../../images/dashkit" !default;
2 | $path-to-fonts: "../../fonts" !default;
3 |
4 | $font-family-sans-serif: "Exo 2", sans-serif !default;
5 | $font-family-base: $font-family-sans-serif !default;
6 |
7 | $border-radius: 0.125rem !default;
8 | $gray-700: #888 !default;
9 | $gray-600-dark: #555 !default;
10 | $gray-700-dark: #444 !default;
11 | $gray-750-dark: #333 !default;
12 | $gray-800-dark: #222 !default;
13 | $gray-900-dark: #111 !default;
14 | $black-dark: #000000;
15 |
16 | $black: #000000;
17 | $white: #ffffff;
18 | $green: #00ffad;
19 | $pink: #ff00a8;
20 | $purple: #ac62dd;
21 | $red: #ea134d;
22 | $orange: #e8873f;
23 |
24 | $info: $purple;
25 | $danger: $red;
26 | $warning: $pink;
27 | $primary: $green;
28 |
29 | $card-border-radius: 0.25rem !default;
30 | $card-outline-color: $gray-600-dark !default;
31 | $spacer: 1rem !default;
32 |
33 | $modal-md: 400px !default;
34 |
--------------------------------------------------------------------------------
/client/src/styles/custom.scss:
--------------------------------------------------------------------------------
1 | @import "custom/app";
2 | @import "custom/game";
3 | @import "custom/home";
4 | @import "custom/header";
5 | @import "custom/payment-card";
6 | @import "custom/results";
7 | @import "custom/sidebar";
8 | @import "custom/tx-container";
9 | @import "custom/tx-square";
10 |
11 | .card-body,
12 | .card-header,
13 | .card-footer {
14 | padding: 1rem;
15 | }
16 |
17 | .card-header {
18 | height: 50px;
19 | }
20 |
21 | .card {
22 | margin-bottom: 0.75rem;
23 | }
24 |
25 | .header {
26 | margin-bottom: 1rem;
27 | }
28 |
29 | .header .header-body {
30 | padding: 1rem 0;
31 | }
32 |
33 | .sticky {
34 | position: sticky;
35 | top: 0px;
36 | border-top: 0px solid black !important;
37 | }
38 |
39 | .btn {
40 | letter-spacing: 0.1em;
41 | }
42 |
43 | .btn.input-group {
44 | input {
45 | cursor: pointer;
46 |
47 | &:focus {
48 | cursor: text;
49 | }
50 | }
51 | }
52 |
53 | .border-dark-purple {
54 | border-color: darken($purple, 20%);
55 | }
56 |
57 | .border-top-dark {
58 | border-top: 1px solid $dark;
59 | }
60 |
61 | .btn-pink {
62 | &.btn-secondary {
63 | border-color: darken($pink, 10%);
64 | }
65 |
66 | background-color: $pink;
67 |
68 | @include hover-focus {
69 | background-color: darken($pink, 10%);
70 | }
71 | }
72 |
73 | .cluster-modal {
74 | z-index: 3000;
75 | }
76 |
77 | .header {
78 | z-index: 2000;
79 | }
80 |
81 | .modal-backdrop {
82 | pointer-events: none;
83 | }
84 |
85 | .modal.show {
86 | display: block;
87 | }
88 |
89 | .b-white {
90 | background: $white;
91 | }
92 |
93 | .b-black {
94 | background: $black !important;
95 | }
96 |
97 | .c-pointer {
98 | cursor: pointer;
99 | }
100 |
101 | .touch-action-none {
102 | touch-action: none;
103 | }
104 |
105 | .popover-container {
106 | position: relative;
107 |
108 | .popover {
109 | max-width: fit-content;
110 |
111 | &.right {
112 | left: -1.5rem;
113 | }
114 |
115 | &.bs-popover-bottom {
116 | background: $gray-750-dark;
117 | top: 1.5rem;
118 | }
119 |
120 | .popover-body {
121 | color: white;
122 | min-width: max-content;
123 | }
124 |
125 | .arrow::after {
126 | border-top-color: $gray-750-dark;
127 | border-bottom-color: $gray-750-dark;
128 | }
129 | }
130 | }
131 |
132 | .flex-basis-auto {
133 | flex-basis: auto !important;
134 | }
135 |
136 | .z-auto {
137 | z-index: auto !important;
138 | }
139 |
140 | .debug-row {
141 | cursor: pointer;
142 |
143 | &:hover {
144 | background: #111;
145 | }
146 | }
147 |
148 | .min-width-0 {
149 | min-width: 0;
150 | }
151 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_app.scss:
--------------------------------------------------------------------------------
1 | img.graphic {
2 | position: fixed;
3 | top: 0;
4 | left: 0;
5 | height: 100vh;
6 | z-index: -1;
7 | }
8 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_game.scss:
--------------------------------------------------------------------------------
1 | @include media-breakpoint-up(lg) {
2 | .stat-card {
3 | padding: 0px 10px;
4 |
5 | &:first-child {
6 | padding-left: 12px;
7 | }
8 |
9 | &:last-child {
10 | padding-right: 12px;
11 | }
12 | }
13 | }
14 |
15 | @include media-breakpoint-down(md) {
16 | .stat-card {
17 | &:nth-child(odd) {
18 | padding-right: 6px;
19 | }
20 |
21 | &:nth-child(even) {
22 | padding-left: 6px;
23 | }
24 | }
25 | }
26 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_header.scss:
--------------------------------------------------------------------------------
1 | .solana-header.header {
2 | margin-bottom: 0;
3 | }
4 |
5 | .solana-header .btn-group {
6 | pointer-events: none;
7 | cursor: default;
8 |
9 | &.countdown {
10 | min-width: 60px;
11 | }
12 | }
13 |
14 | .solana-header img.break {
15 | height: 40px;
16 |
17 | @include media-breakpoint-down(md) {
18 | height: 25px;
19 | }
20 | }
21 |
22 | .solana-header img.solana {
23 | height: 15px;
24 |
25 | @include media-breakpoint-down(md) {
26 | height: 10px;
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_home.scss:
--------------------------------------------------------------------------------
1 | .introduction {
2 | line-height: 3rem;
3 | font-weight: 100;
4 | font-size: 1.5rem;
5 |
6 | @include media-breakpoint-down(sm) {
7 | line-height: 2rem;
8 | font-size: 1.2rem;
9 | }
10 | }
11 |
12 | @include media-breakpoint-down(md) {
13 | .home-page {
14 | max-width: 768px;
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_payment-card.scss:
--------------------------------------------------------------------------------
1 | .qr-code {
2 | width: 250px;
3 | height: 250px;
4 | }
5 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_results.scss:
--------------------------------------------------------------------------------
1 | .stats {
2 | overflow-x: scroll;
3 | flex-wrap: nowrap;
4 | max-width: 800px;
5 |
6 | @include media-breakpoint-down(sm) {
7 | padding-bottom: 10px !important;
8 | }
9 |
10 | scrollbar-color: #2a2a2a transparent;
11 | scrollbar-width: thin;
12 |
13 | &::-webkit-scrollbar {
14 | height: 0.4rem;
15 | }
16 |
17 | &::-webkit-scrollbar-track {
18 | background: transparent;
19 | }
20 |
21 | &::-webkit-scrollbar-thumb {
22 | border-radius: 4px;
23 | background: #2a2a2a;
24 | }
25 | }
26 |
27 | .stat-circle {
28 | height: 200px;
29 | width: 200px;
30 | @include media-breakpoint-down(sm) {
31 | height: 175px;
32 | width: 175px;
33 | }
34 | }
35 |
36 | .results-summary h3 {
37 | line-height: 1.5rem;
38 | font-weight: 100;
39 | }
40 |
41 | .donut-content {
42 | &.allow-events {
43 | pointer-events: none;
44 | }
45 |
46 | position: absolute;
47 | top: 0px;
48 | left: 0px;
49 | right: 0px;
50 | bottom: 0px;
51 | display: flex;
52 | flex-direction: column;
53 | justify-content: center;
54 | align-items: center;
55 | padding: 30px;
56 | }
57 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_sidebar.scss:
--------------------------------------------------------------------------------
1 | .page-sidebar {
2 | min-width: 250px;
3 | max-width: 300px;
4 | width: 25vw;
5 | background-color: $gray-800-dark;
6 | border-right: 1px solid $purple;
7 | }
8 |
9 | .sidebar-body {
10 | margin: 1rem;
11 | }
12 |
13 | @media screen and (max-width: 768px) {
14 | .page-sidebar {
15 | display: none;
16 | }
17 | }
18 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_tx-container.scss:
--------------------------------------------------------------------------------
1 | .main {
2 | flex: 1;
3 | }
4 |
5 | .content {
6 | display: flex;
7 | flex-direction: column;
8 | height: 100%;
9 | }
10 |
11 | .debug-wrapper {
12 | overflow-y: auto;
13 | flex-grow: 1;
14 | height: 100px;
15 |
16 | outline: none;
17 | outline-color: transparent;
18 |
19 | scrollbar-color: rgba(0, 255, 173, 0.4) transparent;
20 | scrollbar-width: thin;
21 |
22 | &::-webkit-scrollbar {
23 | width: 0.5rem;
24 | }
25 |
26 | &::-webkit-scrollbar-track {
27 | background: transparent;
28 | }
29 |
30 | &::-webkit-scrollbar-thumb {
31 | background: rgba(0, 255, 173, 0.4);
32 | }
33 | }
34 |
35 | .tx-wrapper {
36 | min-height: 200px;
37 |
38 | h2 {
39 | line-height: 2rem;
40 | }
41 |
42 | .square-container {
43 | position: absolute;
44 | top: 0px;
45 | left: 0px;
46 | right: 0px;
47 | bottom: 0px;
48 | padding-right: 1rem;
49 | overflow-x: hidden;
50 | overflow-y: scroll;
51 | display: flex;
52 | flex-wrap: wrap;
53 | align-content: flex-start;
54 |
55 | outline: none;
56 | outline-color: transparent;
57 |
58 | scrollbar-color: rgba(0, 255, 173, 0.4) transparent;
59 | scrollbar-width: thin;
60 |
61 | &::-webkit-scrollbar {
62 | width: 0.5rem;
63 | }
64 |
65 | &::-webkit-scrollbar-track {
66 | background: transparent;
67 | }
68 |
69 | &::-webkit-scrollbar-thumb {
70 | background: rgba(0, 255, 173, 0.4);
71 | }
72 | }
73 | }
74 |
--------------------------------------------------------------------------------
/client/src/styles/custom/_tx-square.scss:
--------------------------------------------------------------------------------
1 | .square {
2 | width: 0.8rem;
3 | height: 0.8rem;
4 | margin: 0.2rem;
5 | padding: 0;
6 | border-radius: 2px;
7 | font-size: 10px;
8 |
9 | &.legend {
10 | border-radius: 3px;
11 | width: 1.5rem;
12 | height: 1.5rem;
13 | margin: 0.2rem;
14 | }
15 |
16 | @include media-breakpoint-down(sm) {
17 | border-radius: 3px;
18 | width: 1.5rem;
19 | height: 1.5rem;
20 | margin: 0.2rem;
21 | }
22 | }
23 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_alert.scss:
--------------------------------------------------------------------------------
1 | //
2 | // alerts
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | // Allow for a text-decoration since links are the same color as the alert text.
11 |
12 | .alert-link {
13 | text-decoration: $alert-link-text-decoration;
14 | }
15 |
16 | // Color variants
17 | //
18 | // Using Bootstrap's core alert-variant mixin to generate solid background color + yiq colorized text (and making close/links match those colors)
19 |
20 | @each $color, $value in $theme-colors {
21 | .alert-#{$color} {
22 | @include alert-variant(theme-color-level($color, $alert-bg-level), theme-color-level($color, $alert-border-level), color-yiq(theme-color-level($color, $alert-bg-level)));
23 | .close,
24 | .alert-link {
25 | color: color-yiq(theme-color-level($color, $alert-bg-level));
26 | }
27 | }
28 | }
29 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_avatar.scss:
--------------------------------------------------------------------------------
1 | //
2 | // avatar.scss
3 | // Dashkit component
4 | //
5 |
6 | // General
7 |
8 | .avatar {
9 | position: relative;
10 | display: inline-block;
11 | width: $avatar-size-base;
12 | height: $avatar-size-base;
13 | font-size: $avatar-size-base / 3;
14 |
15 | // Loads mask images so they don't lag on hover
16 |
17 | &:after {
18 | content: '';
19 | position: absolute;
20 | width: 0;
21 | height: 0;
22 | background-image: url(#{$path-to-img}/masks/avatar-status.svg),
23 | url(#{$path-to-img}/masks/avatar-group.svg),
24 | url(#{$path-to-img}/masks/avatar-group-hover.svg),
25 | url(#{$path-to-img}/masks/avatar-group-hover-last.svg);
26 | }
27 | }
28 |
29 | .avatar-img {
30 | width: 100%;
31 | height: 100%;
32 | object-fit: cover;
33 | }
34 |
35 | .avatar-title {
36 | display: flex;
37 | align-items: center;
38 | justify-content: center;
39 | width: 100%;
40 | height: 100%;
41 | line-height: 0;
42 | background-color: $avatar-title-bg;
43 | color: $avatar-title-color;
44 | }
45 |
46 |
47 | // Status
48 |
49 | .avatar-online,
50 | .avatar-offline {
51 |
52 | &::before {
53 | content: '';
54 | position: absolute;
55 | bottom: 5%;
56 | right: 5%;
57 | width: 20%;
58 | height: 20%;
59 | border-radius: 50%;
60 | }
61 |
62 | .avatar-img {
63 | mask-image: url(#{$path-to-img}/masks/avatar-status.svg);
64 | mask-size: 100% 100%;
65 | }
66 | }
67 |
68 | .avatar-online::before {
69 | background-color: $success;
70 | }
71 |
72 | .avatar-offline::before {
73 | background-color: $gray-500;
74 | }
75 |
76 |
77 | // Sizing
78 |
79 | .avatar-xs {
80 | width: $avatar-size-xs;
81 | height: $avatar-size-xs;
82 | font-size: $avatar-size-xs / 3;
83 | }
84 |
85 | .avatar-sm {
86 | width: $avatar-size-sm;
87 | height: $avatar-size-sm;
88 | font-size: $avatar-size-sm / 3;
89 | }
90 |
91 | .avatar-lg {
92 | width: $avatar-size-lg;
93 | height: $avatar-size-lg;
94 | font-size: $avatar-size-lg / 3;
95 | }
96 |
97 | .avatar-xl {
98 | width: $avatar-size-xl;
99 | height: $avatar-size-xl;
100 | font-size: $avatar-size-xl / 3;
101 | }
102 |
103 | .avatar-xxl {
104 | width: $avatar-size-xl;
105 | height: $avatar-size-xl;
106 | font-size: $avatar-size-xl / 3;
107 |
108 | @include media-breakpoint-up(md) {
109 | width: $avatar-size-xxl;
110 | height: $avatar-size-xxl;
111 | font-size: $avatar-size-xxl / 3;
112 | }
113 | }
114 |
115 |
116 | // Ratio
117 |
118 | .avatar.avatar-4by3 {
119 | width: $avatar-size-base * 4 / 3;
120 | }
121 |
122 | .avatar-xs.avatar-4by3 {
123 | width: $avatar-size-xs * 4 / 3;
124 | }
125 |
126 | .avatar-sm.avatar-4by3 {
127 | width: $avatar-size-sm * 4 / 3;
128 | }
129 |
130 | .avatar-lg.avatar-4by3 {
131 | width: $avatar-size-lg * 4 / 3;
132 | }
133 |
134 | .avatar-xl.avatar-4by3 {
135 | width: $avatar-size-xl * 4 / 3;
136 | }
137 |
138 | .avatar-xxl.avatar-4by3 {
139 | width: $avatar-size-xxl * 4 / 3;
140 | }
141 |
142 |
143 | // Group
144 |
145 | .avatar-group {
146 | display: inline-flex;
147 |
148 | // Shift every next avatar left
149 |
150 | .avatar + .avatar {
151 | margin-left: -$avatar-size-base / 4;
152 | }
153 |
154 | .avatar-xs + .avatar-xs {
155 | margin-left: -$avatar-size-xs / 4;
156 | }
157 |
158 | .avatar-sm + .avatar-sm {
159 | margin-left: -$avatar-size-sm / 4;
160 | }
161 |
162 | .avatar-lg + .avatar-lg {
163 | margin-left: -$avatar-size-lg / 4;
164 | }
165 |
166 | .avatar-xl + .avatar-xl {
167 | margin-left: -$avatar-size-xl / 4;
168 | }
169 |
170 | .avatar-xxl + .avatar-xxl {
171 | margin-left: -$avatar-size-xxl / 4;
172 | }
173 |
174 | // Add some spacing between avatars
175 |
176 | .avatar:not(:last-child) {
177 | mask-image: url(#{$path-to-img}/masks/avatar-group.svg);
178 | mask-size: 100% 100%;
179 | }
180 |
181 | // Bring an avatar to front on hover
182 |
183 | .avatar:hover {
184 | mask-image: none;
185 | z-index: 1;
186 |
187 | + .avatar {
188 | mask-image: url(#{$path-to-img}/masks/avatar-group-hover.svg);
189 | mask-size: 100% 100%;
190 |
191 | &:last-child {
192 | mask-image: url(#{$path-to-img}/masks/avatar-group-hover-last.svg);
193 | }
194 | }
195 | }
196 |
197 | }
198 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_badge.scss:
--------------------------------------------------------------------------------
1 | //
2 | // badge.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | .badge {
11 | vertical-align: middle;
12 | }
13 |
14 | // Quick fix for badges in buttons
15 | .btn .badge {
16 | top: -2px;
17 | }
18 |
19 | //
20 | // Dashkit =====================================
21 | //
22 |
23 | // Creates the "soft" badge variant
24 | @each $color, $value in $theme-colors {
25 | .badge-soft-#{$color} {
26 | @include badge-variant-soft(theme-color-level($color, $badge-soft-bg-level), $value);
27 | }
28 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_breadcrumb.scss:
--------------------------------------------------------------------------------
1 | //
2 | // breadcrumb.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | .breadcrumb-item {
11 | // The separator between breadcrumbs
12 | + .breadcrumb-item::before {
13 | width: .3rem;
14 | height: .6rem;
15 | margin-right: $breadcrumb-item-padding;
16 | -webkit-mask: url() no-repeat 50% 50%;
17 | mask: url() no-repeat 50% 50%;
18 | -webkit-mask-size: contain;
19 | mask-size: contain;
20 | background: $breadcrumb-divider-color;
21 | }
22 | }
23 |
24 |
25 | //
26 | // Dashkit =====================================
27 | //
28 |
29 | // Small
30 | //
31 | // Reduces font size
32 |
33 | .breadcrumb-sm {
34 | font-size: $breadcrumb-font-size-sm;
35 | }
36 |
37 |
38 | // Overflow
39 | //
40 | // Allows the breadcrumb to be overflown horizontally
41 |
42 | .breadcrumb-overflow {
43 | display: flex;
44 | flex-direction: row;
45 | flex-wrap: nowrap;
46 | overflow-x: auto;
47 |
48 | &::-webkit-scrollbar {
49 | display: none;
50 | }
51 | }
52 |
53 | .breadcrumb-overflow .breadcrumb-item {
54 | white-space: nowrap;
55 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_buttons.scss:
--------------------------------------------------------------------------------
1 | //
2 | // buttons.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 |
7 | //
8 | // Dashkit ===================================
9 | //
10 |
11 | // Button white
12 |
13 | .btn-white {
14 | border-color: $gray-300;
15 |
16 | @include hover-focus {
17 | background-color: $gray-100;
18 | border-color: $gray-400;
19 | }
20 | }
21 |
22 | .btn-group-toggle .btn-white:not(:disabled):not(.disabled):active,
23 | .btn-group-toggle .btn-white:not(:disabled):not(.disabled).active {
24 | background-color: $input-bg;
25 | border-color: $input-focus-border-color;
26 | color: $primary;
27 | }
28 |
29 | .btn-group-toggle .btn-white:focus,
30 | .btn-group-toggle .btn-white.focus {
31 | box-shadow: none;
32 | }
33 |
34 |
35 | // Button outline secondary
36 |
37 | .btn-outline-secondary {
38 | &:not(:hover):not([aria-expanded="true"]):not([aria-pressed="true"]){
39 | border-color: $gray-400;
40 | }
41 | }
42 |
43 |
44 | // Button rounded
45 | //
46 | // Creates circle button variations
47 |
48 | .btn-rounded-circle {
49 | width: calc(1em * #{$btn-line-height} + #{$input-btn-padding-y * 2 } + #{$btn-border-width} * 2);
50 | padding-left: 0;
51 | padding-right: 0;
52 | border-radius: 50%;
53 | }
54 | .btn-rounded-circle.btn-lg {
55 | width: calc(1em * #{$btn-line-height-lg} + #{$input-btn-padding-y-lg * 2 } + #{$btn-border-width} * 2);
56 | }
57 | .btn-rounded-circle.btn-sm {
58 | width: calc(1em * #{$btn-line-height-sm} + #{$input-btn-padding-y-sm * 2 } + #{$btn-border-width} * 2);
59 | }
60 |
61 |
62 | // Button group
63 | //
64 | // Prevent buttons from jittering on hover
65 |
66 | .btn-group .btn + .btn {
67 | margin-left: 0;
68 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_chart.scss:
--------------------------------------------------------------------------------
1 | //
2 | // chart.scss
3 | // Dashkit component
4 | //
5 |
6 | // Chart
7 | //
8 | // General styles
9 |
10 | .chart {
11 | position: relative;
12 | height: $chart-height;
13 | }
14 |
15 | .chart.chart-appended {
16 | height: calc(#{$chart-height} - #{$chart-legend-height});
17 | }
18 |
19 | .chart-sm {
20 | height: $chart-height-sm;
21 | }
22 |
23 | .chart-sm.chart-appended {
24 | height: calc(#{$chart-height-sm} - #{$chart-legend-height});
25 | }
26 |
27 |
28 | // Sparkline
29 |
30 | .chart-sparkline {
31 | width: $chart-sparkline-width;
32 | height: $chart-sparkline-height;
33 | }
34 |
35 |
36 | // Legend
37 | //
38 | // Custom legend
39 |
40 | .chart-legend {
41 | display: flex;
42 | justify-content: center;
43 | margin-top: $chart-legend-margin-top;
44 | font-size: $chart-legend-font-size;
45 | text-align: center;
46 | color: $chart-legend-color;
47 | }
48 |
49 | .chart-legend-item {
50 | display: inline-flex;
51 | align-items: center;
52 |
53 | + .chart-legend-item {
54 | margin-left: 1rem;
55 | }
56 | }
57 |
58 | .chart-legend-indicator {
59 | display: inline-block;
60 | width: .5rem;
61 | height: .5rem;
62 | margin-right: 0.375rem;
63 | border-radius: 50%;
64 | }
65 |
66 |
67 | // Tooltip
68 | //
69 | // Custom tooltip
70 |
71 | #chart-tooltip {
72 | z-index: 0;
73 | }
74 |
75 | #chart-tooltip .arrow {
76 | top: 100%;
77 | left: 50%;
78 | transform: translateX(-50%) translateX(-.5rem);
79 | }
80 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_checklist.scss:
--------------------------------------------------------------------------------
1 | //
2 | // checklist.scss
3 | // Dashkit component
4 | //
5 |
6 | .checklist {
7 | outline: none;
8 | }
9 |
10 | .checklist-control {
11 | display: flex;
12 | flex-wrap: nowrap;
13 | outline: none;
14 | user-select: none;
15 | }
16 |
17 | .checklist-control .custom-control-input:checked ~ .custom-control-caption {
18 | text-decoration: line-through;
19 | color: $checklist-control-checked-color;
20 | }
21 |
22 | .checklist-control + .checklist-control {
23 | margin-top: $checklist-control-spacer;
24 | }
25 |
26 | .checklist-control:first-child[style*="display: none"] + .checklist-control {
27 | margin-top: 0;
28 | }
29 |
30 | .checklist-control.draggable-mirror {
31 | z-index: $zindex-fixed;
32 | }
33 |
34 | .checklist-control.draggable-source--is-dragging {
35 | opacity: .2;
36 | }
37 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_close.scss:
--------------------------------------------------------------------------------
1 | //
2 | // close.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | // Small
7 |
8 | .close-sm {
9 | font-size: $close-font-size-sm;
10 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_code.scss:
--------------------------------------------------------------------------------
1 | //
2 | // code.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Dashkit =================================
8 | //
9 |
10 | // Highlight
11 | //
12 | // Hightlight.js overrides
13 |
14 | .highlight {
15 | padding: 0;
16 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_comment.scss:
--------------------------------------------------------------------------------
1 | //
2 | // comment.scss
3 | // Dashkit component
4 | //
5 |
6 | // Comment
7 | //
8 | // General styles
9 |
10 | .comment {
11 | margin-bottom: $comment-margin-bottom;
12 | }
13 |
14 | .comment-body {
15 | display: inline-block;
16 | padding: $comment-body-padding-y $comment-body-padding-x;
17 | background-color: $comment-body-bg;
18 | border-radius: $comment-body-border-radius;
19 | }
20 |
21 | .comment-time {
22 | display: block;
23 | margin-bottom: $comment-time-margin-bottom;
24 | font-size: $comment-time-font-size;
25 | color: $comment-time-color;
26 | }
27 |
28 | .comment-text {
29 | font-size: $comment-body-font-size;
30 | }
31 |
32 | .comment-text:last-child {
33 | margin-bottom: 0;
34 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_custom-forms.scss:
--------------------------------------------------------------------------------
1 | //
2 | // custom-forms.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides ===================================
8 | //
9 |
10 | // Switch
11 |
12 | .custom-switch {
13 | min-height: $custom-switch-height;
14 |
15 | .custom-control-label {
16 |
17 | &::before {
18 | top: 0;
19 | height: $custom-switch-height;
20 | border-radius: $custom-switch-height / 2;
21 | }
22 |
23 | &::after {
24 | top: $custom-switch-spacing;
25 | left: $custom-switch-spacing - $custom-control-gutter - $custom-switch-width;
26 | background-color: $custom-switch-indicator-bg;
27 | }
28 | }
29 |
30 | .custom-control-input:checked ~ .custom-control-label {
31 |
32 | &::after {
33 | background-color: $custom-switch-indicator-active-bg;
34 | transform: translateX($custom-switch-width - $custom-switch-spacing * 2 - $custom-switch-indicator-size);
35 | }
36 | }
37 | }
38 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_dashkit.scss:
--------------------------------------------------------------------------------
1 | // Extended Bootstrap components
2 |
3 | @import 'mixins';
4 | @import 'alert';
5 | @import 'badge';
6 | @import 'breadcrumb';
7 | @import 'buttons';
8 | @import 'card';
9 | @import 'close';
10 | @import 'code';
11 | @import 'custom-forms';
12 | @import 'dropdowns';
13 | @import 'forms';
14 | @import 'jumbotron';
15 | @import 'list-group';
16 | @import 'modal';
17 | @import 'nav';
18 | @import 'navbar';
19 | @import 'popover';
20 | @import 'progress';
21 | @import 'reboot';
22 | @import 'root';
23 | @import 'tables';
24 | @import 'toasts';
25 | @import 'type';
26 | @import 'utilities';
27 |
28 | // Dashkit only components
29 |
30 | @import 'avatar';
31 | @import 'chart';
32 | @import 'comment';
33 | @import 'checklist';
34 | @import 'header';
35 | @import 'icon';
36 | @import 'kanban';
37 | @import 'main-content';
38 | @import 'vendors';
39 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_dropdowns.scss:
--------------------------------------------------------------------------------
1 | //
2 | // dropdowns.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap overrides ==================================
8 | //
9 |
10 | // Dropdown arrows
11 | //
12 | // Replace original arrows with Feather icons
13 |
14 | .dropdown-toggle {
15 |
16 | &::after {
17 | width: auto;
18 | height: auto;
19 | border: none !important;
20 | vertical-align: middle;
21 | font-family: 'Feather';
22 | }
23 |
24 | &::after {
25 | content: '\e92e';
26 | }
27 | }
28 |
29 | .dropup > .dropdown-toggle {
30 |
31 | &::after {
32 | content: '\e931';
33 | }
34 | }
35 |
36 | .dropright > .dropdown-toggle {
37 |
38 | &::after {
39 | content: '\e930';
40 | }
41 | }
42 |
43 | .dropleft > .dropdown-toggle {
44 |
45 | &::before {
46 | content: '\e92f';
47 | width: auto;
48 | height: auto;
49 | border: none !important;
50 | vertical-align: middle;
51 | font-family: 'Feather';
52 | }
53 | }
54 |
55 |
56 | // Dropdown toggle
57 | //
58 | // Right align arrows
59 |
60 | .dropdown-item.dropdown-toggle {
61 | display: flex;
62 | justify-content: space-between;
63 | }
64 |
65 |
66 | // Dropdown menu animation
67 | //
68 | // Animate dropdown menu appearance
69 |
70 | .dropdown-menu {
71 | animation: dropdownMenu .15s;
72 | }
73 |
74 | @keyframes dropdownMenu {
75 | from {
76 | opacity: 0;
77 | }
78 |
79 | to {
80 | opacity: 1;
81 | }
82 | }
83 |
84 |
85 |
86 | //
87 | // Dashkit ===================================
88 | //
89 |
90 |
91 | // Dropdown ellipses
92 | //
93 | // Styles the ellipses icon and removes the dropdown arrow
94 |
95 | .dropdown-ellipses {
96 | font-size: $font-size-lg;
97 | color: $gray-400;
98 | }
99 |
100 | .dropdown-ellipses::after {
101 | display: none;
102 | }
103 |
104 |
105 | // Dropdown card
106 | //
107 | // Makes the dropdown menu act like a card
108 |
109 | .dropdown-menu-card {
110 | min-width: $dropdown-card-min-width;
111 | padding-top: 0;
112 | padding-bottom: 0;
113 | background-color: $card-bg;
114 | border-color: $dropdown-card-border-color;
115 |
116 | .card-header {
117 | min-height: $dropdown-card-header-min-height;
118 | }
119 |
120 | .card-body {
121 | max-height: $dropdown-card-body-max-height;
122 | overflow-y: auto;
123 | }
124 | }
125 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_header.scss:
--------------------------------------------------------------------------------
1 | //
2 | // header.scss
3 | // Dashkit component
4 | //
5 |
6 | // Header
7 | //
8 | // General styles
9 |
10 | .header {
11 | margin-bottom: $header-margin-bottom;
12 | }
13 |
14 | .header-img-top {
15 | width: 100%; height: auto;
16 | }
17 |
18 | .header-body {
19 | padding-top: $header-spacing-y;
20 | padding-bottom: $header-spacing-y;
21 | border-bottom: $header-body-border-width solid $header-body-border-color;
22 | }
23 |
24 | .header.bg-dark .header-body,
25 | .header.bg-hero .header-body {
26 | border-bottom-color: $header-body-border-color-dark;
27 | }
28 |
29 | .header-footer {
30 | padding-top: $header-spacing-y;
31 | padding-bottom: $header-spacing-y;
32 | }
33 |
34 | .header-pretitle {
35 | text-transform: uppercase;
36 | letter-spacing: .08em;
37 | color: $text-muted;
38 | }
39 |
40 | .header-title {
41 | margin-bottom: 0;
42 | }
43 |
44 | .header-subtitle {
45 | margin-top: map-get($spacers, 2);
46 | margin-bottom: 0;
47 | color: $text-muted;
48 | }
49 |
50 | .header-tabs {
51 | margin-bottom: -$header-spacing-y;
52 | border-bottom-width: 0;
53 |
54 | .nav-link {
55 | padding-top: $header-spacing-y;
56 | padding-bottom: $header-spacing-y;
57 | }
58 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_icon.scss:
--------------------------------------------------------------------------------
1 | //
2 | // icon.scss
3 | // Dashkit component
4 | //
5 |
6 | // Icon
7 | //
8 | // General styles
9 |
10 | .icon {
11 | display: inline-block;
12 |
13 | // Feather icon
14 |
15 | > .fe {
16 | display: block;
17 | min-width: 1em * $line-height-base;
18 | min-height: 1em * $line-height-base;
19 | text-align: center;
20 | font-size: $font-size-lg;
21 | }
22 |
23 | // Active state
24 |
25 | &.active {
26 | position: relative;
27 |
28 | // Feather icon
29 |
30 | > .fe {
31 | mask-image: url(#{$path-to-img}/masks/icon-status.svg);
32 | mask-size: 100% 100%;
33 | }
34 |
35 | // Indicator
36 |
37 | &::after {
38 | content: "";
39 | position: absolute;
40 | top: 10%; right: 20%;
41 | width: 20%; height: 20%;
42 | border-radius: 50%;
43 | background-color: $primary;
44 | }
45 | }
46 | }
47 |
48 | // Feather icons
49 | //
50 | // Fixes icon / font vertical alignment issue
51 |
52 | .fe {
53 | line-height: inherit;
54 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_jumbotron.scss:
--------------------------------------------------------------------------------
1 | //
2 | // jumbotron.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | .jumbotron {
11 | padding: ($jumbotron-padding / 2);
12 | @include media-breakpoint-up(sm) {
13 | padding: $jumbotron-padding;
14 | }
15 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_kanban.scss:
--------------------------------------------------------------------------------
1 | //
2 | // kanban.scss
3 | // Dashkit component
4 | //
5 |
6 | // Container
7 |
8 | .container-fluid.kanban-container {
9 | min-height: calc(100vh - 129px);
10 | }
11 |
12 | .container.kanban-container {
13 | min-height: calc(100vh - 129px - 69px);
14 | }
15 |
16 | .kanban-container {
17 | overflow-x: scroll;
18 | -webkit-overflow-scrolling: touch;
19 | }
20 |
21 | .kanban-container > .row {
22 | flex-wrap: nowrap;
23 | }
24 |
25 | .kanban-container > .row > [class*="col"] {
26 | max-width: $kanban-col-width;
27 | }
28 |
29 |
30 | // Category
31 |
32 | .kanban-category {
33 | min-height: 1rem;
34 | }
35 |
36 |
37 | // Item
38 |
39 | .kanban-item {
40 | outline: none;
41 | user-select: none;
42 | }
43 |
44 | .kanban-item.draggable-source--is-dragging {
45 | opacity: .2;
46 | }
47 |
48 | .kanban-item.draggable-mirror {
49 | z-index: $zindex-fixed;
50 | }
51 |
52 | .card-body .kanban-item.draggable-mirror > .card {
53 | transform: rotateZ(-3deg);
54 | }
55 |
56 |
57 | // Card
58 |
59 | .kanban-item > .card[data-toggle="modal"] {
60 | cursor: pointer;
61 | }
62 |
63 |
64 | // Add form
65 |
66 | .kanban-add-form .form-control[data-toggle="flatpickr"] {
67 | width: 12ch; // there is no CSS way to set input's width to auto so hardcoding this value
68 | }
69 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_list-group.scss:
--------------------------------------------------------------------------------
1 | //
2 | // list-group.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | // Contextual variants
11 | //
12 | // Changing the Bootstrap color modifier classes to be full opacity background with yiq calculated font color
13 |
14 | @each $color, $value in $theme-colors {
15 | @include list-group-item-variant($color, $value, color-yiq($value));
16 | }
17 |
18 |
19 | // List group large
20 |
21 | .list-group-lg .list-group-item {
22 | padding-top: $list-group-item-padding-y-lg;
23 | padding-bottom: $list-group-item-padding-y-lg;
24 | }
25 |
26 |
27 | // List group flush
28 |
29 | .list-group-flush > .list-group-item {
30 | padding-left: 0;
31 | padding-right: 0;
32 | }
33 |
34 |
35 | //
36 | // Dashkit ===================================
37 | //
38 |
39 |
40 | // Activity
41 |
42 | .list-group-activity .list-group-item {
43 | border: 0;
44 | }
45 |
46 | .list-group-activity .list-group-item:not(:last-child)::before {
47 | content: '';
48 | position: absolute;
49 | top: $list-group-item-padding-y;
50 | left: $avatar-size-sm / 2;
51 | height: 100%;
52 | border-left: $border-width solid $border-color;
53 | }
54 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_main-content.scss:
--------------------------------------------------------------------------------
1 | //
2 | // main-content.scss
3 | // Dashkit component
4 | //
5 |
6 | // Main content
7 | //
8 | // General styles
9 |
10 | .navbar-vertical:not([style*="display: none"]) ~ .main-content,
11 | .navbar-vertical-sm:not([style*="display: none"]) ~ .main-content {
12 |
13 | .container,
14 | .container-fluid {
15 |
16 | @include media-breakpoint-up(md) {
17 | padding-left: ($main-content-padding-x + $grid-gutter-width / 2) !important;
18 | padding-right: ($main-content-padding-x + $grid-gutter-width / 2) !important;
19 | }
20 | }
21 | }
22 |
23 |
24 | // Main content offset
25 | //
26 | // Offsets the main content depending on the sidebar positioning
27 |
28 | .navbar-vertical.navbar-expand {
29 |
30 | @each $breakpoint, $value in $grid-breakpoints {
31 | &-#{$breakpoint} {
32 | @include media-breakpoint-up(#{$breakpoint}) {
33 |
34 | // Left
35 |
36 | &.fixed-left:not([style*="display: none"]) ~ .main-content {
37 | margin-left: $navbar-vertical-width;
38 | }
39 |
40 | // Right
41 |
42 | &.fixed-right:not([style*="display: none"]) ~ .main-content {
43 | margin-right: $navbar-vertical-width;
44 | }
45 | }
46 | }
47 | }
48 | }
49 |
50 | .navbar-vertical-sm.navbar-expand {
51 |
52 | @each $breakpoint, $value in $grid-breakpoints {
53 | &-#{$breakpoint} {
54 | @include media-breakpoint-up(#{$breakpoint}) {
55 |
56 | // Left
57 |
58 | &.fixed-left:not([style*="display: none"]) ~ .main-content {
59 | margin-left: $navbar-vertical-width-sm;
60 | }
61 |
62 | // Right
63 |
64 | &.fixed-right:not([style*="display: none"]) ~ .main-content {
65 | margin-right: $navbar-vertical-width-sm;
66 | }
67 | }
68 | }
69 | }
70 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_mixins.scss:
--------------------------------------------------------------------------------
1 | // Mixins
2 | //
3 | //
4 |
5 | // Utilities
6 | @import "mixins/breakpoints";
7 | @import "mixins/badge";
8 |
9 | // Components
10 | // ...
11 |
12 | // Skins
13 | // ...
14 |
15 | // Layout
16 | // ...
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_modal.scss:
--------------------------------------------------------------------------------
1 | //
2 | // modal.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =======================
8 | //
9 |
10 | .modal-dialog {
11 |
12 | // When fading in the modal, animate it to slide down
13 | .modal.fade & {
14 | transform: translate(0, -150px);
15 | }
16 |
17 | .modal.show & {
18 | transform: translate(0, 0);
19 | }
20 | }
21 |
22 | .modal-header .close {
23 | margin: -1.5rem -1.5rem -1.5rem auto;
24 | }
25 |
26 |
27 | //
28 | // Dashkit ===================================
29 | //
30 |
31 | // Modal dialog vertical
32 | //
33 | // Creates a vertically aligned version of the modal dialog
34 |
35 | .modal-dialog-vertical {
36 | height: 100%;
37 | max-width: $modal-dialog-vertical-width;
38 | margin: 0;
39 |
40 | .modal-content {
41 | height: inherit;
42 | border-width: 0 $modal-content-border-width 0 0;
43 | border-radius: 0;
44 | }
45 |
46 | .modal-header {
47 | border-radius: inherit;
48 | }
49 |
50 | .modal-body {
51 | height: inherit;
52 | overflow-y: auto;
53 | }
54 | }
55 |
56 | .modal.fade .modal-dialog-vertical {
57 | transform: translateX(-100%);
58 | }
59 |
60 | .modal.show .modal-dialog-vertical {
61 | transform: translateX(0);
62 | }
63 |
64 |
65 | // Positioning
66 |
67 | .modal.fixed-right {
68 | padding-right: 0 !important;
69 | }
70 |
71 | .modal.fixed-right .modal-dialog-vertical {
72 | margin-left: auto;
73 | }
74 |
75 | .modal.fixed-right.fade .modal-dialog-vertical {
76 | transform: translateX(100%);
77 | }
78 |
79 | .modal.fixed-right.show .modal-dialog-vertical {
80 | transform: translateX(0);
81 | }
82 |
83 |
84 | // Modal card
85 |
86 | .modal-card {
87 | margin-bottom: 0;
88 |
89 | .card-body {
90 | max-height: $modal-card-body-max-height;
91 | overflow-y: auto;
92 | }
93 | }
94 |
95 |
96 | // Modal tabs
97 |
98 | .modal-header-tabs {
99 | margin-top: -$modal-header-padding-y;
100 | margin-bottom: calc(-#{$modal-header-padding-y} - #{$border-width});
101 | }
102 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_nav.scss:
--------------------------------------------------------------------------------
1 | //
2 | // nav.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | // Changing nav tabs to be bottom highlight style
11 |
12 | .nav-tabs {
13 |
14 | .nav-item {
15 | margin-left: $nav-tabs-link-margin-x;
16 | margin-right: $nav-tabs-link-margin-x;
17 | }
18 |
19 | .nav-link {
20 | padding: $nav-tabs-link-padding-y 0;
21 | border-bottom: $nav-tabs-link-active-border-width solid transparent;
22 | border-left-width: 0;
23 | border-right-width: 0;
24 | border-top-width: 0;
25 |
26 | &:not(.active) {
27 | color: $gray-600;
28 |
29 | &:hover {
30 | color: $gray-700;
31 | }
32 | }
33 |
34 | }
35 |
36 | .nav-item:first-child {
37 | margin-left: 0;
38 | }
39 |
40 | .nav-item:last-child {
41 | margin-right: 0;
42 | }
43 |
44 | // Removes the primary color underline from dropdowns in .nav-tabs
45 | .nav-item.show .nav-link {
46 | border-color: transparent;
47 | }
48 |
49 | }
50 |
51 |
52 | //
53 | // Dashkit =====================================
54 | //
55 |
56 | // Nav overflow
57 | //
58 | // Allow links to overflow and make horizontally scrollable
59 |
60 | .nav-overflow {
61 | display: flex;
62 | flex-wrap: nowrap;
63 | overflow-x: auto;
64 | padding-bottom: 1px; // to prevent active links border bottom from hiding
65 |
66 | // Hide scrollbar
67 |
68 | &::-webkit-scrollbar {
69 | display: none;
70 | }
71 | }
72 |
73 |
74 | // Creates a small version of the .nav-tabs
75 |
76 | .nav-tabs-sm {
77 | font-size: $nav-tabs-sm-font-size;
78 |
79 | .nav-item {
80 | margin-left: $nav-tabs-sm-link-margin-x;
81 | margin-right: $nav-tabs-sm-link-margin-x;
82 | }
83 |
84 | .nav-link {
85 | // Calculates the exact padding necessary to vertically fill the .card-header
86 | padding-top: (($font-size-base / $nav-tabs-sm-font-size) * $nav-tabs-link-padding-y);
87 | padding-bottom: (($font-size-base / $nav-tabs-sm-font-size) * $nav-tabs-link-padding-y);
88 | }
89 | }
90 |
91 | // Creates a small version of the .nab
92 |
93 | .nav-sm {
94 |
95 | .nav-link {
96 | font-size: $font-size-sm;
97 | }
98 | }
99 |
100 |
101 | // Nav + button group
102 | //
103 | // Change the look of .btn-white when .active
104 |
105 | .nav.btn-group {
106 |
107 | .btn-white.active {
108 | background-color: $primary;
109 | border-color: $primary;
110 | color: $white;
111 | }
112 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_popover.scss:
--------------------------------------------------------------------------------
1 | //
2 | // popover.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | .popover {
11 | padding: $popover-padding-y $popover-padding-x;
12 |
13 | &:hover {
14 | visibility: visible !important;
15 | }
16 | }
17 |
18 | .popover-header {
19 | margin-bottom: $popover-header-margin-bottom;
20 | border-bottom: 0;
21 | }
22 |
23 | .popover-body-label {
24 | margin-left: .25rem;
25 | }
26 |
27 | .popover-body-value {
28 | margin-left: .25rem;
29 | }
30 |
31 | .popover-body-indicator {
32 | display: inline-block;
33 | width: .5rem;
34 | height: .5rem;
35 | border-radius: 50%;
36 | }
37 |
38 |
39 | // Large
40 |
41 | .popover-lg {
42 | max-width: $popover-lg-max-width;
43 | }
44 |
45 |
46 | // Dark
47 |
48 | .popover-dark {
49 | background-color: $popover-dark-bg;
50 | border-color: $popover-dark-border-color;
51 | }
52 |
53 | .popover-dark .popover-header {
54 | font-weight: $font-weight-normal;
55 | background-color: $popover-dark-header-bg;
56 | color: $popover-dark-header-color;
57 | }
58 |
59 | .popover-dark.bs-popover-top .arrow {
60 |
61 | &::before {
62 | border-top-color: $popover-dark-border-color;
63 | }
64 |
65 | &::after {
66 | border-top-color: $popover-dark-bg;
67 | }
68 | }
69 |
70 | .popover-dark.bs-popover-right .arrow {
71 |
72 | &::before {
73 | border-right-color: $popover-dark-border-color;
74 | }
75 |
76 | &::after {
77 | border-right-color: $popover-dark-bg;
78 | }
79 | }
80 |
81 | .popover-dark.bs-popover-bottom .arrow {
82 |
83 | &::before {
84 | border-bottom-color: $popover-dark-border-color;
85 | }
86 |
87 | &::after {
88 | border-bottom-color: $popover-dark-bg;
89 | }
90 | }
91 |
92 | .popover-dark.bs-popover-left .arrow {
93 |
94 | &::before {
95 | border-left-color: $popover-dark-border-color;
96 | }
97 |
98 | &::after {
99 | border-left-color: $popover-dark-bg;
100 | }
101 | }
102 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_progress.scss:
--------------------------------------------------------------------------------
1 | //
2 | // progress.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | // Rounds the progress bar, even for "multiple bar" progress bars
11 | .progress-bar:first-child {
12 | border-top-left-radius: $progress-border-radius;
13 | border-bottom-left-radius: $progress-border-radius;
14 | }
15 | .progress-bar:last-child {
16 | border-top-right-radius: $progress-border-radius;
17 | border-bottom-right-radius: $progress-border-radius;
18 | }
19 |
20 |
21 | //
22 | // Dashkit ===================================
23 | //
24 |
25 | .progress-sm {
26 | height: $progress-height-sm;
27 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_reboot.scss:
--------------------------------------------------------------------------------
1 | //
2 | // reboot.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Remove the cancel buttons in Chrome and Safari on macOS.
8 | //
9 |
10 | [type="search"]::-webkit-search-cancel-button {
11 | -webkit-appearance: none;
12 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_root.scss:
--------------------------------------------------------------------------------
1 | //
2 | // root.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | html {
7 | height: 100%;
8 | }
9 |
10 | body {
11 | min-height: 100%;
12 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_tables.scss:
--------------------------------------------------------------------------------
1 | //
2 | // tables.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | .table {
11 |
12 | thead th {
13 | background-color: $table-head-bg;
14 | text-transform: uppercase;
15 | font-size: $font-size-sm;
16 | font-weight: $font-weight-bold;
17 | letter-spacing: .08em;
18 | color: $table-head-color;
19 | border-bottom-width: $table-border-width;
20 | }
21 |
22 | thead th, tbody th, tbody td {
23 | vertical-align: middle;
24 | }
25 | }
26 |
27 | .table-sm {
28 | font-size: $font-size-sm;
29 |
30 | thead th {
31 | font-size: $font-size-xs;
32 | }
33 | }
34 |
35 |
36 | //
37 | // Dashkit =====================================
38 | //
39 |
40 | // No wrap
41 | //
42 | // Prevents table content from wrapping to the next line
43 |
44 | .table-nowrap {
45 |
46 | th, td {
47 | white-space: nowrap;
48 | }
49 | }
50 |
51 |
52 | // Sort
53 | //
54 | // Adds sorting icons
55 |
56 | .table [data-sort] {
57 | white-space: nowrap;
58 |
59 | &::after {
60 | content: str-replace(url("data:image/svg+xml;utf8, "), "#", "%23");
61 | margin-left: .25rem;
62 | }
63 | }
64 |
65 |
66 | // Table checkbox
67 |
68 | .table-checkbox {
69 | min-height: 0;
70 | }
71 |
72 | .table-checkbox .custom-control-label::before,
73 | .table-checkbox .custom-control-label::after {
74 | top: 50%;
75 | transform: translateY(-50%);
76 | }
77 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_toasts.scss:
--------------------------------------------------------------------------------
1 | //
2 | // toasts.scss
3 | // Extended from Bootstrap
4 | //
5 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_type.scss:
--------------------------------------------------------------------------------
1 | //
2 | // type.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | //
7 | // Bootstrap Overrides =====================================
8 | //
9 |
10 | h1, .h1 {
11 | margin-bottom: $headings-margin-bottom;
12 | font-size: 1.5rem;
13 |
14 | @include media-breakpoint-up(md) {
15 | font-size: $h1-font-size;
16 | }
17 | }
18 |
19 | h2, .h2 {
20 | margin-bottom: $headings-margin-bottom;
21 | }
22 |
23 | h3, .h3 {
24 | margin-bottom: ($headings-margin-bottom * .75);
25 | }
26 |
27 | h4, .h4 {
28 | margin-bottom: ($headings-margin-bottom * .5);
29 | }
30 |
31 | h5, .h5 {
32 | margin-bottom: ($headings-margin-bottom * .5);
33 | }
34 |
35 | h6, .h6 {
36 | margin-bottom: ($headings-margin-bottom * .5);
37 | }
38 |
39 |
40 | // Links
41 |
42 | h1, .h1, h2, .h2, h3, .h3, h4, .h4, h5, .h5, h6, .h6 {
43 |
44 | > a {
45 | color: inherit;
46 | }
47 | }
48 |
49 | // Type display classes
50 |
51 | .display-1,
52 | .display-2,
53 | .display-3,
54 | .display-4 {
55 | letter-spacing: $display-letter-spacing;
56 | }
57 |
58 | // Headings
59 |
60 | h1, .h1, h2, .h2, h3, .h3, h4, .h4, h5, .h5, h6, .h6 {
61 | letter-spacing: $headings-letter-spacing;
62 | }
63 |
64 | h6.text-uppercase {
65 | letter-spacing: .08em;
66 | }
67 |
68 | // Bold, strong
69 |
70 | b, strong {
71 | font-weight: $font-weight-bold;
72 | }
73 |
74 |
75 | // Links, buttons
76 | //
77 | // Removes focus outline
78 |
79 | a, button {
80 |
81 | &:focus {
82 | outline: none !important;
83 | }
84 | }
85 |
86 |
87 | //
88 | // Dashkit =====================================
89 | //
90 |
91 | // Include Cerebri Sans
92 |
93 | @font-face {
94 | font-family: 'Cerebri Sans';
95 | src: url('#{$path-to-fonts}/cerebrisans/cerebrisans-regular.eot');
96 | src: url('#{$path-to-fonts}/cerebrisans/cerebrisans-regular.eot?#iefix') format('embedded-opentype'), url('#{$path-to-fonts}/cerebrisans/cerebrisans-regular.woff') format('woff'), url('#{$path-to-fonts}/cerebrisans/cerebrisans-regular.ttf') format('truetype');
97 | font-weight: 400;
98 | font-style: normal;
99 | }
100 |
101 | @font-face {
102 | font-family: 'Cerebri Sans';
103 | src: url('#{$path-to-fonts}/cerebrisans/cerebrisans-medium.eot');
104 | src: url('#{$path-to-fonts}/cerebrisans/cerebrisans-medium.eot?#iefix') format('embedded-opentype'), url('#{$path-to-fonts}/cerebrisans/cerebrisans-medium.woff') format('woff'), url('#{$path-to-fonts}/cerebrisans/cerebrisans-medium.ttf') format('truetype');
105 | font-weight: 500;
106 | font-style: normal;
107 | }
108 |
109 | @font-face {
110 | font-family: 'Cerebri Sans';
111 | src: url('#{$path-to-fonts}/cerebrisans/cerebrisans-semibold.eot');
112 | src: url('#{$path-to-fonts}/cerebrisans/cerebrisans-semibold.eot?#iefix') format('embedded-opentype'), url('#{$path-to-fonts}/cerebrisans/cerebrisans-semibold.woff') format('woff'), url('#{$path-to-fonts}/cerebrisans/cerebrisans-semibold.ttf') format('truetype');
113 | font-weight: 600;
114 | font-style: normal;
115 | }
116 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_utilities.scss:
--------------------------------------------------------------------------------
1 | @import 'utilities/background';
2 | @import 'utilities/borders';
3 | @import 'utilities/lift';
4 | @import 'utilities/sizing';
5 | @import 'utilities/type';
--------------------------------------------------------------------------------
/client/src/styles/dashkit/_vendors.scss:
--------------------------------------------------------------------------------
1 | @import 'vendors/dropzone';
2 | @import 'vendors/flatpickr';
3 | @import 'vendors/quill';
4 | @import 'vendors/list';
5 | @import 'vendors/select2';
6 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/dark/_overrides-dark.scss:
--------------------------------------------------------------------------------
1 | //
2 | // overrides.scss
3 | // Dark mode overrides
4 | //
5 |
6 | //
7 | // Table of contents
8 | //
9 | // 1. Buttons
10 | // 2. Forms
11 | // 3. Input groups
12 | // 4. Quill
13 | // 5. Select2
14 | //
15 |
16 |
17 | // Buttons
18 |
19 | .btn-white, .btn-light {
20 | @include button-variant($gray-800-dark, $gray-600-dark);
21 |
22 | &:not(:disabled):not(.disabled):hover,
23 | &:not(:disabled):not(.disabled):focus,
24 | &:not(:disabled):not(.disabled):active,
25 | &:not(:disabled):not(.disabled).active,
26 | &:not(:disabled):not(.disabled):active:focus,
27 | &:not(:disabled):not(.disabled).active:focus,
28 | .show > &.dropdown-toggle {
29 | background-color: $black-dark;
30 | border-color: $gray-700-dark;
31 | color: $white;
32 | }
33 | }
34 |
35 |
36 | // Forms
37 |
38 | .form-control {
39 | border-color: $input-bg;
40 | }
41 |
42 |
43 | // Input groups
44 |
45 | .input-group .input-group-text {
46 | border-color: $input-bg;
47 | }
48 |
49 |
50 | // Quill
51 |
52 | .ql-toolbar {
53 | border-color: $input-bg;
54 | }
55 |
56 | .ql-editor {
57 | border-left-color: $input-bg;
58 | border-right-color: $input-bg;
59 | border-bottom-color: $input-bg;
60 | }
61 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/mixins/_badge.scss:
--------------------------------------------------------------------------------
1 | // Badge Mixins
2 | //
3 | // This is a custom mixin for badge-#{color}-soft variant of Bootstrap's .badge class
4 |
5 | @mixin badge-variant-soft($bg, $color) {
6 | color: $color;
7 | background-color: $bg;
8 |
9 | &[href] {
10 | @include hover-focus {
11 | color: $color;
12 | text-decoration: none;
13 | background-color: darken($bg, 5%);
14 | }
15 | }
16 | }
17 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/mixins/_breakpoints.scss:
--------------------------------------------------------------------------------
1 | //
2 | // breakpoint.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | @function breakpoint-prev($name, $breakpoints: $grid-breakpoints, $breakpoint-names: map-keys($breakpoints)) {
7 | $n: index($breakpoint-names, $name);
8 | @return if($n != null and $n != 1, nth($breakpoint-names, $n - 1), null);
9 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/utilities/_background.scss:
--------------------------------------------------------------------------------
1 | //
2 | // background.scss
3 | //
4 |
5 |
6 | // Fixed at the bottom
7 |
8 | .bg-fixed-bottom {
9 | background-repeat: no-repeat;
10 | background-position: right bottom;
11 | background-size: 100% auto;
12 | background-attachment: fixed;
13 | }
14 |
15 | // Calculate the width of the main container because
16 | // the background-attachment property will use 100vw instead
17 |
18 | .navbar-vertical ~ .main-content.bg-fixed-bottom {
19 | background-size: 100%;
20 |
21 | @include media-breakpoint-up(md) {
22 | background-size: calc(100% - #{$navbar-vertical-width});
23 | }
24 | }
25 |
26 |
27 | // Cover
28 |
29 | .bg-cover {
30 | background-repeat: no-repeat;
31 | background-position: center center;
32 | background-size: cover;
33 | }
34 |
35 |
36 | // Auth
37 |
38 | .bg-auth {
39 | background-color: $auth-bg;
40 | }
41 |
42 |
43 | // Ellipses
44 |
45 | @each $color, $value in $theme-colors {
46 |
47 | .bg-ellipses.bg-#{$color} {
48 | background-color: transparent !important;
49 | background-repeat: no-repeat;
50 | background-image: radial-gradient(#{$value}, #{$value} 70%, transparent 70.1%);
51 | background-size: 200% 150%;
52 | background-position: center bottom;
53 | }
54 | }
55 |
56 |
57 | // Hero
58 |
59 | .bg-hero {
60 | background-image: linear-gradient(to bottom, fade-out($black, .15), fade-out($black, .15)), url(../#{$path-to-img}/covers/header-cover.jpg);
61 | background-repeat: no-repeat, no-repeat;
62 | background-position: center center, center center;
63 | background-size: cover, cover;
64 | }
65 |
66 |
67 | // Colors
68 |
69 | .bg-lighter {
70 | background-color: $lighter !important;
71 | }
72 |
73 |
74 | // Soft colors
75 |
76 | @each $color, $value in $theme-colors {
77 | .bg-#{$color}-soft {
78 | background-color: theme-color-level($color, $bg-soft-level) !important;
79 | }
80 | }
81 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/utilities/_borders.scss:
--------------------------------------------------------------------------------
1 | //
2 | // borders.scss
3 | //
4 |
5 |
6 | // Sizing
7 |
8 | $border-sizing: () !default;
9 | $border-sizing: map-merge((
10 | "2": 2,
11 | "3": 3,
12 | "4": 4,
13 | "5": 5
14 | ), $border-sizing);
15 |
16 | @each $size, $value in $border-sizing {
17 |
18 | .border-#{$size} {
19 | border-width: $border-width * $value !important;
20 | }
21 |
22 | .border-top-#{$size} {
23 | border-top-width: $border-width * $value !important;
24 | }
25 |
26 | .border-right-#{$size} {
27 | border-right-width: $border-width * $value !important;
28 | }
29 |
30 | .border-bottom-#{$size} {
31 | border-bottom-width: $border-width * $value !important;
32 | }
33 |
34 | .border-left-#{$size} {
35 | border-left-width: $border-width * $value !important;
36 | }
37 | }
38 |
39 |
40 | // Contextual classes
41 |
42 | .border-body {
43 | border-color: $body-bg !important;
44 | }
45 |
46 | .border-card {
47 | border-color: $card-bg !important;
48 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/utilities/_lift.scss:
--------------------------------------------------------------------------------
1 | //
2 | // lift.scss
3 | // Theme utility
4 | //
5 |
6 | .lift {
7 | transition: box-shadow .25s ease, transform .25s ease;
8 | }
9 |
10 | .lift:hover,
11 | .lift:focus {
12 | box-shadow: $box-shadow-lift !important;
13 | transform: translate3d(0, -3px, 0);
14 | }
15 |
16 | .lift-lg:hover,
17 | .lift-lg:focus {
18 | box-shadow: $box-shadow-lift-lg !important;
19 | transform: translate3d(0, -5px, 0);
20 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/utilities/_sizing.scss:
--------------------------------------------------------------------------------
1 | //
2 | // sizing.scss
3 | //
4 |
5 | .vw-100 {
6 | width: 100vw !important;
7 | }
8 | .vh-100 {
9 | height: 100vh !important;
10 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/utilities/_type.scss:
--------------------------------------------------------------------------------
1 | //
2 | // type.scss
3 | // Extended from Bootstrap
4 | //
5 |
6 | // Font size
7 |
8 | .font-size-base {
9 | font-size: $font-size-base !important;
10 | }
11 |
12 | .font-size-sm {
13 | font-size: $font-size-sm !important;
14 | }
15 |
16 | .font-size-lg {
17 | font-size: $font-size-lg !important;
18 | }
19 |
20 |
21 | // Decoration
22 |
23 | .text-decoration-underline {
24 | text-decoration: underline !important;
25 | }
26 |
27 |
28 | // Gray colors
29 |
30 | @each $color, $value in $grays {
31 | .text-gray-#{$color} {
32 | color: $value !important;
33 | }
34 | }
35 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/vendors/_dropzone.scss:
--------------------------------------------------------------------------------
1 | //
2 | // dropzone.scss
3 | // Dropzone plugin overrides
4 | //
5 |
6 | .dropzone {
7 | position: relative;
8 | display: flex;
9 | flex-direction: column;
10 | }
11 |
12 | .dz-message {
13 | padding: 5rem 1rem;
14 | background-color: $input-bg;
15 | border: $input-border-width dashed $input-border-color;
16 | border-radius: $border-radius;
17 | text-align: center;
18 | color: $text-muted;
19 | transition: $transition-base;
20 | order: -1;
21 | cursor: pointer;
22 | z-index: 999;
23 |
24 | &:hover {
25 | border-color: $text-muted;
26 | color: $body-color;
27 | }
28 | }
29 |
30 | .dz-drag-hover .dz-message {
31 | border-color: $primary;
32 | color: $primary;
33 | }
34 |
35 | .dropzone-multiple .dz-message {
36 | padding-top: 2rem;
37 | padding-bottom: 2rem;
38 | }
39 |
40 | .dropzone-single.dz-max-files-reached .dz-message {
41 | background-color: fade-out($black, .1);
42 | color: white;
43 | opacity: 0;
44 |
45 | &:hover {
46 | opacity: 1;
47 | }
48 | }
49 |
50 | .dz-preview-single {
51 | position: absolute;
52 | top: 0; right: 0; bottom: 0; left: 0;
53 | border-radius: $border-radius;
54 | }
55 |
56 | .dz-preview-cover {
57 | position: absolute;
58 | top: 0; right: 0; bottom: 0; left: 0;
59 | border-radius: $border-radius;
60 | }
61 |
62 | .dz-preview-img {
63 | object-fit: cover;
64 | width: 100%; height: 100%;
65 | border-radius: $border-radius;
66 | }
67 |
68 | .dz-preview-multiple .list-group-item:last-child {
69 | padding-bottom: 0;
70 | border-bottom: 0;
71 | }
72 |
73 | [data-dz-size] strong {
74 | font-weight: $font-weight-normal;
75 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/vendors/_flatpickr.scss:
--------------------------------------------------------------------------------
1 | //
2 | // flatpickr.scss
3 | // Flatpickr plugin overrides
4 | //
5 |
6 | .flatpickr-calendar {
7 | background-color: $input-bg;
8 | border: $input-border-width solid $input-border-color;
9 | color: $input-color;
10 | box-shadow: none;
11 |
12 | * {
13 | color: inherit !important;
14 | fill: currentColor !important;
15 | }
16 |
17 | &.arrowTop:before {
18 | border-bottom-color: $input-border-color;
19 | }
20 |
21 | &.arrowTop:after {
22 | border-bottom-color: $input-bg;
23 | }
24 |
25 | .flatpickr-months {
26 | padding-top: .625rem;
27 | padding-bottom: .625rem;
28 | }
29 |
30 | .flatpickr-prev-month,
31 | .flatpickr-next-month {
32 | top: .625rem;
33 | }
34 |
35 | .flatpickr-current-month {
36 | font-size: 115%;
37 | }
38 |
39 | .flatpickr-day {
40 | border-radius: $border-radius;
41 |
42 | &:hover {
43 | background-color: $light;
44 | border-color: $input-border-color;
45 | }
46 | }
47 |
48 | .flatpickr-day.prevMonthDay {
49 | color: $text-muted !important;
50 | }
51 |
52 | .flatpickr-day.today {
53 | border-color: $border-color;
54 | }
55 |
56 | .flatpickr-day.selected {
57 | background-color: $primary;
58 | border-color: $primary;
59 | color: $white !important;
60 | }
61 |
62 | .flatpickr-day.inRange {
63 | background-color: $light;
64 | border: none;
65 | border-radius: 0;
66 | box-shadow: -5px 0 0 $light, 5px 0 0 $light;
67 | }
68 |
69 | }
--------------------------------------------------------------------------------
/client/src/styles/dashkit/vendors/_list.scss:
--------------------------------------------------------------------------------
1 | //
2 | // list.scss
3 | //
4 |
5 | [data-toggle="lists"] .pagination > li {
6 | @extend .page-item;
7 | }
8 |
9 | [data-toggle="lists"] .pagination .page {
10 | @extend .page-link;
11 | }
12 |
--------------------------------------------------------------------------------
/client/src/styles/dashkit/vendors/_select2.scss:
--------------------------------------------------------------------------------
1 | //
2 | // select2.scss
3 | // Select2 plugin overrides
4 | //
5 |
6 | [class*="select2"] {
7 | display: block;
8 | }
9 |
10 | .select2 {
11 | width: 100% !important;
12 | }
13 |
14 | .select2-hidden-accessible {
15 | display: none;
16 | }
17 |
18 | .select2-selection[aria-expanded="true"] {
19 | border-bottom-left-radius: 0;
20 | border-bottom-right-radius: 0;
21 | }
22 |
23 | .select2-container {
24 | display: block;
25 | }
26 |
27 | .select2-dropdown {
28 | margin-top: -$input-border-width;
29 | border-top-left-radius: 0;
30 | border-top-right-radius: 0;
31 | }
32 |
33 | .select2-search--dropdown {
34 | padding: $dropdown-item-padding-y $input-padding-x;
35 | }
36 |
37 | .select2-search--dropdown .select2-search__field {
38 | width: 100%;
39 | height: $input-height-sm;
40 | padding: $input-padding-y-sm $input-padding-x-sm;
41 | background-color: $input-bg;
42 | border: $input-border-width solid $input-border-color;
43 | border-radius: $input-border-radius-sm;
44 | line-height: $input-line-height-sm;
45 | font-size: $input-font-size-sm;
46 | color: $input-color;
47 | transition: $input-transition;
48 |
49 | &:focus {
50 | border-color: $input-focus-border-color;
51 | box-shadow: $input-focus-box-shadow;
52 | outline: none;
53 | }
54 | }
55 |
56 | .select2-results__options {
57 | padding-left: 0;
58 | margin-bottom: 0;
59 | }
60 |
61 | .select2-results__option {
62 | padding: $dropdown-item-padding-y $input-padding-x;
63 | color: $dropdown-link-color;
64 |
65 | &:not(.select2-results__message) {
66 | cursor: pointer;
67 |
68 | @include hover-focus {
69 | color: $dropdown-link-hover-color;
70 | }
71 | }
72 | }
73 |
74 | .select2-results__option[aria-selected="true"],
75 | .select2-results__option--highlighted {
76 | color: $dropdown-link-active-color;
77 | }
78 |
79 | .select2-selection--multiple {
80 | height: auto;
81 | }
82 |
83 | .select2-selection__rendered {
84 | display: flex;
85 | flex-wrap: wrap;
86 | padding-left: 0;
87 | margin: 0 -.25rem -.25rem 0;
88 | }
89 |
90 | .select2-selection__choice {
91 | display: inline-flex;
92 | align-items: center;
93 | padding-left: .375rem;
94 | padding-right: .375rem;
95 | margin: 0 .25rem .25rem 0;
96 | font-size: $font-size-sm;
97 | background-color: $light;
98 | border-radius: $border-radius-xs;
99 | }
100 |
101 | .select2-selection__choice__remove {
102 | order: 2;
103 | margin-left: .5rem;
104 | color: $text-muted;
105 | cursor: pointer;
106 |
107 | @include hover {
108 | color: $body-color;
109 | }
110 | }
111 |
112 | .select2-search--inline .select2-search__field {
113 | height: calc(1em * #{$input-line-height});
114 | padding-bottom: .25rem;
115 | background-color: transparent;
116 | border: 0;
117 | box-shadow: none;
118 | outline: none;
119 | color: $input-color;
120 |
121 | &::placeholder {
122 | color: $input-placeholder-color;
123 | }
124 | }
125 |
126 | .select2-selection__placeholder {
127 | color: $input-placeholder-color;
128 | }
129 |
--------------------------------------------------------------------------------
/client/src/styles/index.scss:
--------------------------------------------------------------------------------
1 | // Icon font
2 | @import "../fonts/feather/feather";
3 |
4 | // Bootstrap functions
5 | @import "~bootstrap/scss/functions.scss";
6 |
7 | // Custom variables
8 | @import "custom-variables";
9 |
10 | // Dark mode variables
11 | @import "dashkit/dark/variables-dark";
12 |
13 | // Dashkit variables
14 | @import "dashkit/variables";
15 |
16 | // Bootstrap core
17 | @import "~bootstrap/scss/bootstrap.scss";
18 |
19 | // Dashkit core
20 | @import "dashkit/dashkit";
21 |
22 | // Dark mode overrides
23 | @import "dashkit/dark/overrides-dark";
24 |
25 | // Custom core
26 | @import "custom";
27 |
--------------------------------------------------------------------------------
/client/src/styles/tools/shadows.scss:
--------------------------------------------------------------------------------
1 | //shadows
2 |
3 | //example
4 | // @include box-shadow(3);
5 |
6 | @mixin box-shadow($level) {
7 | @if $level == 1 {
8 | box-shadow: 1px 1px 8px rgba(0, 7, 43, 0.04);
9 | } @else if $level == 2 {
10 | box-shadow: 4px 8px 24px rgba(0, 7, 43, 0.04);
11 | } @else if $level == 3 {
12 | box-shadow: 8px 24px 48px rgba(0, 7, 43, 0.08);
13 | } @else if $level == 4 {
14 | box-shadow: 40px 40px 96px rgba(0, 7, 43, 0.08);
15 | }
16 | }
17 |
18 | //Modal Overlay
19 | @mixin modal-overlay {
20 | background: rgba(0, 7, 43, 0.4);
21 | }
22 |
23 | //Active Inputs Shadow
24 | @mixin active-input-shadow {
25 | box-shadow: 1px 1px 4px rgba(0, 85, 255, 0.08);
26 | }
27 |
28 | //Avatar Inner Shadow
29 | @mixin avatar-inner-shadow {
30 | box-shadow: inset 0px 4px 4px rgba(0, 7, 43, 0.25);
31 | }
32 |
--------------------------------------------------------------------------------
/client/src/styles/tools/transitions.scss:
--------------------------------------------------------------------------------
1 | @mixin transition() {
2 | transition: 0.2s ease-in-out;
3 | }
4 |
5 | @mixin transition-hover() {
6 | transition: 0.5s ease-in-out;
7 | }
8 |
--------------------------------------------------------------------------------
/client/src/utils/bytes.ts:
--------------------------------------------------------------------------------
1 | // Program data is stored as little-endian
2 | // [0x1, 0x0] -> id #0, [0x0, 0x1] -> id #8
3 | export function programDataToIds(bytes: Uint8Array): Array {
4 | const ids = new Array();
5 | bytes.forEach((byte, i) => {
6 | for (let j = 7; j >= 0; j--) {
7 | if ((byte & (1 << j)) === 1 << j) {
8 | ids.push(8 * i + (7 - j));
9 | }
10 | }
11 | });
12 | return ids;
13 | }
14 |
15 | // Instruction data uses big-endian
16 | // id #0 -> [0x0, 0x0], id #256 -> [0x1, 0x0]
17 | const MAX_ID = Math.pow(2, 16) - 1;
18 | export function instructionDataFromId(id: number): Uint8Array {
19 | if (id > MAX_ID || id < 0 || !Number.isInteger(id)) {
20 | throw new Error("invalid id");
21 | }
22 |
23 | const bytes = new Uint8Array(2);
24 | bytes[0] = Math.floor(id / 256);
25 | bytes[1] = id % 256;
26 | return bytes;
27 | }
28 |
29 | export function xor(a: Uint8Array, b: Uint8Array): Uint8Array {
30 | if (a.length !== b.length) throw new Error("bytes are not the same length");
31 | const bytes = new Uint8Array(a);
32 | for (let i = 0; i < b.length; i++) {
33 | bytes[i] ^= b[i];
34 | }
35 | return bytes;
36 | }
37 |
--------------------------------------------------------------------------------
/client/src/utils/index.ts:
--------------------------------------------------------------------------------
1 | import { Keypair } from "@solana/web3.js";
2 | // import * as Sentry from "@sentry/react";
3 |
4 | export function sleep(ms: number): Promise {
5 | return new Promise((resolve) => setTimeout(resolve, ms));
6 | }
7 |
8 | export function reportError(err: unknown, context: string) {
9 | if (err instanceof Error) {
10 | console.error(context, err);
11 | }
12 | // if (process.env.NODE_ENV === "production") {
13 | // const query = new URLSearchParams(window.location.search);
14 | // if (query.get("cluster") === "custom") return;
15 | // Sentry.captureException(err, {
16 | // tags: { context },
17 | // });
18 | // }
19 | }
20 |
21 | export function isLocalHost() {
22 | return window.location.hostname === "localhost";
23 | }
24 |
25 | export const getLocalStorageKeypair = (key: string): Keypair => {
26 | const base64Keypair = window.localStorage.getItem(key);
27 | if (base64Keypair) {
28 | return Keypair.fromSecretKey(Buffer.from(base64Keypair, "base64"));
29 | } else {
30 | const keypair = new Keypair();
31 | window.localStorage.setItem(
32 | key,
33 | Buffer.from(keypair.secretKey).toString("base64")
34 | );
35 | return keypair;
36 | }
37 | };
38 |
39 | export const getFeePayers = (num: number) => {
40 | const accounts = [];
41 | for (let i = 0; i < num; i++) {
42 | accounts.push(getLocalStorageKeypair(`feePayerKey${i + 1}`));
43 | }
44 | return accounts;
45 | };
46 |
--------------------------------------------------------------------------------
/client/src/worker-loader.d.ts:
--------------------------------------------------------------------------------
1 | declare module "worker-loader!*" {
2 | class WebpackWorker extends Worker {
3 | constructor();
4 | }
5 |
6 | export default WebpackWorker;
7 | }
8 |
--------------------------------------------------------------------------------
/client/src/workers/create-transaction-rpc.ts:
--------------------------------------------------------------------------------
1 | // eslint-disable-next-line import/no-webpack-loader-syntax
2 | import CreateTransactionWorker from "worker-loader!./create-transaction-worker-script";
3 |
4 | import { Blockhash } from "@solana/web3.js";
5 |
6 | export interface CreateTransactionMessage {
7 | trackingId: number;
8 | blockhash: Blockhash;
9 | programId: string;
10 | bitId: number;
11 | feeAccountSecretKey: Uint8Array;
12 | programDataAccount: string;
13 | computeUnitPrice?: number;
14 | extraWriteAccount?: string;
15 | }
16 |
17 | export interface CreateTransactionResponseMessage {
18 | trackingId: number;
19 | signature: Buffer;
20 | serializedTransaction: Buffer;
21 | }
22 |
23 | export interface CreateTransactionErrorMessage {
24 | trackingId: string;
25 | error: Error;
26 | }
27 |
28 | export class CreateTransactionRPC {
29 | private worker: CreateTransactionWorker;
30 |
31 | private callbacks: { [trackingId: string]: Function[] } = {};
32 |
33 | constructor() {
34 | this.worker = new CreateTransactionWorker();
35 | this.worker.onmessage = this.handleMessages.bind(this);
36 | }
37 |
38 | handleMessages(event: MessageEvent) {
39 | let message = event.data;
40 |
41 | if (message.trackingId in this.callbacks) {
42 | let callbacks = this.callbacks[message.trackingId];
43 | delete this.callbacks[message.trackingId];
44 |
45 | if ("error" in message) {
46 | callbacks[1](message.error);
47 | return;
48 | }
49 |
50 | callbacks[0](message);
51 | }
52 | }
53 |
54 | createTransaction(
55 | message: CreateTransactionMessage
56 | ): Promise {
57 | return new Promise((resolve, reject) => {
58 | this.callbacks[message.trackingId] = [resolve, reject];
59 | this.worker.postMessage(message);
60 | });
61 | }
62 | }
63 |
--------------------------------------------------------------------------------
/client/src/workers/create-transaction-worker-script.ts:
--------------------------------------------------------------------------------
1 | import {
2 | Transaction,
3 | PublicKey,
4 | Keypair,
5 | ComputeBudgetProgram,
6 | } from "@solana/web3.js";
7 | import * as Bytes from "utils/bytes";
8 | import { CreateTransactionMessage } from "./create-transaction-rpc";
9 |
10 | const self: any = globalThis;
11 |
12 | function createTransaction(message: CreateTransactionMessage) {
13 | const {
14 | trackingId,
15 | blockhash,
16 | programId,
17 | bitId,
18 | feeAccountSecretKey,
19 | programDataAccount,
20 | computeUnitPrice,
21 | extraWriteAccount,
22 | } = message;
23 |
24 | const transaction = new Transaction();
25 | if (computeUnitPrice) {
26 | const units = 1000;
27 | const MICRO_LAMPORTS_PER_LAMPORT = 1_000_000;
28 | const additionalFee = Math.ceil(
29 | (computeUnitPrice * units) / MICRO_LAMPORTS_PER_LAMPORT
30 | );
31 | transaction.add(
32 | ComputeBudgetProgram.requestUnits({
33 | units,
34 | additionalFee,
35 | })
36 | );
37 | }
38 | const breakAccountInputs = [
39 | {
40 | pubkey: new PublicKey(programDataAccount),
41 | isWritable: true,
42 | isSigner: false,
43 | },
44 | ];
45 | if (extraWriteAccount) {
46 | breakAccountInputs.push({
47 | pubkey: new PublicKey(extraWriteAccount),
48 | isWritable: true,
49 | isSigner: false,
50 | });
51 | }
52 | transaction.add({
53 | keys: breakAccountInputs,
54 | programId: new PublicKey(programId),
55 | data: Buffer.from(Bytes.instructionDataFromId(bitId)),
56 | });
57 | transaction.recentBlockhash = blockhash;
58 | transaction.sign(Keypair.fromSecretKey(feeAccountSecretKey));
59 |
60 | const signatureBuffer = transaction.signature;
61 |
62 | self.postMessage({
63 | trackingId: trackingId,
64 | signature: signatureBuffer,
65 | serializedTransaction: transaction.serialize(),
66 | });
67 | }
68 |
69 | self.onmessage = (event: any) => {
70 | const message = event.data;
71 |
72 | try {
73 | createTransaction(message);
74 | } catch (error) {
75 | self.postMessage({
76 | trackingId: message.trackingId,
77 | error: error,
78 | });
79 | }
80 | };
81 |
--------------------------------------------------------------------------------
/client/svgTransform.js:
--------------------------------------------------------------------------------
1 | module.exports = {
2 | process(): string {
3 | return "module.exports = {};";
4 | },
5 | getCacheKey(): string {
6 | // The output is always the same.
7 | return "svgTransform";
8 | },
9 | };
10 |
--------------------------------------------------------------------------------
/client/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "target": "es5",
4 | "lib": ["dom", "dom.iterable", "esnext", "webworker"],
5 | "allowJs": true,
6 | "skipLibCheck": true,
7 | "downlevelIteration": true,
8 | "esModuleInterop": true,
9 | "allowSyntheticDefaultImports": true,
10 | "strict": true,
11 | "forceConsistentCasingInFileNames": true,
12 | "module": "esnext",
13 | "moduleResolution": "node",
14 | "resolveJsonModule": true,
15 | "isolatedModules": true,
16 | "noEmit": true,
17 | "jsx": "react-jsx",
18 | "baseUrl": "src",
19 | "noFallthroughCasesInSwitch": true
20 | },
21 | "include": ["src"]
22 | }
23 |
--------------------------------------------------------------------------------
/package-lock.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@solana/break-app",
3 | "lockfileVersion": 2,
4 | "requires": true,
5 | "packages": {
6 | "": {
7 | "name": "@solana/break-app",
8 | "hasInstallScript": true
9 | }
10 | }
11 | }
12 |
--------------------------------------------------------------------------------
/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@solana/break-app",
3 | "cacheDirectories": [
4 | "client/node_modules",
5 | "server/node_modules"
6 | ],
7 | "scripts": {
8 | "client:build": "./package.sh client run build",
9 | "client:install": "./package.sh client install",
10 | "server:build": "./package.sh server run build",
11 | "server:start": "./package.sh server run start",
12 | "server:install": "./package.sh server install",
13 | "program:build": "./program/do.sh build",
14 | "program:clean": "./program/do.sh clean",
15 | "start": "npm run server:start",
16 | "build": "./package.sh both run build",
17 | "lint": "./package.sh both run lint",
18 | "lint:fix": "./package.sh both run lint:fix",
19 | "format": "./package.sh both run format",
20 | "format:fix": "./package.sh both run format:fix",
21 | "postinstall": "./package.sh both install"
22 | }
23 | }
24 |
--------------------------------------------------------------------------------
/package.sh:
--------------------------------------------------------------------------------
1 | #!/bin/bash
2 |
3 | dir="$1"
4 | shift
5 | if [ "$dir" = "both" ]; then
6 | set -ex
7 | (cd client && npm "$@" )
8 | (cd server && npm "$@" )
9 | else
10 | set -ex
11 | (cd $dir && npm "$@" )
12 | fi
13 |
--------------------------------------------------------------------------------
/program/.gitignore:
--------------------------------------------------------------------------------
1 | # dist is intentionally not ignored
2 | target
3 |
--------------------------------------------------------------------------------
/program/Cargo.toml:
--------------------------------------------------------------------------------
1 | # Note: This crate must be built using build.sh
2 |
3 | [package]
4 | name = "break-solana-program"
5 | version = "0.1.0"
6 | description = "Break Solana program"
7 | authors = ["Solana Maintainers "]
8 | repository = "https://github.com/solana-labs/break"
9 | license = "Apache-2.0"
10 | homepage = "https://break.solana.com/"
11 | edition = "2018"
12 |
13 | [features]
14 | no-entrypoint = []
15 |
16 | [dependencies]
17 | solana-program = "=1.6.6"
18 |
19 | [lib]
20 | crate-type = ["cdylib", "lib"]
21 | name = "break_solana_program"
22 |
--------------------------------------------------------------------------------
/program/Xargo.toml:
--------------------------------------------------------------------------------
1 | [target.bpfel-unknown-unknown.dependencies.std]
2 | features = []
--------------------------------------------------------------------------------
/program/dist/break_solana_program.so:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/solana-labs/break/c97989858498f056954649c0f8ae03862a805a4d/program/dist/break_solana_program.so
--------------------------------------------------------------------------------
/program/src/lib.rs:
--------------------------------------------------------------------------------
1 | use solana_program::{
2 | account_info::AccountInfo, entrypoint, entrypoint::ProgramResult, pubkey::Pubkey,
3 | };
4 |
5 | entrypoint!(process_instruction);
6 | fn process_instruction<'a>(
7 | _program_id: &Pubkey,
8 | accounts: &'a [AccountInfo<'a>],
9 | instruction_data: &[u8],
10 | ) -> ProgramResult {
11 | // Assume a writable account is at index 0
12 | let mut account_data = accounts[0].try_borrow_mut_data()?;
13 |
14 | // xor with the account data using byte and bit from ix data
15 | let index = u16::from_be_bytes([instruction_data[0], instruction_data[1]]);
16 | let byte = index >> 3;
17 | let bit = (index & 0x7) as u8;
18 | account_data[byte as usize] ^= 1 << (7 - bit);
19 |
20 | Ok(())
21 | }
22 |
23 | // Sanity tests
24 | #[cfg(test)]
25 | mod test {
26 | use super::*;
27 | use solana_program::clock::Epoch;
28 |
29 | #[test]
30 | fn test_xor() {
31 | let program_id = Pubkey::default();
32 | let key = Pubkey::default();
33 | let mut lamports = 0;
34 | let mut data = vec![0; 4];
35 | let owner = Pubkey::default();
36 | let account = AccountInfo::new(
37 | &key,
38 | false,
39 | true,
40 | &mut lamports,
41 | &mut data,
42 | &owner,
43 | false,
44 | Epoch::default(),
45 | );
46 |
47 | let accounts = vec![account];
48 |
49 | process_instruction(&program_id, &accounts, &[1, 1, 1, 1]).unwrap();
50 | assert_eq!(*accounts[0].data.borrow(), &[1, 1, 1, 1]);
51 |
52 | process_instruction(&program_id, &accounts, &[1, 1, 1, 1]).unwrap();
53 | assert_eq!(*accounts[0].data.borrow(), &[0, 0, 0, 0]);
54 | }
55 |
56 | #[test]
57 | #[should_panic]
58 | fn test_bad_instruction_data() {
59 | let program_id = Pubkey::default();
60 | let key = Pubkey::default();
61 | let mut lamports = 0;
62 | let mut data = vec![0; 2];
63 | let owner = Pubkey::default();
64 | let account = AccountInfo::new(
65 | &key,
66 | false,
67 | true,
68 | &mut lamports,
69 | &mut data,
70 | &owner,
71 | false,
72 | Epoch::default(),
73 | );
74 |
75 | let accounts = vec![account];
76 | process_instruction(&program_id, &accounts, &[0, 1, 2]).unwrap();
77 | }
78 |
79 | #[test]
80 | #[should_panic]
81 | fn test_bad_account() {
82 | let program_id = Pubkey::default();
83 | let accounts = vec![];
84 | process_instruction(&program_id, &accounts, &[0, 1, 2, 4]).unwrap();
85 | }
86 | #[test]
87 | #[should_panic]
88 | fn test_bad_account_data() {
89 | let program_id = Pubkey::default();
90 | let key = Pubkey::default();
91 | let mut lamports = 0;
92 | let mut data = vec![0; 3];
93 | let owner = Pubkey::default();
94 | let account = AccountInfo::new(
95 | &key,
96 | false,
97 | true,
98 | &mut lamports,
99 | &mut data,
100 | &owner,
101 | false,
102 | Epoch::default(),
103 | );
104 |
105 | let accounts = vec![account];
106 | process_instruction(&program_id, &accounts, &[0, 1, 2, 4]).unwrap();
107 | }
108 | }
109 |
--------------------------------------------------------------------------------
/server/.eslintignore:
--------------------------------------------------------------------------------
1 | dist
2 | webpack.config.js
3 | src/types
4 |
--------------------------------------------------------------------------------
/server/.eslintrc.json:
--------------------------------------------------------------------------------
1 | {
2 | "env": {
3 | "browser": true,
4 | "es6": true
5 | },
6 | "extends": [
7 | "eslint:recommended",
8 | "plugin:@typescript-eslint/eslint-recommended",
9 | "plugin:@typescript-eslint/recommended"
10 | ],
11 | "globals": {
12 | "Atomics": "readonly",
13 | "SharedArrayBuffer": "readonly"
14 | },
15 | "parser": "@typescript-eslint/parser",
16 | "parserOptions": {
17 | "ecmaVersion": 2018,
18 | "sourceType": "module"
19 | },
20 | "plugins": ["@typescript-eslint"],
21 | "rules": {}
22 | }
23 |
--------------------------------------------------------------------------------
/server/.gitignore:
--------------------------------------------------------------------------------
1 | dist
2 |
--------------------------------------------------------------------------------
/server/.prettierignore:
--------------------------------------------------------------------------------
1 | dist
2 | package-lock.json
3 |
--------------------------------------------------------------------------------
/server/package.json:
--------------------------------------------------------------------------------
1 | {
2 | "name": "@solana/break-backend",
3 | "private": true,
4 | "version": "1.0.0",
5 | "description": "Try to break Solana's network",
6 | "author": "Solana Maintainers ",
7 | "license": "MIT",
8 | "repository": {
9 | "type": "git",
10 | "url": "https://github.com/solana-labs/break"
11 | },
12 | "scripts": {
13 | "build": "webpack --mode production",
14 | "test": "echo 'no test specified'",
15 | "start": "node ./dist/server.js",
16 | "start:dev": "ts-node src/index.ts",
17 | "build:dev": "webpack --mode development",
18 | "format": "prettier --check \"**/*.{js,jsx,ts,tsx,json,md}\"",
19 | "format:fix": "prettier --write \"**/*.{js,jsx,ts,tsx,json,md}\"",
20 | "lint": "set -ex; eslint --ext .js,.ts,.tsx .",
21 | "lint:fix": "npm run lint -- --fix"
22 | },
23 | "testnetDefaultChannel": "edge",
24 | "dependencies": {
25 | "@sentry/node": "^7.1.1",
26 | "@sentry/tracing": "^7.0.0",
27 | "@solana/web3.js": "^1.43.6",
28 | "@types/cors": "^2.8.10",
29 | "@types/express": "^4.17.12",
30 | "@types/node": "^17.0.41",
31 | "@types/ws": "^8.5.3",
32 | "@typescript-eslint/eslint-plugin": "^4.33.0",
33 | "@typescript-eslint/parser": "^4.33.0",
34 | "bs58": "^5.0.0",
35 | "cors": "^2.8.5",
36 | "eslint": "^7.27.0",
37 | "express": "^4.18.1",
38 | "prettier": "^2.6.2",
39 | "ts-loader": "^9.3.0",
40 | "ts-node": "^10.8.1",
41 | "typescript": "^4.7.3",
42 | "webpack": "^5.73.0",
43 | "webpack-cli": "^4.9.2",
44 | "webpack-node-externals": "^3.0.0",
45 | "ws": "^8.7.0"
46 | }
47 | }
48 |
--------------------------------------------------------------------------------
/server/src/api.ts:
--------------------------------------------------------------------------------
1 | import { Connection } from "@solana/web3.js";
2 | import { cluster, url, urlTls } from "./urls";
3 | import { PROGRAM_ID } from "./program";
4 | import TpuProxy from "./tpu_proxy";
5 | import WebSocketServer from "./websocket";
6 | import { Express } from "express";
7 | import http from "http";
8 |
9 | export default class ApiServer {
10 | static async start(app: Express, httpServer: http.Server): Promise {
11 | const connection = new Connection(url, "confirmed");
12 | const tpuProxy = await TpuProxy.create(connection);
13 | WebSocketServer.start(httpServer, tpuProxy);
14 |
15 | await tpuProxy.connect();
16 |
17 | const programId = PROGRAM_ID?.toBase58();
18 | if (!programId) {
19 | throw new Error("Internal error: program id is missing");
20 | }
21 |
22 | app.post("/init", async (req, res) => {
23 | res
24 | .send(
25 | JSON.stringify({
26 | programId,
27 | clusterUrl: urlTls,
28 | cluster,
29 | paymentRequired: process.env.REQUIRE_PAYMENT === "true",
30 | })
31 | )
32 | .end();
33 | });
34 | }
35 | }
36 |
--------------------------------------------------------------------------------
/server/src/available_nodes.ts:
--------------------------------------------------------------------------------
1 | import { Connection } from "@solana/web3.js";
2 | import { endlessRetry } from "./utils";
3 |
4 | type NodeAddress = string;
5 | type TpuEndpoint = string;
6 | type AvailableNodes = Map;
7 | type DelinquentNodes = Set;
8 |
9 | // Polls cluster to determine which nodes are available
10 | export default class AvailableNodesService {
11 | refreshing = false;
12 |
13 | constructor(
14 | private connection: Connection,
15 | public nodes: AvailableNodes,
16 | public delinquents: DelinquentNodes
17 | ) {
18 | // Refresh every 5min in case nodes leave the cluster or change port configuration
19 | setInterval(() => this.refresh(), 5 * 60 * 1000);
20 | }
21 |
22 | static start = async (
23 | connection: Connection
24 | ): Promise => {
25 | const nodes = await AvailableNodesService.getAvailableNodes(connection);
26 | return new AvailableNodesService(connection, nodes, new Set());
27 | };
28 |
29 | private static getAvailableNodes = async (
30 | connection: Connection
31 | ): Promise => {
32 | const availableNodes = new Map();
33 | const nodes = await endlessRetry("getClusterNodes", async () =>
34 | connection.getClusterNodes()
35 | );
36 | for (const node of nodes) {
37 | if (node.tpu) {
38 | availableNodes.set(node.pubkey, node.tpu);
39 | }
40 | }
41 | return availableNodes;
42 | };
43 |
44 | private refresh = async (): Promise => {
45 | if (this.refreshing) return;
46 | this.refreshing = true;
47 | this.nodes = await AvailableNodesService.getAvailableNodes(this.connection);
48 | this.delinquents.clear();
49 | this.refreshing = false;
50 | };
51 | }
52 |
--------------------------------------------------------------------------------
/server/src/index.ts:
--------------------------------------------------------------------------------
1 | import express from "express";
2 | import http from "http";
3 | import cors from "cors";
4 | import path from "path";
5 | import ApiServer from "./api";
6 | import * as Sentry from "@sentry/node";
7 | // eslint-disable-next-line @typescript-eslint/no-unused-vars
8 | import * as Tracing from "@sentry/tracing";
9 |
10 | Sentry.init({
11 | dsn: "https://f74dafc67c914776b018c3be136bca19@o434108.ingest.sentry.io/5411826",
12 | // send 10% of all errors to Sentry
13 | tracesSampleRate: 0.1,
14 | });
15 |
16 | const app = express();
17 |
18 | // Redirect to https on Heroku
19 | if (
20 | process.env.NODE_ENV === "production" &&
21 | process.env.FORCE_HTTPS !== undefined
22 | ) {
23 | app.use((req, res, next) => {
24 | if (req.header("x-forwarded-proto") !== "https") {
25 | res.redirect(`https://${req.header("host")}${req.url}`);
26 | } else {
27 | next();
28 | }
29 | });
30 | }
31 |
32 | const rootPath = path.join(__dirname, "..", "..");
33 | const staticPath = path.join(rootPath, "client", "build");
34 | console.log(`Serving static files from: ${staticPath}`);
35 | app.use("/", express.static(staticPath));
36 | app.get("/*", (req, res) => {
37 | res.sendFile(path.join(staticPath, "/index.html"));
38 | });
39 |
40 | const httpServer = http.createServer(app);
41 | const port = process.env.PORT || 8080;
42 | httpServer.listen(port);
43 | console.log(`Server listening on port: ${port}`);
44 |
45 | if (!process.env.DISABLE_API) {
46 | app.use(cors());
47 | app.use(express.json());
48 | ApiServer.start(app, httpServer);
49 | }
50 |
--------------------------------------------------------------------------------
/server/src/leader_schedule.ts:
--------------------------------------------------------------------------------
1 | import { Connection } from "@solana/web3.js";
2 | import { endlessRetry, reportError } from "./utils";
3 |
4 | // Number of upcoming slots to include when building upcoming node set
5 | export const UPCOMING_SLOT_SEARCH = parseInt(
6 | process.env.LEADER_SLOT_FANOUT || "40"
7 | );
8 |
9 | // Number of past slots to include when building upcoming node set
10 | export const PAST_SLOT_SEARCH = 4;
11 |
12 | // Updates the leader schedule every epoch and provides a set of the
13 | // upcoming nodes in the schedule
14 | export default class LeaderScheduleService {
15 | refreshing = false;
16 |
17 | constructor(
18 | private connection: Connection,
19 | private leaderAddresses: Array,
20 | private scheduleFirstSlot: number
21 | ) {}
22 |
23 | static start = async (
24 | connection: Connection,
25 | currentSlot: number
26 | ): Promise => {
27 | const leaderService = new LeaderScheduleService(
28 | connection,
29 | [],
30 | currentSlot
31 | );
32 |
33 | leaderService.leaderAddresses = await endlessRetry("getSlotLeaders", () =>
34 | leaderService.fetchLeaders(currentSlot)
35 | );
36 |
37 | return leaderService;
38 | };
39 |
40 | private async fetchLeaders(startSlot: number): Promise> {
41 | const leaders = await this.connection.getSlotLeaders(
42 | startSlot,
43 | 2 * UPCOMING_SLOT_SEARCH
44 | );
45 | return leaders.map((l) => l.toBase58());
46 | }
47 |
48 | getFirstSlot = (): number => {
49 | return this.scheduleFirstSlot;
50 | };
51 |
52 | getSlotLeader = (slot: number): string | null => {
53 | const firstSlot = this.scheduleFirstSlot;
54 | const lastSlot = this.lastSlot();
55 | if (slot < firstSlot) {
56 | console.error(
57 | `getSlotLeader failed: Tried to get ${slot} before first schedule slot ${firstSlot}`
58 | );
59 | } else if (slot > lastSlot) {
60 | console.error(
61 | `getSlotLeader failed: Tried to get ${slot} after last schedule slot ${lastSlot}`
62 | );
63 | } else {
64 | return this.leaderAddresses[slot - this.scheduleFirstSlot];
65 | }
66 | return null;
67 | };
68 |
69 | private lastSlot = (): number => {
70 | return this.scheduleFirstSlot + this.leaderAddresses.length - 1;
71 | };
72 |
73 | shouldRefresh = (currentSlot: number): boolean => {
74 | const shouldRefreshAt = this.lastSlot() - UPCOMING_SLOT_SEARCH;
75 | return currentSlot >= shouldRefreshAt;
76 | };
77 |
78 | refresh = async (currentSlot: number): Promise => {
79 | if (this.refreshing) return;
80 | this.refreshing = true;
81 | try {
82 | const firstSlot = Math.max(0, currentSlot - PAST_SLOT_SEARCH);
83 | const leaderAddresses = await this.fetchLeaders(firstSlot);
84 | this.scheduleFirstSlot = firstSlot;
85 | this.leaderAddresses = leaderAddresses;
86 | } catch (err) {
87 | reportError(err, "failed to refresh slot leaders");
88 | } finally {
89 | this.refreshing = false;
90 | }
91 | };
92 | }
93 |
--------------------------------------------------------------------------------
/server/src/leader_tracker.ts:
--------------------------------------------------------------------------------
1 | import { Connection } from "@solana/web3.js";
2 |
3 | const MAX_RECENT_SLOTS_LENGTH = 12;
4 |
5 | // 48 chosen because it's unlikely that 12 leaders in a row will miss their slots
6 | const MAX_SLOT_SKIP_DISTANCE = 48;
7 |
8 | export default class LeaderTrackerService {
9 | private recentSlots: Array = [];
10 |
11 | constructor(
12 | private connection: Connection,
13 | private currentSlot: number,
14 | callback: (slot: number) => Promise
15 | ) {
16 | this.recentSlots.push(currentSlot);
17 |
18 | let receivedShredNotification = false;
19 | let processingCallback = false;
20 | this.connection.onSlotUpdate((update) => {
21 | const previousCurrentSlot = this.currentSlot;
22 | let newCurrentSlot = this.currentSlot;
23 | switch (update.type) {
24 | case "firstShredReceived": {
25 | receivedShredNotification = true;
26 | newCurrentSlot = this.updateRecentSlots(update.slot);
27 | break;
28 | }
29 | case "completed": {
30 | receivedShredNotification = true;
31 | newCurrentSlot = this.updateRecentSlots(update.slot + 1);
32 | break;
33 | }
34 | case "createdBank": {
35 | // Fallback to bank created slot updates if no shred notifications
36 | // are received (ie. connected to single node cluster leader).
37 | if (!receivedShredNotification) {
38 | newCurrentSlot = this.updateRecentSlots(update.slot);
39 | }
40 | break;
41 | }
42 | }
43 |
44 | if (newCurrentSlot != previousCurrentSlot) {
45 | // console.debug(`Leader tracker detected new slot: ${newCurrentSlot}`);
46 | if (!processingCallback) {
47 | processingCallback = true;
48 | callback(newCurrentSlot)
49 | .then(() => {
50 | // console.debug(
51 | // `Leader tracker handled new slot: ${newCurrentSlot}`
52 | // );
53 | })
54 | .catch((err) => {
55 | console.error("Failed to handle new slot", err);
56 | })
57 | .finally(() => {
58 | processingCallback = false;
59 | });
60 | }
61 | }
62 | });
63 | }
64 |
65 | private updateRecentSlots = (slot: number) => {
66 | this.recentSlots.push(slot);
67 | while (this.recentSlots.length > MAX_RECENT_SLOTS_LENGTH) {
68 | this.recentSlots.shift();
69 | }
70 |
71 | // After updating recent slots, calculate the current slot
72 |
73 | const recentSlots = this.recentSlots.slice(0);
74 | recentSlots.sort();
75 |
76 | // Validators can broadcast invalid blocks that are far in the future
77 | // so check if the current slot is in line with the recent progression.
78 | const maxIndex = recentSlots.length - 1;
79 | const medianIndex = Math.floor(maxIndex / 2);
80 | const medianRecentSlot = recentSlots[medianIndex];
81 | const expectedCurrentSlot = medianRecentSlot + (maxIndex - medianIndex);
82 | const maxReasonableCurrentSlot =
83 | expectedCurrentSlot + MAX_SLOT_SKIP_DISTANCE;
84 |
85 | // Return the highest slot that doesn't exceed what we believe is a
86 | // reasonable slot.
87 | recentSlots.reverse();
88 | for (const slot of recentSlots) {
89 | if (slot <= maxReasonableCurrentSlot) {
90 | return slot;
91 | }
92 | }
93 |
94 | // This fall back is impossible
95 | return this.currentSlot;
96 | };
97 | }
98 |
--------------------------------------------------------------------------------
/server/src/program.ts:
--------------------------------------------------------------------------------
1 | import { Keypair, PublicKey } from "@solana/web3.js";
2 | import path from "path";
3 | import fs from "fs";
4 |
5 | const DEPLOYED_PROGRAM_ADDRESS = process.env.DEPLOYED_PROGRAM_ADDRESS;
6 |
7 | const PROGRAM_KEYPAIR_PATH = path.resolve(
8 | "..",
9 | "program",
10 | "target",
11 | "deploy",
12 | "break_solana_program-keypair.json"
13 | );
14 |
15 | export const PROGRAM_ID = (() => {
16 | if (DEPLOYED_PROGRAM_ADDRESS) {
17 | return new PublicKey(DEPLOYED_PROGRAM_ADDRESS);
18 | } else if (!process.env.DISABLE_API) {
19 | return readKeypairFromFile(PROGRAM_KEYPAIR_PATH).publicKey;
20 | }
21 | })();
22 |
23 | /**
24 | * Create a Keypair from a keypair file
25 | */
26 | function readKeypairFromFile(filePath: string): Keypair {
27 | const keypairString = fs.readFileSync(filePath, { encoding: "utf8" });
28 | const keypairBuffer = Buffer.from(JSON.parse(keypairString));
29 | return Keypair.fromSecretKey(keypairBuffer);
30 | }
31 |
--------------------------------------------------------------------------------
/server/src/urls.ts:
--------------------------------------------------------------------------------
1 | // To connect to a public cluster, set `export LIVE=1` in your
2 | // environment. By default, `LIVE=1` will connect to the devnet cluster.
3 |
4 | import { clusterApiUrl, Cluster } from "@solana/web3.js";
5 |
6 | function chooseCluster(): Cluster | undefined {
7 | if (!process.env.LIVE) return;
8 | switch (process.env.CLUSTER) {
9 | case "devnet":
10 | case "testnet":
11 | case "mainnet-beta": {
12 | return process.env.CLUSTER;
13 | }
14 | }
15 | return "devnet";
16 | }
17 |
18 | export const cluster = chooseCluster();
19 |
20 | export const url =
21 | process.env.RPC_URL ||
22 | (process.env.LIVE ? clusterApiUrl(cluster, false) : "http://localhost:8899");
23 |
24 | export const urlTls =
25 | process.env.RPC_URL ||
26 | (process.env.LIVE ? clusterApiUrl(cluster, true) : "http://localhost:8899");
27 |
--------------------------------------------------------------------------------
/server/src/utils.ts:
--------------------------------------------------------------------------------
1 | import * as Sentry from "@sentry/node";
2 |
3 | export function sleep(ms: number): Promise {
4 | return new Promise((resolve) => setTimeout(resolve, ms));
5 | }
6 |
7 | export function notUndefined(x: T | undefined): x is T {
8 | return x !== undefined;
9 | }
10 |
11 | export async function endlessRetry(
12 | name: string,
13 | call: () => Promise
14 | ): Promise {
15 | let result: T | undefined;
16 | while (result == undefined) {
17 | try {
18 | console.log(name, "fetching");
19 | result = await call();
20 | } catch (err) {
21 | reportError(err, `Request ${name} failed, retrying`);
22 | await sleep(1000);
23 | }
24 | }
25 | console.log(name, "fetched!");
26 | return result;
27 | }
28 |
29 | const cluster = process.env.CLUSTER;
30 | export function reportError(err: unknown, context: string): void {
31 | if (err instanceof Error) {
32 | console.error(context, err);
33 | if (process.env.NODE_ENV === "production") {
34 | Sentry.captureException(err, {
35 | tags: { context, cluster },
36 | });
37 | }
38 | }
39 | }
40 |
--------------------------------------------------------------------------------
/server/src/websocket.ts:
--------------------------------------------------------------------------------
1 | import WebSocket from "ws";
2 | import http from "http";
3 | import TpuProxy from "./tpu_proxy";
4 |
5 | // eslint-disable-next-line @typescript-eslint/no-empty-function
6 | function noop() {}
7 |
8 | export default class WebSocketServer {
9 | static start(httpServer: http.Server, tpuProxy: TpuProxy): void {
10 | // Start websocket server
11 | let activeUsers = 0;
12 | const wss = new WebSocket.Server({ server: httpServer });
13 | wss.on("connection", function connection(ws) {
14 | let isAlive = true;
15 | let useRpc = true;
16 | let rpcOverride: string;
17 |
18 | function heartbeat() {
19 | isAlive = true;
20 | }
21 |
22 | const interval = setInterval(function ping() {
23 | if (isAlive === false) return ws.terminate();
24 | isAlive = false;
25 | ws.ping(noop);
26 | }, 30000);
27 |
28 | activeUsers++;
29 | ws.on("close", () => {
30 | clearInterval(interval);
31 | activeUsers--;
32 | });
33 | ws.on("message", (data: Buffer, isBinary: boolean) => {
34 | const message = isBinary ? data : data.toString();
35 | if (typeof message === "string") {
36 | if (message === "tpu") {
37 | console.log("Client switched to TPU mode");
38 | useRpc = false;
39 | } else if (message === "rpc") {
40 | console.log("Client switched to RPC mode");
41 | useRpc = true;
42 | } else {
43 | try {
44 | const rpcEndpoint = new URL(message);
45 | rpcOverride = rpcEndpoint.toString();
46 | console.log("Client overrode RPC endpoint to", rpcEndpoint.href);
47 | } catch (err) {
48 | console.warn("Ignoring client message", message);
49 | }
50 | }
51 | } else {
52 | tpuProxy.sendRawTransaction(
53 | message,
54 | useRpc,
55 | rpcOverride,
56 | (message: string) => {
57 | ws.send(message);
58 | }
59 | );
60 | }
61 | });
62 | ws.on("pong", heartbeat);
63 | });
64 |
65 | // Start active user broadcast loop
66 | setInterval(() => {
67 | wss.clients.forEach((client) => {
68 | if (client.readyState === WebSocket.OPEN) {
69 | client.send(JSON.stringify({ type: "heartbeat", activeUsers }));
70 | }
71 | });
72 | }, 1000);
73 | }
74 | }
75 |
--------------------------------------------------------------------------------
/server/tsconfig.json:
--------------------------------------------------------------------------------
1 | {
2 | "compilerOptions": {
3 | "strict": true,
4 | "esModuleInterop": true,
5 | "downlevelIteration": true,
6 | "module": "commonjs",
7 | "moduleResolution": "node",
8 | "baseUrl": "./",
9 | "typeRoots": ["./src/types", "./node_modules/@types"],
10 | "paths": {
11 | "@/*": ["./src/*"]
12 | }
13 | },
14 | "include": ["src"]
15 | }
16 |
--------------------------------------------------------------------------------
/server/webpack.config.js:
--------------------------------------------------------------------------------
1 | const webpack = require("webpack");
2 | const path = require("path");
3 | const nodeExternals = require("webpack-node-externals");
4 |
5 | module.exports = (env, argv) => {
6 | let mode = argv.mode;
7 | let config = {
8 | mode,
9 | target: "node",
10 | entry: ["./src/index.ts"],
11 | node: {
12 | __dirname: false,
13 | },
14 | output: {
15 | path: path.resolve(__dirname, "dist"),
16 | filename: "server.js",
17 | },
18 | module: {
19 | rules: [
20 | {
21 | test: /\.ts$/,
22 | use: "ts-loader",
23 | exclude: /node_modules/,
24 | },
25 | ],
26 | },
27 | resolve: {
28 | extensions: [".ts", ".js", ".json"],
29 | },
30 | externals: [nodeExternals()],
31 | };
32 |
33 | if (mode === "development") {
34 | config.watch = true;
35 | }
36 |
37 | return config;
38 | };
39 |
--------------------------------------------------------------------------------