diff --git a/.eslintrc.js b/.eslintrc.js index fcfe8fb..457ff32 100644 --- a/.eslintrc.js +++ b/.eslintrc.js @@ -6,4 +6,4 @@ module.exports = { parserOptions: { project: false, }, -} +}; diff --git a/.github/workflows/deploy-staging.yml b/.github/workflows/deploy-staging.yml index deed27a..7155591 100644 --- a/.github/workflows/deploy-staging.yml +++ b/.github/workflows/deploy-staging.yml @@ -6,7 +6,7 @@ name: "Staging" on: push: - branches: [ "master" ] + branches: ["master"] workflow_dispatch: concurrency: @@ -23,7 +23,7 @@ jobs: packages: write # Needed to push to the GitHub Container Registry strategy: matrix: - component: [ "api", "web" ] + component: ["api", "web"] with: version: "${{ github.sha }}" context: "." @@ -55,7 +55,7 @@ jobs: deploy: name: "Deploy to staging" uses: hemilabs/actions/.github/workflows/deploy-kustomize.yml@main - needs: [ "docker", "deploy-registry" ] + needs: ["docker", "deploy-registry"] permissions: contents: read with: diff --git a/.github/workflows/deploy-testnet.yml b/.github/workflows/deploy-testnet.yml index 6892ada..02bb82d 100644 --- a/.github/workflows/deploy-testnet.yml +++ b/.github/workflows/deploy-testnet.yml @@ -48,13 +48,13 @@ jobs: docker: name: "Docker (${{ matrix.component }})" uses: hemilabs/actions/.github/workflows/docker.yml@main - needs: [ "prepare" ] + needs: ["prepare"] permissions: contents: read packages: write # Needed to push to the GitHub Container Registry strategy: matrix: - component: [ "api", "web" ] + component: ["api", "web"] with: version: "${{ needs.prepare.outputs.version }}" context: "." @@ -90,7 +90,7 @@ jobs: deploy: name: "Deploy to testnet" uses: hemilabs/actions/.github/workflows/deploy-kustomize.yml@main - needs: [ "prepare", "docker", "deploy-registry" ] + needs: ["prepare", "docker", "deploy-registry"] permissions: contents: read with: diff --git a/.github/workflows/js-checks.yml b/.github/workflows/js-checks.yml new file mode 100644 index 0000000..c041af6 --- /dev/null +++ b/.github/workflows/js-checks.yml @@ -0,0 +1,58 @@ +name: JS Checks + +on: + pull_request: + push: + +concurrency: + group: ${{ github.workflow }}-${{ github.ref }}-${{ github.event_name }} + cancel-in-progress: true + +jobs: + build-shared: + runs-on: ubuntu-latest + steps: + - name: Checkout Repository + uses: actions/checkout@v3 + + - name: Install Dependencies + run: npm install --workspaces --include-workspace-root + + - name: Build Shared Package + run: npm run build --workspaces --include-workspace-root + + - name: Save Build Output + run: tar -czf shared-build.tar.gz packages/shared/dist + + - name: Upload Build Artifact + uses: actions/upload-artifact@v3 + with: + name: shared-build + path: shared-build.tar.gz + + prepare-environment: + needs: build-shared + runs-on: ubuntu-latest + steps: + - name: Checkout Repository + uses: actions/checkout@v3 + + - name: Download Shared Build + uses: actions/download-artifact@v3 + with: + name: shared-build + + - name: Create Directory for Shared Build + run: mkdir -p packages/shared/dist + + - name: Extract Shared Build + run: tar -xzf shared-build.tar.gz -C packages/shared/dist + + - name: Install Dependencies + run: npm install --workspaces --include-workspace-root + + js-checks: + needs: prepare-environment + uses: hemilabs/actions/.github/workflows/js-checks.yml@main + with: + node-versions: '["16", "18", "20", "22"]' diff --git a/.nvmrc b/.nvmrc new file mode 100644 index 0000000..2edeafb --- /dev/null +++ b/.nvmrc @@ -0,0 +1 @@ +20 \ No newline at end of file diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index 7f8b233..f29e74e 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -119,4 +119,3 @@ While the prerequisites above must be satisfied prior to having your pull reques | `bug` | Confirmed bugs or reports that are very likely to be bugs. | | `question` | Questions more than bug reports or feature requests (e.g. how do I do X). | | `feedback` | General feedback more than bug reports or feature requests. | - diff --git a/CodingChallenge.md b/CodingChallenge.md index 32e6e93..7ce51f6 100644 --- a/CodingChallenge.md +++ b/CodingChallenge.md @@ -1,16 +1,21 @@ # Coding Challenge Submission

## Description + Describe your project here ## Prerequisites, Dependencies, Versions + Please add anything here you've built that requires the above specifics ## Licensing + Please state if license is CC, MIT or ISC or other (please no GPL) ## Outside Libraries + Link any outside open source libraries used ## Steps + Tell us how to run it locally diff --git a/README.md b/README.md index 67fab62..e4ad103 100644 --- a/README.md +++ b/README.md @@ -1,9 +1,11 @@ # CryptoChords + CryptoChords is a tool for visualizing activities on the [Hemi network](https://hemi.xyz). It features a web application that enables users to explore the network and visualize transactions in a user-friendly manner. This document provides instructions on how to set up, run, and contribute to the project. ## Visual Representation of the Network + CryptoChords provides a visual representation of the Hemi network using a piano keyboard as a metaphor. Every time a new transaction is added to the network, a cube appears on a piano key, representing the transaction. The transaction details are displayed on the screen, and if sound is enabled, the corresponding piano note is played simultaneously. There are four types of transactions, each represented by a different color and associated with a different quadrant of the piano keyboard: @@ -15,8 +17,8 @@ There are four types of transactions, each represented by a different color and A random key within the corresponding quadrant is selected to represent each transaction. - ## Folder Structure + The project is organized into the following folders: - `apps`: Contains the API and the Web application. @@ -24,6 +26,7 @@ The project is organized into the following folders: - `infrastucture`: Contains the necessary files to deploy the project. ## Architecture + This project is a monorepo composed of two applications: the API and the Web application and it is managed by [Turborepo](https://turbo.build/repo/docs). Both the API and the Web application follow the Domain-Driven Design (DDD) principles. You can find more information about the architecture in the `readme.md` file of each project. @@ -85,18 +88,23 @@ npm run lint ``` ## Deployment + We have two environments: `stage` and `production`. -The project is automatically deployed using GitHub Actions. +The project is automatically deployed using GitHub Actions. ### Stage + The `stage` environment is automatically deployed when a new tag is created. The tag must follow the pattern `v*.*.*`. For example, `v1.0.0`. ### Production + The `production` environment is automatically deployed when a Commit is pushed to the `main` branch. ## Contributing + If you want to contribute to this project and make it better, your help is very welcome. You can find more information about how to contribute in the [`CONTRIBUTING.md`](./CONTRIBUTING.md) file. ## License + This project is licensed under the MIT License - see the [`LICENSE`](./LICENSE) file for details. diff --git a/apps/api/.eslintrc.js b/apps/api/.eslintrc.js deleted file mode 100644 index d961833..0000000 --- a/apps/api/.eslintrc.js +++ /dev/null @@ -1,10 +0,0 @@ -module.exports = { - root: true, - env: { browser: true, es2020: true }, - extends: [ - 'eslint:recommended', - 'plugin:@typescript-eslint/recommended' - ], - ignorePatterns: ['dist', '.eslintrc.js', '**/*.spec.ts'], - parser: '@typescript-eslint/parser', -} diff --git a/apps/api/.eslintrc.json b/apps/api/.eslintrc.json new file mode 100644 index 0000000..96a012c --- /dev/null +++ b/apps/api/.eslintrc.json @@ -0,0 +1,26 @@ +{ + "extends": ["bloq", "prettier"], + "ignorePatterns": ["dist", "_esm/*", "_cjs/*", "_types/*"], + "overrides": [ + { + "extends": ["bloq/typescript", "prettier"], + "files": ["src/**/*.ts"] + }, + { + "extends": ["bloq/markdown"], + "files": ["*.md"] + }, + { + "extends": ["bloq/vitest", "prettier"], + "files": ["*.spec.{js,ts}"] + } + ], + "parserOptions": { + "sourceType": "module" + }, + "root": true, + "rules": { + "arrow-body-style": "off", + "no-console": "off" + } +} diff --git a/apps/api/.prettierrc.json b/apps/api/.prettierrc.json new file mode 100644 index 0000000..21c021c --- /dev/null +++ b/apps/api/.prettierrc.json @@ -0,0 +1,7 @@ +{ + "arrowParens": "avoid", + "quoteProps": "consistent", + "semi": false, + "singleQuote": true, + "tabWidth": 2 +} diff --git a/apps/api/README.md b/apps/api/README.md index 6348138..cb2a5b4 100644 --- a/apps/api/README.md +++ b/apps/api/README.md @@ -90,7 +90,6 @@ The environment variables are defined in the `.env` file. The following variable - `ENABLE_MAINNET`: A boolean that enables mainnet.[Hemi Docs](https://github.com/hemilabs/infrastructure/blob/main/NETWORK_INFO.md). - Example of the .env file ``` @@ -99,8 +98,10 @@ ENABLE_MAINNET=false ``` ## Contribution + If you want to contribute to this project and make it better, your help is very welcome. You can find more information about how to contribute in the [`CONTRIBUTING.md`](../../CONTRIBUTING.md) file. ## License + This project is licensed under the MIT License - see the [LICENSE](../../LICENSE) file for details. diff --git a/apps/api/package.json b/apps/api/package.json index 1295926..5efe6f6 100644 --- a/apps/api/package.json +++ b/apps/api/package.json @@ -8,15 +8,16 @@ "test:cov": "vitest run src --coverage", "test:watch": "vitest watch src", "dev": "nodemon", + "format:check": "prettier --check .", + "format:fix": "npx prettier --write .", "build": "tsc -p tsconfig.build.json", - "lint": "eslint . --ext ts --report-unused-disable-directives --max-warnings 0", + "lint": "eslint . --ext ts", "start:api": "node dist/server.js" }, "author": "", "license": "MIT", "dependencies": { "@cryptochords/shared": "*", - "@typescript-eslint/eslint-plugin": "8.18.0", "dotenv": "16.4.7", "express": "4.21.2", "web3": "4.16.0" @@ -27,8 +28,12 @@ "@types/node": "22.10.2", "@types/ws": "8.5.13", "@vitest/coverage-v8": "2.1.8", - "eslint": "8.56.0", + "@typescript-eslint/eslint-plugin": "^8.18.2", + "@typescript-eslint/parser": "^8.18.2", + "eslint": "^8.57.1", + "eslint-config-bloq": "^4.4.1", "nodemon": "3.1.9", + "prettier": "^3.3.3", "ts-node": "10.9.2", "typescript": "5.3.3", "vitest": "2.1.8" diff --git a/apps/api/src/application/helpers/BroadcastToClients.ts b/apps/api/src/application/helpers/BroadcastToClients.ts index 25fb9bc..5b21c7e 100644 --- a/apps/api/src/application/helpers/BroadcastToClients.ts +++ b/apps/api/src/application/helpers/BroadcastToClients.ts @@ -1,12 +1,12 @@ -import WebSocket, { WebSocketServer } from 'ws'; -import { L2Block } from '@cryptochords/shared'; +import WebSocket, { WebSocketServer } from 'ws' +import { L2Block } from '@cryptochords/shared' -const BroadcastToClients = (wss: WebSocketServer, l2Block: L2Block): void => { +const broadcastToClients = (wss: WebSocketServer, l2Block: L2Block): void => { wss.clients.forEach((client: WebSocket) => { if (client.readyState === WebSocket.OPEN) { - client.send(JSON.stringify(l2Block.toJSON())); + client.send(JSON.stringify(l2Block.toJSON())) } - }); -}; + }) +} -export default BroadcastToClients +export default broadcastToClients diff --git a/apps/api/src/application/helpers/broadcastToClients.spec.ts b/apps/api/src/application/helpers/broadcastToClients.spec.ts index ff56996..47de08d 100644 --- a/apps/api/src/application/helpers/broadcastToClients.spec.ts +++ b/apps/api/src/application/helpers/broadcastToClients.spec.ts @@ -1,12 +1,12 @@ /* eslint-disable @typescript-eslint/no-explicit-any */ -import { describe, it, expect, vi } from 'vitest'; -import WebSocket, { WebSocketServer } from 'ws'; -import BroadcastToClients from './BroadcastToClients'; +import { describe, it, expect, vi } from 'vitest' +import WebSocket, { WebSocketServer } from 'ws' +import broadcastToClients from './BroadcastToClients' describe('BroadcastToClients', () => { it('should send a message to all connected clients', () => { - const mockSend = vi.fn(); - const wss: any = new WebSocketServer({ noServer: true }); + const mockSend = vi.fn() + const wss: any = new WebSocketServer({ noServer: true }) wss.clients = new Set([ { readyState: WebSocket.OPEN, @@ -16,15 +16,15 @@ describe('BroadcastToClients', () => { readyState: WebSocket.CLOSED, send: mockSend, }, - ]); + ]) const l2Block = { toJSON: () => ({ key: 'value' }), - }; + } - BroadcastToClients(wss, l2Block as any); + broadcastToClients(wss, l2Block as any) - expect(mockSend).toHaveBeenCalledTimes(1); - expect(mockSend).toHaveBeenCalledWith(JSON.stringify({ key: 'value' })); - }); -}); + expect(mockSend).toHaveBeenCalledTimes(1) + expect(mockSend).toHaveBeenCalledWith(JSON.stringify({ key: 'value' })) + }) +}) diff --git a/apps/api/src/application/polling/pollingService.spec.ts b/apps/api/src/application/polling/pollingService.spec.ts index 42682ae..6337e66 100644 --- a/apps/api/src/application/polling/pollingService.spec.ts +++ b/apps/api/src/application/polling/pollingService.spec.ts @@ -1,49 +1,49 @@ -import { describe, it, expect, vi, beforeEach } from 'vitest'; -import { WebSocketServer } from 'ws'; -import { EventEmitter } from 'events'; -import BroadcastToClients from '../helpers/BroadcastToClients'; -import { TxTypesEnum } from '@cryptochords/shared'; -import { PollingService } from './pollingService'; - -vi.mock('ws'); +import { describe, it, expect, vi, beforeEach } from 'vitest' +import { WebSocketServer } from 'ws' +import { EventEmitter } from 'events' +import broadcastToClients from '../helpers/broadcastToClients' +import { TxTypesEnum } from '@cryptochords/shared' +import { PollingService } from './pollingService' + +vi.mock('ws') vi.mock('../helpers/BroadcastToClients', () => ({ __esModule: true, default: vi.fn(), -})); +})) class MockBlockRepository extends EventEmitter { - execute = vi.fn(); - stop = vi.fn(); + execute = vi.fn() + stop = vi.fn() } describe('PollingService', () => { - let pollingService: PollingService; - let mockBlockRepository: MockBlockRepository; - let mockWss: WebSocketServer; + let pollingService: PollingService + let mockBlockRepository: MockBlockRepository + let mockWss: WebSocketServer beforeEach(() => { - mockBlockRepository = new MockBlockRepository(); - mockWss = new WebSocketServer({ noServer: true }); - pollingService = new PollingService(mockBlockRepository); - }); + mockBlockRepository = new MockBlockRepository() + mockWss = new WebSocketServer({ noServer: true }) + pollingService = new PollingService(mockBlockRepository) + }) it('should execute and handle Block and Eth events', () => { - const websocketUrl = 'ws://test.url'; - const mockL2Block = { key: 'Value' }; + const websocketUrl = 'ws://test.url' + const mockL2Block = { key: 'Value' } - pollingService.execute(mockWss, websocketUrl); + pollingService.execute(mockWss, websocketUrl) - expect(mockBlockRepository.execute).toHaveBeenCalledWith(websocketUrl); + expect(mockBlockRepository.execute).toHaveBeenCalledWith(websocketUrl) - mockBlockRepository.emit(TxTypesEnum.Block, mockL2Block); - mockBlockRepository.emit(TxTypesEnum.Eth, mockL2Block); + mockBlockRepository.emit(TxTypesEnum.Block, mockL2Block) + mockBlockRepository.emit(TxTypesEnum.Eth, mockL2Block) - expect(BroadcastToClients).toHaveBeenCalledWith(mockWss, mockL2Block); - expect(BroadcastToClients).toHaveBeenCalledTimes(2); - }); + expect(broadcastToClients).toHaveBeenCalledWith(mockWss, mockL2Block) + expect(broadcastToClients).toHaveBeenCalledTimes(2) + }) it('should stop the repository on stop', () => { - pollingService.stop(); - expect(mockBlockRepository.stop).toHaveBeenCalled(); - }); -}); + pollingService.stop() + expect(mockBlockRepository.stop).toHaveBeenCalled() + }) +}) diff --git a/apps/api/src/application/polling/pollingService.ts b/apps/api/src/application/polling/pollingService.ts index fcd1666..383b524 100644 --- a/apps/api/src/application/polling/pollingService.ts +++ b/apps/api/src/application/polling/pollingService.ts @@ -1,19 +1,23 @@ -import { WebSocketServer } from "ws"; -import { BlockRepository } from "../../domain/repositories/BlockRepository"; -import BroadcastToClients from "../helpers/BroadcastToClients"; -import { TxTypesEnum } from "@cryptochords/shared"; -import { L2Block } from "@cryptochords/shared"; +import { WebSocketServer } from 'ws' +import { BlockRepository } from '../../domain/repositories/BlockRepository' +import broadcastToClients from '../helpers/BroadcastToClients' +import { TxTypesEnum } from '@cryptochords/shared' +import { L2Block } from '@cryptochords/shared' export class PollingService { constructor(private blockRepository: BlockRepository) {} execute(wss: WebSocketServer, url: string): void { this.blockRepository.execute(url) - this.blockRepository.on(TxTypesEnum.Block, (l2Block: L2Block) => BroadcastToClients(wss, l2Block)); - this.blockRepository.on(TxTypesEnum.Eth, (l2Block: L2Block) => BroadcastToClients(wss, l2Block)); + this.blockRepository.on(TxTypesEnum.Block, (l2Block: L2Block) => + broadcastToClients(wss, l2Block), + ) + this.blockRepository.on(TxTypesEnum.Eth, (l2Block: L2Block) => + broadcastToClients(wss, l2Block), + ) } stop() { this.blockRepository.stop() } -} \ No newline at end of file +} diff --git a/apps/api/src/domain/base/Entity.spec.ts b/apps/api/src/domain/base/Entity.spec.ts index b572ccc..e1c1caa 100644 --- a/apps/api/src/domain/base/Entity.spec.ts +++ b/apps/api/src/domain/base/Entity.spec.ts @@ -22,7 +22,7 @@ describe('src/domain/base/Entity', () => { }) it('should be an instance of ValueObject', () => { - const entity = TestEntity.create({ value: 'test'}) + const entity = TestEntity.create({ value: 'test' }) expect(entity).toBeInstanceOf(ValueObject) }) diff --git a/apps/api/src/domain/enums/BlockTypesEnum.spec.ts b/apps/api/src/domain/enums/BlockTypesEnum.spec.ts index 68d7a92..23bcbb5 100644 --- a/apps/api/src/domain/enums/BlockTypesEnum.spec.ts +++ b/apps/api/src/domain/enums/BlockTypesEnum.spec.ts @@ -1,16 +1,16 @@ -import { describe, it, expect } from 'vitest'; -import { BlockTypesEnum } from './BlockTypesEnum'; +import { describe, it, expect } from 'vitest' +import { BlockTypesEnum } from './BlockTypesEnum' describe('BlockTypesEnum', () => { it('should have a LEGACY type with value "0"', () => { - expect(BlockTypesEnum.LEGACY).toBe('0'); - }); + expect(BlockTypesEnum.LEGACY).toBe('0') + }) it('should have an EIP2930 type with value "1"', () => { - expect(BlockTypesEnum.EIP2930).toBe('1'); - }); + expect(BlockTypesEnum.EIP2930).toBe('1') + }) it('should have an EIP1559 type with value "2"', () => { - expect(BlockTypesEnum.EIP1559).toBe('2'); - }); -}); + expect(BlockTypesEnum.EIP1559).toBe('2') + }) +}) diff --git a/apps/api/src/domain/repositories/BlockRepository.ts b/apps/api/src/domain/repositories/BlockRepository.ts index 019cb6a..816c9ac 100644 --- a/apps/api/src/domain/repositories/BlockRepository.ts +++ b/apps/api/src/domain/repositories/BlockRepository.ts @@ -1,4 +1,4 @@ -import { EventEmitter } from 'events'; +import { EventEmitter } from 'events' export interface BlockRepository extends EventEmitter { execute(url: string): void diff --git a/apps/api/src/infrastructure/repositories/blockPolling.spec.ts b/apps/api/src/infrastructure/repositories/blockPolling.spec.ts index ba45d8f..44340f0 100644 --- a/apps/api/src/infrastructure/repositories/blockPolling.spec.ts +++ b/apps/api/src/infrastructure/repositories/blockPolling.spec.ts @@ -1,9 +1,15 @@ -import Web3 from 'web3'; -import { beforeEach, describe, expect, it, vi, afterEach } from 'vitest'; -import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum'; -import { BlockPollingRepository } from './blockPolling'; +import Web3 from 'web3' +import { beforeEach, describe, expect, it, vi, afterEach } from 'vitest' +import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum' +import { BlockPollingRepository } from './blockPolling' vi.mock('@cryptochords/shared', () => ({ + Address: { + create: vi.fn().mockReturnValue({}), + }, + L2Block: { + create: vi.fn().mockReturnValue({}), + }, TxType: { create: vi.fn().mockReturnValue({}), }, @@ -11,59 +17,60 @@ vi.mock('@cryptochords/shared', () => ({ Block: 'Block', Eth: 'Eth', }, - Address: { - create: vi.fn().mockReturnValue({}), - }, - L2Block: { - create: vi.fn().mockReturnValue({}), - }, -})); +})) describe('BlockPollingRepository', () => { - let blockPollingRepository: BlockPollingRepository; - let mockWeb3: Web3; + let blockPollingRepository: BlockPollingRepository + let mockWeb3: Web3 beforeEach(() => { - blockPollingRepository = new BlockPollingRepository(); - mockWeb3 = new Web3(); - mockWeb3.eth.getBlockNumber = vi.fn().mockResolvedValue(100); + blockPollingRepository = new BlockPollingRepository() + mockWeb3 = new Web3() + mockWeb3.eth.getBlockNumber = vi.fn().mockResolvedValue(100) mockWeb3.eth.getBlock = vi.fn().mockResolvedValue({ hash: '0xhash', transactions: [], - }); - vi.useFakeTimers(); + }) + vi.useFakeTimers() - vi.spyOn(blockPollingRepository, 'emit'); - }); + vi.spyOn(blockPollingRepository, 'emit') + }) afterEach(() => { - vi.runOnlyPendingTimers(); - vi.useRealTimers(); - vi.clearAllMocks(); - }); + vi.runOnlyPendingTimers() + vi.useRealTimers() + vi.clearAllMocks() + }) it('should start and stop polling correctly', () => { - blockPollingRepository.execute('http://localhost:8545', 1000); - expect(vi.getTimerCount()).toBe(1); - blockPollingRepository.stop(); - expect(vi.getTimerCount()).toBe(0); - }); + blockPollingRepository.execute('http://localhost:8545', 1000) + expect(vi.getTimerCount()).toBe(1) + blockPollingRepository.stop() + expect(vi.getTimerCount()).toBe(0) + }) it('should emit "Block" event for new blocks with no transactions', async () => { - await blockPollingRepository.checkNewBlocks(mockWeb3); - expect(blockPollingRepository.emit).toHaveBeenCalledWith('Block', expect.anything()); - }); + await blockPollingRepository.checkNewBlocks(mockWeb3) + expect(blockPollingRepository.emit).toHaveBeenCalledWith( + 'Block', + expect.anything(), + ) + }) it('should emit "Block" and "Eth" events for new blocks with EIP1559 transactions', async () => { mockWeb3.eth.getBlock = vi.fn().mockResolvedValue({ hash: '0xnewhash', - transactions: [ - { type: BlockTypesEnum.EIP1559, from: '0xSomeAddress' }, - ], - }); + transactions: [{ from: '0xSomeAddress', type: BlockTypesEnum.EIP1559 }], + }) - await blockPollingRepository.checkNewBlocks(mockWeb3); - expect(blockPollingRepository.emit).toHaveBeenCalledWith('Block', expect.anything()); - expect(blockPollingRepository.emit).toHaveBeenCalledWith('Eth', expect.anything()); - }); -}); + await blockPollingRepository.checkNewBlocks(mockWeb3) + expect(blockPollingRepository.emit).toHaveBeenCalledWith( + 'Block', + expect.anything(), + ) + expect(blockPollingRepository.emit).toHaveBeenCalledWith( + 'Eth', + expect.anything(), + ) + }) +}) diff --git a/apps/api/src/infrastructure/repositories/blockPolling.ts b/apps/api/src/infrastructure/repositories/blockPolling.ts index 60f144e..20c8ad0 100644 --- a/apps/api/src/infrastructure/repositories/blockPolling.ts +++ b/apps/api/src/infrastructure/repositories/blockPolling.ts @@ -1,61 +1,69 @@ -import { EventEmitter } from "stream"; -import { BlockRepository } from "../../domain/repositories/BlockRepository"; -import Web3 from "web3"; -import { TxTypesEnum } from "@cryptochords/shared"; -import { TxType } from "@cryptochords/shared"; -import { Address } from "@cryptochords/shared"; -import { BlockTypesEnum } from "../../domain/enums/BlockTypesEnum"; -import { L2Block } from "@cryptochords/shared"; - -export class BlockPollingRepository extends EventEmitter implements BlockRepository { +import { EventEmitter } from 'stream' +import { BlockRepository } from '../../domain/repositories/BlockRepository' +import Web3 from 'web3' +import { TxTypesEnum } from '@cryptochords/shared' +import { TxType } from '@cryptochords/shared' +import { Address } from '@cryptochords/shared' +import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum' +import { L2Block } from '@cryptochords/shared' + +export class BlockPollingRepository + extends EventEmitter + implements BlockRepository +{ private latestBlockNumber = BigInt(0) - private pollingIntervalId: NodeJS.Timeout | null = null; + private pollingIntervalId: NodeJS.Timeout | null = null execute(rpcUrl: string, pollingInterval = 5000): void { - const web3 = new Web3(rpcUrl); + const web3 = new Web3(rpcUrl) this.poll(web3, pollingInterval) } stop() { if (this.pollingIntervalId) { - clearInterval(this.pollingIntervalId); - this.pollingIntervalId = null; + clearInterval(this.pollingIntervalId) + this.pollingIntervalId = null } - console.log('Polling stopped.'); } async checkNewBlocks(web3: Web3): Promise { - console.log('Checking for new block'); - const currentBlockNumber = await web3.eth.getBlockNumber(); + const currentBlockNumber = await web3.eth.getBlockNumber() if (currentBlockNumber > this.latestBlockNumber) { - this.latestBlockNumber = currentBlockNumber; + this.latestBlockNumber = currentBlockNumber - const block = await web3.eth.getBlock(currentBlockNumber, true); + const block = await web3.eth.getBlock(currentBlockNumber, true) if (block) { - console.log('New block header:', block.hash); - this.emit(TxTypesEnum.Block, L2Block.create({ - txType: TxType.create(TxTypesEnum.Block), - address: Address.create(block.hash ? block.hash.toString() : '') - })); + this.emit( + TxTypesEnum.Block, + L2Block.create({ + address: Address.create(block.hash ? block.hash.toString() : ''), + txType: TxType.create(TxTypesEnum.Block), + }), + ) if (block.transactions) { // eslint-disable-next-line @typescript-eslint/no-explicit-any block.transactions.forEach((tx: any) => { if (tx.type.toString() === BlockTypesEnum.EIP1559) { - console.log('New eth tx from', tx.from); - this.emit(TxTypesEnum.Eth, L2Block.create({ - txType: TxType.create(TxTypesEnum.Eth), - address: Address.create(tx.from) - })); + this.emit( + TxTypesEnum.Eth, + L2Block.create({ + address: Address.create(tx.from), + txType: TxType.create(TxTypesEnum.Eth), + }), + ) } - }); + }) } } } } poll(web3: Web3, pollingInterval: number) { - this.pollingIntervalId = setInterval(() => this.checkNewBlocks(web3), pollingInterval); + this.pollingIntervalId = setInterval( + () => this.checkNewBlocks(web3), + pollingInterval, + ) } } diff --git a/apps/api/src/infrastructure/repositories/blockWebsocket.spec.ts b/apps/api/src/infrastructure/repositories/blockWebsocket.spec.ts index 22fdd12..11db960 100644 --- a/apps/api/src/infrastructure/repositories/blockWebsocket.spec.ts +++ b/apps/api/src/infrastructure/repositories/blockWebsocket.spec.ts @@ -1,28 +1,36 @@ -import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'; -import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum'; -import { BlockWebsocketRepository } from './blockWebsocket'; +import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest' +import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum' +import { BlockWebsocketRepository } from './blockWebsocket' vi.mock('web3', () => ({ __esModule: true, default: vi.fn().mockImplementation(() => ({ eth: { - subscribe: vi.fn().mockImplementation(() => Promise.resolve({ - on: vi.fn((event, callback) => { - if (event === 'data') { - const fakeBlockHeader = { hash: 'fakeHash' }; - callback(fakeBlockHeader); - } - }), - unsubscribe: vi.fn(), - })), getBlock: vi.fn().mockResolvedValue({ - transactions: [{ type: BlockTypesEnum.EIP1559, from: 'fakeAddress' }], + transactions: [{ from: 'fakeAddress', type: BlockTypesEnum.EIP1559 }], }), + subscribe: vi.fn().mockImplementation(() => + Promise.resolve({ + on: vi.fn((event, callback) => { + if (event === 'data') { + const fakeBlockHeader = { hash: 'fakeHash' } + callback(fakeBlockHeader) + } + }), + unsubscribe: vi.fn(), + }), + ), }, })), -})); +})) vi.mock('@cryptochords/shared', () => ({ + Address: { + create: vi.fn().mockReturnValue({}), + }, + L2Block: { + create: vi.fn().mockReturnValue({}), + }, TxType: { create: vi.fn().mockReturnValue({}), }, @@ -30,50 +38,54 @@ vi.mock('@cryptochords/shared', () => ({ Block: 'Block', Eth: 'Eth', }, - Address: { - create: vi.fn().mockReturnValue({}), - }, - L2Block: { - create: vi.fn().mockReturnValue({}), - }, -})); +})) describe('BlockWebsocketRepository', () => { - let blockWebsocketRepository: BlockWebsocketRepository; + let blockWebsocketRepository: BlockWebsocketRepository beforeEach(() => { - blockWebsocketRepository = new BlockWebsocketRepository(); - }); + blockWebsocketRepository = new BlockWebsocketRepository() + }) afterEach(() => { - vi.clearAllMocks(); - }); + vi.clearAllMocks() + }) it('should initialize and subscribe to new block headers', async () => { - blockWebsocketRepository.execute('fakeWebsocketUrl'); + blockWebsocketRepository.execute('fakeWebsocketUrl') await vi.waitFor(() => { - expect(blockWebsocketRepository['subscription']).toBeDefined(); - expect(blockWebsocketRepository['web3']).toBeDefined(); - expect(blockWebsocketRepository['subscription'].on).toHaveBeenCalledWith('data', expect.any(Function)); - }); - }); + expect(blockWebsocketRepository['subscription']).toBeDefined() + expect(blockWebsocketRepository['web3']).toBeDefined() + expect(blockWebsocketRepository['subscription'].on).toHaveBeenCalledWith( + 'data', + expect.any(Function), + ) + }) + }) it('should stop subscription', async () => { - blockWebsocketRepository.execute('fakeWebsocketUrl'); - await vi.waitFor(() => expect(blockWebsocketRepository['subscription']).toBeDefined()); + blockWebsocketRepository.execute('fakeWebsocketUrl') + await vi.waitFor(() => + expect(blockWebsocketRepository['subscription']).toBeDefined(), + ) - blockWebsocketRepository.stop(); - expect(blockWebsocketRepository['subscription']).toBeNull(); - }); + blockWebsocketRepository.stop() + expect(blockWebsocketRepository['subscription']).toBeNull() + }) it('should emit events for new block headers and transactions', async () => { - const emitSpy = vi.spyOn(blockWebsocketRepository, 'emit'); - blockWebsocketRepository.execute('fakeWebsocketUrl'); + const emitSpy = vi.spyOn(blockWebsocketRepository, 'emit') + blockWebsocketRepository.execute('fakeWebsocketUrl') - await vi.waitFor(() => expect(emitSpy).toHaveBeenCalledWith('Block', expect.anything())); - await vi.waitFor(() => expect(emitSpy).toHaveBeenCalledWith('Eth', expect.anything()), { timeout: 500 }); + await vi.waitFor(() => + expect(emitSpy).toHaveBeenCalledWith('Block', expect.anything()), + ) + await vi.waitFor( + () => expect(emitSpy).toHaveBeenCalledWith('Eth', expect.anything()), + { timeout: 500 }, + ) - expect(emitSpy).toHaveBeenNthCalledWith(1, 'Block', expect.anything()); - expect(emitSpy).toHaveBeenNthCalledWith(2, 'Eth', expect.anything()); - }); -}); + expect(emitSpy).toHaveBeenNthCalledWith(1, 'Block', expect.anything()) + expect(emitSpy).toHaveBeenNthCalledWith(2, 'Eth', expect.anything()) + }) +}) diff --git a/apps/api/src/infrastructure/repositories/blockWebsocket.ts b/apps/api/src/infrastructure/repositories/blockWebsocket.ts index 310d690..24ed6d3 100644 --- a/apps/api/src/infrastructure/repositories/blockWebsocket.ts +++ b/apps/api/src/infrastructure/repositories/blockWebsocket.ts @@ -1,60 +1,71 @@ -import Web3, { BlockHeaderOutput } from 'web3'; -import { EventEmitter } from 'events'; -import { TxType } from '@cryptochords/shared'; -import { TxTypesEnum } from '@cryptochords/shared'; -import { Address } from '@cryptochords/shared'; -import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum'; -import { BlockRepository } from '../../domain/repositories/BlockRepository'; -import { L2Block } from '@cryptochords/shared'; +import Web3, { BlockHeaderOutput } from 'web3' +import { EventEmitter } from 'events' +import { TxType } from '@cryptochords/shared' +import { TxTypesEnum } from '@cryptochords/shared' +import { Address } from '@cryptochords/shared' +import { BlockTypesEnum } from '../../domain/enums/BlockTypesEnum' +import { BlockRepository } from '../../domain/repositories/BlockRepository' +import { L2Block } from '@cryptochords/shared' -export class BlockWebsocketRepository extends EventEmitter implements BlockRepository { - private web3: Web3 | null = null; +export class BlockWebsocketRepository + extends EventEmitter + implements BlockRepository +{ + private web3: Web3 | null = null // eslint-disable-next-line @typescript-eslint/no-explicit-any - private subscription: any | null = null; + private subscription: any | null = null execute(websocketUrl: string): void { - this.web3 = new Web3(websocketUrl); - this.subscribeToNewBlockHeaders(); + this.web3 = new Web3(websocketUrl) + this.subscribeToNewBlockHeaders() } stop(): void { if (this.subscription) { - this.subscription.unsubscribe(); - this.subscription = null; + this.subscription.unsubscribe() + this.subscription = null } } private async subscribeToNewBlockHeaders(): Promise { if (!this.web3) { - console.error('Web3 is not initialized.'); - return; + return } - this.subscription = await this.web3?.eth.subscribe('newBlockHeaders'); + this.subscription = await this.web3?.eth.subscribe('newBlockHeaders') this.subscription.on('data', async (blockHeader: BlockHeaderOutput) => { - await this.handleNewBlockHeader(this.web3, blockHeader); - }); + await this.handleNewBlockHeader(this.web3, blockHeader) + }) } - private async handleNewBlockHeader(web3: Web3 | null, blockHeader: BlockHeaderOutput): Promise { - console.log('New block header:', blockHeader.hash); - this.emit(TxTypesEnum.Block, L2Block.create({ - txType: TxType.create(TxTypesEnum.Block), - address: Address.create(blockHeader.hash ? blockHeader.hash.toString() : '') - })); + private async handleNewBlockHeader( + web3: Web3 | null, + blockHeader: BlockHeaderOutput, + ): Promise { + this.emit( + TxTypesEnum.Block, + L2Block.create({ + address: Address.create( + blockHeader.hash ? blockHeader.hash.toString() : '', + ), + txType: TxType.create(TxTypesEnum.Block), + }), + ) - const block = await web3?.eth.getBlock(blockHeader.hash, true); + const block = await web3?.eth.getBlock(blockHeader.hash, true) if (block && block.transactions) { // eslint-disable-next-line @typescript-eslint/no-explicit-any block.transactions.forEach((tx: any) => { if (tx.type.toString() === BlockTypesEnum.EIP1559) { - console.log('New eth tx from', tx.from); - this.emit(TxTypesEnum.Eth, L2Block.create({ - txType: TxType.create(TxTypesEnum.Eth), - address: Address.create(tx.from) - })); + this.emit( + TxTypesEnum.Eth, + L2Block.create({ + address: Address.create(tx.from), + txType: TxType.create(TxTypesEnum.Eth), + }), + ) } - }); + }) } } } diff --git a/apps/api/src/presentation/ExpressServer.spec.ts b/apps/api/src/presentation/ExpressServer.spec.ts index 167d45e..3ac891f 100644 --- a/apps/api/src/presentation/ExpressServer.spec.ts +++ b/apps/api/src/presentation/ExpressServer.spec.ts @@ -1,69 +1,76 @@ -import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest'; -import { ExpressServer } from './ExpressServer'; -import { PollingRoute } from './routes/PollingRoute'; -import express from 'express'; -import { HemiTestnet } from '@cryptochords/shared'; +import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest' +import { ExpressServer } from './ExpressServer' +import { PollingRoute } from './routes/PollingRoute' +import express from 'express' +import { HemiTestnet } from '@cryptochords/shared' vi.mock('express', () => { const listenMock = vi.fn((_port, callback) => { - callback(); + callback() return { - close: vi.fn((cb) => cb && cb()), - }; - }); + close: vi.fn(cb => cb && cb()), + } + }) return { __esModule: true, default: vi.fn(() => ({ listen: listenMock, })), - }; -}); + } +}) vi.mock('ws', () => ({ Server: vi.fn().mockImplementation(() => ({ on: vi.fn(), })), -})); +})) vi.mock('./routes/PollingRoute', () => ({ PollingRoute: vi.fn().mockImplementation(() => ({ initialize: vi.fn(), stop: vi.fn(), })), -})); +})) describe('ExpressServer', () => { - let expressServer: ExpressServer; - let pollingRouteMock: PollingRoute; + let expressServer: ExpressServer + let pollingRouteMock: PollingRoute - const MOCK_PORT = 3000; - const USE_WEBSOCKET = true; - const MOCK_WEBSOCKET_URL = HemiTestnet.rpcUrls.default.webSocket[0]; - const MOCK_RPC_URL = HemiTestnet.rpcUrls.default.http[0]; + const MOCK_PORT = 3000 + const USE_WEBSOCKET = true + const MOCK_WEBSOCKET_URL = HemiTestnet.rpcUrls.default.webSocket[0] + const MOCK_RPC_URL = HemiTestnet.rpcUrls.default.http[0] beforeEach(() => { - vi.clearAllMocks(); + vi.clearAllMocks() - pollingRouteMock = new PollingRoute(USE_WEBSOCKET, MOCK_WEBSOCKET_URL, MOCK_RPC_URL); - expressServer = new ExpressServer(pollingRouteMock, MOCK_PORT); - }); + pollingRouteMock = new PollingRoute( + USE_WEBSOCKET, + MOCK_WEBSOCKET_URL, + MOCK_RPC_URL, + ) + expressServer = new ExpressServer(pollingRouteMock, MOCK_PORT) + }) afterEach(() => { - vi.clearAllMocks(); - }); + vi.clearAllMocks() + }) it('should start the server correctly', () => { - expressServer.start(); + expressServer.start() - const mockedExpress = express() as any; - expect(mockedExpress.listen).toHaveBeenCalledWith(MOCK_PORT, expect.any(Function)); - expect(pollingRouteMock.initialize).toHaveBeenCalled(); - }); + const mockedExpress = express() + expect(mockedExpress.listen).toHaveBeenCalledWith( + MOCK_PORT, + expect.any(Function), + ) + expect(pollingRouteMock.initialize).toHaveBeenCalled() + }) it('should stop the server correctly', async () => { - expressServer.start(); - await expressServer.stop(); + expressServer.start() + await expressServer.stop() - expect(pollingRouteMock.stop).toHaveBeenCalled(); - }); -}); + expect(pollingRouteMock.stop).toHaveBeenCalled() + }) +}) diff --git a/apps/api/src/presentation/ExpressServer.ts b/apps/api/src/presentation/ExpressServer.ts index 9d49801..f95b2a4 100644 --- a/apps/api/src/presentation/ExpressServer.ts +++ b/apps/api/src/presentation/ExpressServer.ts @@ -1,50 +1,51 @@ -import express, { Express } from 'express'; -import { Server as HTTPServer } from 'http'; -import { Server as WebSocketServer } from 'ws'; -import 'dotenv/config'; -import { PollingRoute } from './routes/PollingRoute'; +import express, { Express } from 'express' +import { Server as HTTPServer } from 'http' +import { Server as WebSocketServer } from 'ws' +import 'dotenv/config' +import { PollingRoute } from './routes/PollingRoute' export class ExpressServer { - private pollingRoute: PollingRoute; - private api: Express; - private httpServer: HTTPServer | null = null; - private wss: WebSocketServer | null = null; - private port: string | number; + private pollingRoute: PollingRoute + private api: Express + private httpServer: HTTPServer | null = null + private wss: WebSocketServer | null = null + private port: string | number constructor(pollingRoute: PollingRoute, port: number) { - this.pollingRoute = pollingRoute; - this.api = express(); - this.port = port || 3000; + this.pollingRoute = pollingRoute + this.api = express() + this.port = port || 3000 } public start(): void { this.httpServer = this.api.listen(this.port, () => { - console.log(`Express server running on port ${this.port}`); - }); + console.log(`Express server running on port ${this.port}`) + }) - this.wss = new WebSocketServer({ server: this.httpServer }); + this.wss = new WebSocketServer({ server: this.httpServer }) - this.wss.on('connection', (ws) => { - console.log('WebSocket client connected'); + this.wss.on('connection', ws => { + console.log('WebSocket client connected') ws.on('message', (message: string) => { - console.log('Message received:', message); - }); - }); + console.log('Message received:', message) + }) + }) - this.pollingRoute.initialize(this.wss); + this.pollingRoute.initialize(this.wss) } async stop(): Promise { console.info('CryptoChords API | Closing HTTP Server') - return await new Promise((resolve) => { - this.httpServer?.close((error) => { + return await new Promise(resolve => { + this.httpServer?.close(error => { if (error) { console.error( - `CryptoChords API | Error Closing HTTP Server: ${error.message}`) + `CryptoChords API | Error Closing HTTP Server: ${error.message}`, + ) } else { - this.wss = null; - console.info('CryptoChords API | WSS Server successfully closed'); + this.wss = null + console.info('CryptoChords API | WSS Server successfully closed') } this.pollingRoute.stop() resolve() diff --git a/apps/api/src/presentation/routes/PollingRoute.spec.ts b/apps/api/src/presentation/routes/PollingRoute.spec.ts index 740b668..4cc7063 100644 --- a/apps/api/src/presentation/routes/PollingRoute.spec.ts +++ b/apps/api/src/presentation/routes/PollingRoute.spec.ts @@ -1,7 +1,7 @@ -import { describe, it, expect, beforeEach, vi } from 'vitest'; -import { WebSocketServer } from 'ws'; -import { PollingRoute } from './PollingRoute'; -import { HemiTestnet } from '@cryptochords/shared'; +import { describe, it, expect, beforeEach, vi } from 'vitest' +import { WebSocketServer } from 'ws' +import { PollingRoute } from './PollingRoute' +import { HemiTestnet } from '@cryptochords/shared' vi.mock('../../application/polling/pollingService', () => { return { @@ -9,41 +9,52 @@ vi.mock('../../application/polling/pollingService', () => { execute: vi.fn(), stop: vi.fn(), })), - }; -}); + } +}) -vi.mock('ws'); +vi.mock('ws') vi.mock('../../infrastructure/repositories/blockWebsocket', () => ({ BlockWebsocketRepository: vi.fn(), -})); +})) vi.mock('../../infrastructure/repositories/blockPolling', () => ({ BlockPollingRepository: vi.fn(), -})); +})) describe('PollingRoute', () => { - let wss: WebSocketServer; + let wss: WebSocketServer - const MOCK_WEBSOCKET_URL = HemiTestnet.rpcUrls.default.webSocket[0]; - const MOCK_RPC_URL = HemiTestnet.rpcUrls.default.http[0]; + const MOCK_WEBSOCKET_URL = HemiTestnet.rpcUrls.default.webSocket[0] + const MOCK_RPC_URL = HemiTestnet.rpcUrls.default.http[0] beforeEach(() => { - vi.clearAllMocks(); - wss = new WebSocketServer({ noServer: true }); - }); + vi.clearAllMocks() + wss = new WebSocketServer({ noServer: true }) + }) it('should initialize with WebSocket when useWebsocket is true', () => { - const pollingRoute = new PollingRoute(true, MOCK_WEBSOCKET_URL, MOCK_RPC_URL); - pollingRoute.initialize(wss); - - const mockPollingService = pollingRoute['pollingService'] as any; - expect(mockPollingService.execute).toHaveBeenCalledWith(wss, MOCK_WEBSOCKET_URL); - }); + const pollingRoute = new PollingRoute( + true, + MOCK_WEBSOCKET_URL, + MOCK_RPC_URL, + ) + pollingRoute.initialize(wss) + + const mockPollingService = pollingRoute['pollingService'] + expect(mockPollingService.execute).toHaveBeenCalledWith( + wss, + MOCK_WEBSOCKET_URL, + ) + }) it('should initialize with Polling when useWebsocket is false', () => { - const pollingRoute = new PollingRoute(false, MOCK_WEBSOCKET_URL, MOCK_RPC_URL); - pollingRoute.initialize(wss); - - const mockPollingService = pollingRoute['pollingService'] as any; - expect(mockPollingService.execute).toHaveBeenCalledWith(wss, MOCK_RPC_URL); - }); -}); + const pollingRoute = new PollingRoute( + false, + MOCK_WEBSOCKET_URL, + MOCK_RPC_URL, + ) + pollingRoute.initialize(wss) + + const mockPollingService = pollingRoute['pollingService'] + expect(mockPollingService.execute).toHaveBeenCalledWith(wss, MOCK_RPC_URL) + }) +}) diff --git a/apps/api/src/presentation/routes/PollingRoute.ts b/apps/api/src/presentation/routes/PollingRoute.ts index d605576..417e15e 100644 --- a/apps/api/src/presentation/routes/PollingRoute.ts +++ b/apps/api/src/presentation/routes/PollingRoute.ts @@ -1,30 +1,30 @@ -import { WebSocketServer } from 'ws'; -import { BlockWebsocketRepository } from '../../infrastructure/repositories/blockWebsocket'; -import { BlockPollingRepository } from '../../infrastructure/repositories/blockPolling'; -import { PollingService } from '../../application/polling/pollingService'; -import 'dotenv/config'; +import { WebSocketServer } from 'ws' +import { BlockWebsocketRepository } from '../../infrastructure/repositories/blockWebsocket' +import { BlockPollingRepository } from '../../infrastructure/repositories/blockPolling' +import { PollingService } from '../../application/polling/pollingService' +import 'dotenv/config' export class PollingRoute { - private pollingService: PollingService; - private url: string; + private pollingService: PollingService + private url: string constructor(useWebsocket: boolean, websocketUrl: string, rpcUrl: string) { if (useWebsocket) { - this.url = websocketUrl; - const blockWebsocketRepository = new BlockWebsocketRepository(); - this.pollingService = new PollingService(blockWebsocketRepository); + this.url = websocketUrl + const blockWebsocketRepository = new BlockWebsocketRepository() + this.pollingService = new PollingService(blockWebsocketRepository) } else { - this.url = rpcUrl; - const blockPollingRepository = new BlockPollingRepository(); - this.pollingService = new PollingService(blockPollingRepository); + this.url = rpcUrl + const blockPollingRepository = new BlockPollingRepository() + this.pollingService = new PollingService(blockPollingRepository) } } public initialize(wss: WebSocketServer): void { - this.pollingService.execute(wss, this.url); + this.pollingService.execute(wss, this.url) } public stop(): void { - this.pollingService.stop(); - } + this.pollingService.stop() + } } diff --git a/apps/api/src/server.spec.ts b/apps/api/src/server.spec.ts index 9d29e74..4f1dacb 100644 --- a/apps/api/src/server.spec.ts +++ b/apps/api/src/server.spec.ts @@ -1,40 +1,49 @@ -import { describe, it, expect, beforeAll, afterAll, vi, beforeEach, afterEach } from 'vitest'; -import { ExpressServer } from './presentation/ExpressServer'; - -vi.mock('./presentation/ExpressServer'); +import { + describe, + it, + expect, + beforeAll, + afterAll, + vi, + beforeEach, + afterEach, +} from 'vitest' +import { ExpressServer } from './presentation/ExpressServer' + +vi.mock('./presentation/ExpressServer') describe('server', () => { beforeEach(() => { vi.spyOn(process, 'exit').mockImplementation(() => { - return undefined as never; - }); - }); + return undefined as never + }) + }) afterEach(() => { - vi.restoreAllMocks(); - }); + vi.restoreAllMocks() + }) beforeAll(() => { - ExpressServer.prototype.start = vi.fn(); - ExpressServer.prototype.stop = vi.fn(); - }); + ExpressServer.prototype.start = vi.fn() + ExpressServer.prototype.stop = vi.fn() + }) afterAll(() => { - vi.restoreAllMocks(); - }); + vi.restoreAllMocks() + }) it('should start the server', async () => { - await import('./server'); - expect(ExpressServer.prototype.start).toHaveBeenCalled(); - }); + await import('./server') + expect(ExpressServer.prototype.start).toHaveBeenCalled() + }) it('should stop the server on SIGTERM', async () => { - process.emit('SIGTERM'); - expect(ExpressServer.prototype.stop).toHaveBeenCalled(); - }); + process.emit('SIGTERM') + expect(ExpressServer.prototype.stop).toHaveBeenCalled() + }) it('should stop the server on SIGINT', async () => { - process.emit('SIGINT'); - expect(ExpressServer.prototype.stop).toHaveBeenCalled(); - }); -}); + process.emit('SIGINT') + expect(ExpressServer.prototype.stop).toHaveBeenCalled() + }) +}) diff --git a/apps/api/src/server.ts b/apps/api/src/server.ts index ebfc250..4e46305 100644 --- a/apps/api/src/server.ts +++ b/apps/api/src/server.ts @@ -1,19 +1,27 @@ -import 'dotenv/config'; -import { ExpressServer } from './presentation/ExpressServer'; -import { PollingRoute } from './presentation/routes/PollingRoute'; -import { HemiTestnet, HemiMainnet } from "@cryptochords/shared"; +import 'dotenv/config' +import { ExpressServer } from './presentation/ExpressServer' +import { PollingRoute } from './presentation/routes/PollingRoute' +import { HemiTestnet, HemiMainnet } from '@cryptochords/shared' const useWebsocket = process.env['USE_WEBSOCKET_NODE_L2'] === 'true' const useMainnet = process.env['ENABLE_MAINNET'] === 'true' // Testnet const websocketTestnet = HemiTestnet.rpcUrls.default.webSocket[0] const rpcTestnet = HemiTestnet.rpcUrls.default.http[0] -const pollingRouteTestnet = new PollingRoute(useWebsocket, websocketTestnet, rpcTestnet) +const pollingRouteTestnet = new PollingRoute( + useWebsocket, + websocketTestnet, + rpcTestnet, +) const serverTestnet = new ExpressServer(pollingRouteTestnet, 3000) // Mainnet const websocketMainnet = HemiMainnet.rpcUrls.default.webSocket[0] const rpcMainnet = HemiMainnet.rpcUrls.default.http[0] -const pollingRouteMainnet = new PollingRoute(useWebsocket, websocketMainnet, rpcMainnet) +const pollingRouteMainnet = new PollingRoute( + useWebsocket, + websocketMainnet, + rpcMainnet, +) const serverMainnet = new ExpressServer(pollingRouteMainnet, 3001) const startServer = async (): Promise => { @@ -32,12 +40,12 @@ const stopServer = async (): Promise => { process.on('SIGTERM', async () => { await stopServer() - process.exit(0); + process.exit(0) }) process.on('SIGINT', async () => { await stopServer() - process.exit(0); + process.exit(0) }) startServer() diff --git a/apps/api/tsconfig.build.json b/apps/api/tsconfig.build.json index e702d94..5417c7d 100644 --- a/apps/api/tsconfig.build.json +++ b/apps/api/tsconfig.build.json @@ -1,6 +1,4 @@ { - "extends": "./tsconfig.json", - "exclude": [ - "src/**/*.spec.ts" - ] - } \ No newline at end of file + "extends": "./tsconfig.json", + "exclude": ["src/**/*.spec.ts"] +} diff --git a/apps/api/tsconfig.json b/apps/api/tsconfig.json index fc3ad0f..b7aadeb 100644 --- a/apps/api/tsconfig.json +++ b/apps/api/tsconfig.json @@ -1,5 +1,5 @@ { - "include": [ "src/**/*" ], + "include": ["src/**/*"], "compilerOptions": { /* Visit https://aka.ms/tsconfig to read more about this file */ @@ -12,7 +12,7 @@ // "disableReferencedProjectLoad": true, /* Reduce the number of projects loaded automatically by TypeScript. */ /* Language and Environment */ - "target": "es2016", /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */ + "target": "es2016" /* Set the JavaScript language version for emitted JavaScript and include compatible library declarations. */, // "lib": [], /* Specify a set of bundled library declaration files that describe the target runtime environment. */ // "jsx": "preserve", /* Specify what JSX code is generated. */ // "experimentalDecorators": true, /* Enable experimental support for TC39 stage 2 draft decorators. */ @@ -26,9 +26,9 @@ // "moduleDetection": "auto", /* Control what method is used to detect module-format JS files. */ /* Modules */ - "module": "commonjs", /* Specify what module code is generated. */ - "rootDir": "./src", /* Specify the root folder within your source files. */ - "moduleResolution": "node", /* Specify how TypeScript looks up a file from a given module specifier. */ + "module": "commonjs" /* Specify what module code is generated. */, + "rootDir": "./src" /* Specify the root folder within your source files. */, + "moduleResolution": "node" /* Specify how TypeScript looks up a file from a given module specifier. */, // "baseUrl": "./", /* Specify the base directory to resolve non-relative module names. */ // "paths": {}, /* Specify a set of entries that re-map imports to additional lookup locations. */ // "rootDirs": [], /* Allow multiple folders to be treated as one when resolving modules. */ @@ -40,17 +40,17 @@ // "noResolve": true, /* Disallow 'import's, 'require's or ''s from expanding the number of files TypeScript should add to a project. */ /* JavaScript Support */ - "allowJs": true, /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */ + "allowJs": true /* Allow JavaScript files to be a part of your program. Use the 'checkJS' option to get errors from these files. */, // "checkJs": true, /* Enable error reporting in type-checked JavaScript files. */ // "maxNodeModuleJsDepth": 1, /* Specify the maximum folder depth used for checking JavaScript files from 'node_modules'. Only applicable with 'allowJs'. */ /* Emit */ - "declaration": true, /* Generate .d.ts files from TypeScript and JavaScript files in your project. */ - "declarationMap": true, /* Create sourcemaps for d.ts files. */ + "declaration": true /* Generate .d.ts files from TypeScript and JavaScript files in your project. */, + "declarationMap": true /* Create sourcemaps for d.ts files. */, // "emitDeclarationOnly": true, /* Only output d.ts files and not JavaScript files. */ - "sourceMap": true, /* Create source map files for emitted JavaScript files. */ + "sourceMap": true /* Create source map files for emitted JavaScript files. */, // "outFile": "./", /* Specify a file that bundles all outputs into one JavaScript file. If 'declaration' is true, also designates a file that bundles all .d.ts output. */ - "outDir": "./dist", /* Specify an output folder for all emitted files. */ + "outDir": "./dist" /* Specify an output folder for all emitted files. */, // "removeComments": true, /* Disable emitting comments. */ // "noEmit": true, /* Disable emitting files from a compilation. */ // "importHelpers": true, /* Allow importing helper functions from tslib once per project, instead of including them per-file. */ @@ -72,33 +72,33 @@ /* Interop Constraints */ // "isolatedModules": true, /* Ensure that each file can be safely transpiled without relying on other imports. */ // "allowSyntheticDefaultImports": true, /* Allow 'import x from y' when a module doesn't have a default export. */ - "esModuleInterop": true, /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */ + "esModuleInterop": true /* Emit additional JavaScript to ease support for importing CommonJS modules. This enables 'allowSyntheticDefaultImports' for type compatibility. */, // "preserveSymlinks": true, /* Disable resolving symlinks to their realpath. This correlates to the same flag in node. */ - "forceConsistentCasingInFileNames": true, /* Ensure that casing is correct in imports. */ + "forceConsistentCasingInFileNames": true /* Ensure that casing is correct in imports. */, /* Type Checking */ - "strict": true, /* Enable all strict type-checking options. */ - "noImplicitAny": true, /* Enable error reporting for expressions and declarations with an implied 'any' type. */ - "strictNullChecks": true, /* When type checking, take into account 'null' and 'undefined'. */ - "strictFunctionTypes": true, /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */ - "strictBindCallApply": true, /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */ - "strictPropertyInitialization": true, /* Check for class properties that are declared but not set in the constructor. */ - "noImplicitThis": true, /* Enable error reporting when 'this' is given the type 'any'. */ - "useUnknownInCatchVariables": true, /* Default catch clause variables as 'unknown' instead of 'any'. */ - "alwaysStrict": true, /* Ensure 'use strict' is always emitted. */ - "noUnusedLocals": true, /* Enable error reporting when local variables aren't read. */ - "noUnusedParameters": true, /* Raise an error when a function parameter isn't read. */ - "exactOptionalPropertyTypes": true, /* Interpret optional property types as written, rather than adding 'undefined'. */ - "noImplicitReturns": true, /* Enable error reporting for codepaths that do not explicitly return in a function. */ - "noFallthroughCasesInSwitch": true, /* Enable error reporting for fallthrough cases in switch statements. */ - "noUncheckedIndexedAccess": true, /* Add 'undefined' to a type when accessed using an index. */ - "noImplicitOverride": true, /* Ensure overriding members in derived classes are marked with an override modifier. */ - "noPropertyAccessFromIndexSignature": true, /* Enforces using indexed accessors for keys declared using an indexed type. */ - "allowUnusedLabels": true, /* Disable error reporting for unused labels. */ - "allowUnreachableCode": true, /* Disable error reporting for unreachable code. */ + "strict": true /* Enable all strict type-checking options. */, + "noImplicitAny": true /* Enable error reporting for expressions and declarations with an implied 'any' type. */, + "strictNullChecks": true /* When type checking, take into account 'null' and 'undefined'. */, + "strictFunctionTypes": true /* When assigning functions, check to ensure parameters and the return values are subtype-compatible. */, + "strictBindCallApply": true /* Check that the arguments for 'bind', 'call', and 'apply' methods match the original function. */, + "strictPropertyInitialization": true /* Check for class properties that are declared but not set in the constructor. */, + "noImplicitThis": true /* Enable error reporting when 'this' is given the type 'any'. */, + "useUnknownInCatchVariables": true /* Default catch clause variables as 'unknown' instead of 'any'. */, + "alwaysStrict": true /* Ensure 'use strict' is always emitted. */, + "noUnusedLocals": true /* Enable error reporting when local variables aren't read. */, + "noUnusedParameters": true /* Raise an error when a function parameter isn't read. */, + "exactOptionalPropertyTypes": true /* Interpret optional property types as written, rather than adding 'undefined'. */, + "noImplicitReturns": true /* Enable error reporting for codepaths that do not explicitly return in a function. */, + "noFallthroughCasesInSwitch": true /* Enable error reporting for fallthrough cases in switch statements. */, + "noUncheckedIndexedAccess": true /* Add 'undefined' to a type when accessed using an index. */, + "noImplicitOverride": true /* Ensure overriding members in derived classes are marked with an override modifier. */, + "noPropertyAccessFromIndexSignature": true /* Enforces using indexed accessors for keys declared using an indexed type. */, + "allowUnusedLabels": true /* Disable error reporting for unused labels. */, + "allowUnreachableCode": true /* Disable error reporting for unreachable code. */, /* Completeness */ // "skipDefaultLibCheck": true, /* Skip type checking .d.ts files that are included with TypeScript. */ - "skipLibCheck": true /* Skip type checking all .d.ts files. */ + "skipLibCheck": true /* Skip type checking all .d.ts files. */ } } diff --git a/apps/api/vitest.config.ts b/apps/api/vitest.config.ts index 1c0eacf..e21acf9 100644 --- a/apps/api/vitest.config.ts +++ b/apps/api/vitest.config.ts @@ -4,8 +4,8 @@ export default defineConfig({ test: { coverage: { // instanbul excludes interfaces from coverage - provider: 'istanbul', include: ['src/**/*.ts'], + provider: 'istanbul', }, }, -}) \ No newline at end of file +}) diff --git a/apps/api/webSocketTest.html b/apps/api/webSocketTest.html index 070df56..a3c7827 100644 --- a/apps/api/webSocketTest.html +++ b/apps/api/webSocketTest.html @@ -1,29 +1,29 @@ - + - - + + WebSocket Test - - + +

Last block: waiting...

Last eth tx: waiting...

Last piano note: waiting...

- + diff --git a/apps/web/.prettierrc.json b/apps/web/.prettierrc.json new file mode 100644 index 0000000..21c021c --- /dev/null +++ b/apps/web/.prettierrc.json @@ -0,0 +1,7 @@ +{ + "arrowParens": "avoid", + "quoteProps": "consistent", + "semi": false, + "singleQuote": true, + "tabWidth": 2 +} diff --git a/apps/web/@types/tonejs-instruments.d.ts b/apps/web/@types/tonejs-instruments.d.ts index 2b6286f..79a646a 100644 --- a/apps/web/@types/tonejs-instruments.d.ts +++ b/apps/web/@types/tonejs-instruments.d.ts @@ -1,199 +1,139 @@ declare module 'tonejs-instrument-bass-electric-mp3' { import { Sampler } from 'tone' export default class InstrumentBassElectricMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-bassoon-mp3' { import { Sampler } from 'tone' export default class InstrumentBassoonMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-cello-mp3' { import { Sampler } from 'tone' export default class InstrumentCelloMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-clarinet-mp3' { import { Sampler } from 'tone' export default class InstrumentClarinetMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-contrabass-mp3' { import { Sampler } from 'tone' export default class InstrumentContrabassMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-flute-mp3' { import { Sampler } from 'tone' export default class InstrumentFluteMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-french-horn-mp3' { import { Sampler } from 'tone' export default class InstrumentFrenchHornMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-guitar-acoustic-mp3' { import { Sampler } from 'tone' export default class InstrumentGuitarAcousticMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-guitar-electric-mp3' { import { Sampler } from 'tone' export default class InstrumentGuitarElectricMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-guitar-nylon-mp3' { import { Sampler } from 'tone' export default class InstrumentGuitarNylonMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-harmonium-mp3' { import { Sampler } from 'tone' export default class InstrumentHarmoniumMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-harp-mp3' { import { Sampler } from 'tone' export default class InstrumentHarpMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-organ-mp3' { import { Sampler } from 'tone' export default class InstrumentOrganMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-piano-mp3' { import { Sampler } from 'tone' export default class InstrumentPianoMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-saxophone-mp3' { import { Sampler } from 'tone' export default class InstrumentSaxophoneMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-trombone-mp3' { import { Sampler } from 'tone' export default class InstrumentTromboneMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-trumpet-mp3' { import { Sampler } from 'tone' export default class InstrumentTrumpetMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-tuba-mp3' { import { Sampler } from 'tone' export default class InstrumentTubaMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-violin-mp3' { import { Sampler } from 'tone' export default class InstrumentViolinMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } declare module 'tonejs-instrument-xylophone-mp3' { import { Sampler } from 'tone' export default class InstrumentXylophoneMp3 extends Sampler { - constructor(options?: { - minify?: boolean - onload?: unknown - }) + constructor(options?: { minify?: boolean; onload?: unknown }) } } diff --git a/apps/web/README.md b/apps/web/README.md index d0cc11b..0d30ca6 100644 --- a/apps/web/README.md +++ b/apps/web/README.md @@ -1,29 +1,38 @@ # CryptoChords Web App + This is the web application of the CryptoChords project. It is a front-end application that allows users to visualize activities on the [Hemi network](https://hemi.xyz). ## Architecture + It is built using DDD principles with the following layers: + - Application - Domain - Infrastructure - Presentation ### Application + This layer contains the use cases of the application. It is the entry point of the application and it is responsible for coordinating the domain layer and the infrastructure layer. ### Domain + This layer contains the business logic of the application. It is the core of the application and it is responsible for the entities, value objects, and domain services. ### Infrastructure + This layer contains the implementation of the application. It is responsible for the database and the external services. ### Presentation + This layer contains the user interface of the application. #### Presenters + This sublayer contains the presenters of the application. It is responsible for the whole user interface logic. #### React + This sublayer contains the React implementation of the application. Its components are responsible for the style and the interface with the presenters, displaying its state and redirecting the user actions. ## Getting Started @@ -99,8 +108,10 @@ The environment variables are defined in the `.env` file. The following variable - `VITE_LOGO_URL`: The URL of the logo. If it is not defined, the logo will point to the root of the web app and will not display the pointer cursor. ## Contribution + If you want to contribute to this project and make it better, your help is very welcome. You can find more information about how to contribute in the [`CONTRIBUTING.md`](../../CONTRIBUTING.md) file. ## License -This project is licensed under the MIT License - see the [LICENSE](../../LICENSE) file for details. \ No newline at end of file + +This project is licensed under the MIT License - see the [LICENSE](../../LICENSE) file for details. diff --git a/apps/web/index.html b/apps/web/index.html index 50635c9..41901cc 100644 --- a/apps/web/index.html +++ b/apps/web/index.html @@ -2,7 +2,11 @@ - + Crypto Chords diff --git a/apps/web/package.json b/apps/web/package.json index 92faaaf..59af7c8 100644 --- a/apps/web/package.json +++ b/apps/web/package.json @@ -7,7 +7,9 @@ "scripts": { "dev": "vite", "build": "tsc && vite build", - "lint": "eslint . --ext ts,tsx --report-unused-disable-directives --max-warnings 0", + "format:check": "prettier --check .", + "format:fix": "npx prettier --write .", + "lint": "eslint . --ext ts,tsx", "preview": "vite preview", "test": "vitest run", "test:cov": "vitest run --coverage", @@ -58,6 +60,7 @@ "eslint-plugin-react-hooks": "5.1.0", "eslint-plugin-react-refresh": "0.4.16", "postcss": "8.4.49", + "prettier": "^3.3.3", "tailwindcss": "3.4.16", "typescript": "5.2.2", "vite": "5.0.8", diff --git a/apps/web/src/application/ObservableService.ts b/apps/web/src/application/ObservableService.ts index ea145b3..ad29ce5 100644 --- a/apps/web/src/application/ObservableService.ts +++ b/apps/web/src/application/ObservableService.ts @@ -1,10 +1,16 @@ -import { Observable, EventBusInstance, EventSubscription, Event } from '@cryptochords/shared' +import { + Observable, + EventBusInstance, + EventSubscription, + Event, +} from '@cryptochords/shared' import { Service } from './Service' import { ServiceResponseEvent } from './events/ServiceResponseEvent' export abstract class ObservableService -extends Service -implements Observable>{ + extends Service + implements Observable> +{ private eventBus = new EventBusInstance() public async execute(request: Request): Promise { @@ -16,7 +22,12 @@ implements Observable>{ return response } - public async listen(listener: EventSubscription>) { - this.eventBus.subscribe(ServiceResponseEvent.eventKey, listener as EventSubscription) + public async listen( + listener: EventSubscription>, + ) { + this.eventBus.subscribe( + ServiceResponseEvent.eventKey, + listener as EventSubscription, + ) } -} \ No newline at end of file +} diff --git a/apps/web/src/application/Service.ts b/apps/web/src/application/Service.ts index 48e9b38..42a238a 100644 --- a/apps/web/src/application/Service.ts +++ b/apps/web/src/application/Service.ts @@ -1,7 +1,7 @@ -export abstract class Service{ +export abstract class Service { protected abstract process(request: Request): Promise - + public async execute(request: Request): Promise { return this.process(request) } -} \ No newline at end of file +} diff --git a/apps/web/src/application/events/ServiceResponseEvent.ts b/apps/web/src/application/events/ServiceResponseEvent.ts index 6b7b5ba..5514dab 100644 --- a/apps/web/src/application/events/ServiceResponseEvent.ts +++ b/apps/web/src/application/events/ServiceResponseEvent.ts @@ -1,8 +1,11 @@ import { Event } from '@cryptochords/shared' export class ServiceResponseEvent extends Event { - static eventKey = Symbol('ServiceResponseEvent'); - constructor(public readonly request: Request, public readonly response: Response) { + static eventKey = Symbol('ServiceResponseEvent') + constructor( + public readonly request: Request, + public readonly response: Response, + ) { super(ServiceResponseEvent.eventKey) } } diff --git a/apps/web/src/application/services/CreateCube/CreateCubeDtos.ts b/apps/web/src/application/services/CreateCube/CreateCubeDtos.ts index 0e17781..31802e1 100644 --- a/apps/web/src/application/services/CreateCube/CreateCubeDtos.ts +++ b/apps/web/src/application/services/CreateCube/CreateCubeDtos.ts @@ -2,11 +2,11 @@ export interface CreateCubeResponseDto { id: string x: number y: number - color: string + color: string mirrored: boolean } export interface CreateCubeRequestDto { - x: number, + x: number color: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/CreateCube/CreateCubeService.spec.ts b/apps/web/src/application/services/CreateCube/CreateCubeService.spec.ts index 404464b..68dd2ee 100644 --- a/apps/web/src/application/services/CreateCube/CreateCubeService.spec.ts +++ b/apps/web/src/application/services/CreateCube/CreateCubeService.spec.ts @@ -8,12 +8,12 @@ const cubeRepositoryMock: CubeRepository = { x: { value: 0.5 }, y: { value: 0 }, color: { value: 'blue' }, - mirrored: false + mirrored: false, }), delete: vi.fn(), update: vi.fn(), list: vi.fn(), - clear: vi.fn() + clear: vi.fn(), } const create: CreateCubeService = new CreateCubeService(cubeRepositoryMock) @@ -27,7 +27,7 @@ describe('src/application/CreateCube/CreateCubeService', () => { x: 0.5, y: 0, color: 'blue', - mirrored: false + mirrored: false, }) }) }) diff --git a/apps/web/src/application/services/CreateCube/CreateCubeService.ts b/apps/web/src/application/services/CreateCube/CreateCubeService.ts index eaf5af3..07d6987 100644 --- a/apps/web/src/application/services/CreateCube/CreateCubeService.ts +++ b/apps/web/src/application/services/CreateCube/CreateCubeService.ts @@ -6,23 +6,31 @@ import { UnitInterval } from '../../../domain/valueObjects/UnitInterval' import { ObservableService } from '../../ObservableService' import { CreateCubeRequestDto, CreateCubeResponseDto } from './CreateCubeDtos' -export class CreateCubeService extends ObservableService{ +export class CreateCubeService extends ObservableService< + CreateCubeRequestDto, + CreateCubeResponseDto +> { private readonly cubeRepository: CubeRepository - + constructor(cubeRepository: CubeRepository) { super() this.cubeRepository = cubeRepository } - protected async process(request: CreateCubeRequestDto): Promise { - const cube = Cube.create(TransactionColor.create(request.color as TransactionColorEnum), UnitInterval.create(request.x)) + protected async process( + request: CreateCubeRequestDto, + ): Promise { + const cube = Cube.create( + TransactionColor.create(request.color as TransactionColorEnum), + UnitInterval.create(request.x), + ) const createdCube = await this.cubeRepository.create(cube) return { id: createdCube.uuid.value, x: createdCube.x.value, y: createdCube.y.value, color: createdCube.color.value, - mirrored: createdCube.mirrored + mirrored: createdCube.mirrored, } } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/CreateKeyboard/CreateKeyboardDtos.ts b/apps/web/src/application/services/CreateKeyboard/CreateKeyboardDtos.ts index a72c53f..c7eba9d 100644 --- a/apps/web/src/application/services/CreateKeyboard/CreateKeyboardDtos.ts +++ b/apps/web/src/application/services/CreateKeyboard/CreateKeyboardDtos.ts @@ -10,7 +10,7 @@ export interface CreateKeyboardResponseDto { class: string octave: number } - keyShape: string, + keyShape: string x: number color?: string }[] diff --git a/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.spec.ts b/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.spec.ts index 4b51863..2307934 100644 --- a/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.spec.ts +++ b/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.spec.ts @@ -7,9 +7,9 @@ describe('src/application/CreateKeyboard/CreateKeyboardService', () => { let keyboardRepository: KeyboardRepository beforeAll(() => { - keyboardRepository= { + keyboardRepository = { setKeyboard: vi.fn(), - getKeyboard: vi.fn() + getKeyboard: vi.fn(), } createKeyboardService = new CreateKeyboardService(keyboardRepository) }) @@ -22,7 +22,7 @@ describe('src/application/CreateKeyboard/CreateKeyboardService', () => { const response = await createKeyboardService.execute({ initialOctave: 0, numberOfKeys: 88, - initialPitchClass: 'A' + initialPitchClass: 'A', }) expect(response.keys.length).toBe(88) }) @@ -31,8 +31,8 @@ describe('src/application/CreateKeyboard/CreateKeyboardService', () => { await createKeyboardService.execute({ initialOctave: 0, numberOfKeys: 88, - initialPitchClass: 'A' + initialPitchClass: 'A', }) expect(keyboardRepository.setKeyboard).toHaveBeenCalled() }) -}) \ No newline at end of file +}) diff --git a/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.ts b/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.ts index 00050c7..dec7bf4 100644 --- a/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.ts +++ b/apps/web/src/application/services/CreateKeyboard/CreateKeyboardService.ts @@ -2,9 +2,15 @@ import { PitchClassEnum } from '../../../domain/enum/PitchClassEnum' import { KeyboardFactory } from '../../../domain/factories/KeyboardFactory' import { KeyboardRepository } from '../../../domain/repositories/KeyboardRepository' import { ObservableService } from '../../ObservableService' -import { CreateKeyboardRequestDto, CreateKeyboardResponseDto } from './CreateKeyboardDtos' +import { + CreateKeyboardRequestDto, + CreateKeyboardResponseDto, +} from './CreateKeyboardDtos' -export class CreateKeyboardService extends ObservableService{ +export class CreateKeyboardService extends ObservableService< + CreateKeyboardRequestDto, + CreateKeyboardResponseDto +> { private keyboardRepository: KeyboardRepository constructor(keyboardRepository: KeyboardRepository) { @@ -12,11 +18,13 @@ export class CreateKeyboardService extends ObservableService { + protected async process( + request: CreateKeyboardRequestDto, + ): Promise { const keyboard = KeyboardFactory.create({ initialOctave: request.initialOctave, numberOfKeys: request.numberOfKeys, - initialPitchClass: request.initialPitchClass as PitchClassEnum + initialPitchClass: request.initialPitchClass as PitchClassEnum, }) this.keyboardRepository.setKeyboard(keyboard) @@ -25,12 +33,12 @@ export class CreateKeyboardService extends ObservableService ({ pitch: { class: key.pitch.pitchClass.value, - octave: key.pitch.octave + octave: key.pitch.octave, }, keyShape: key.keyShape.value, x: key.x.value, - color: key.color - })) + color: key.color, + })), } } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/CreateTransaction/CreateTransactionDtos.ts b/apps/web/src/application/services/CreateTransaction/CreateTransactionDtos.ts index 0ec3906..2c86d7b 100644 --- a/apps/web/src/application/services/CreateTransaction/CreateTransactionDtos.ts +++ b/apps/web/src/application/services/CreateTransaction/CreateTransactionDtos.ts @@ -3,4 +3,4 @@ export interface CreateTransactionRequest { address: string network: string timestamp: number -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/CreateTransaction/CreateTransactionService.ts b/apps/web/src/application/services/CreateTransaction/CreateTransactionService.ts index 2d6ebfc..0a7ef05 100644 --- a/apps/web/src/application/services/CreateTransaction/CreateTransactionService.ts +++ b/apps/web/src/application/services/CreateTransaction/CreateTransactionService.ts @@ -1,4 +1,10 @@ -import { Address, NetworkEnum, Timestamp, TxType, TxTypesEnum } from '@cryptochords/shared' +import { + Address, + NetworkEnum, + Timestamp, + TxType, + TxTypesEnum, +} from '@cryptochords/shared' import { Transaction } from '../../../domain/entities/Transaction' import { InstrumentEnum } from '../../../domain/enum/InstrumentEnum' import { KeyboardRepository } from '../../../domain/repositories/KeyboardRepository' @@ -11,7 +17,10 @@ import { PressKeyService } from '../PressKey/PressKeyService' import { ReleaseKeyService } from '../ReleaseKey/ReleaseKeyService' import { CreateTransactionRequest } from './CreateTransactionDtos' -export class CreateTransactionService extends ObservableService { +export class CreateTransactionService extends ObservableService< + CreateTransactionRequest, + void +> { private readonly keyboardRepository: KeyboardRepository private readonly transactionRepository: TransactionRepository private readonly createCubeService: CreateCubeService @@ -25,7 +34,7 @@ export class CreateTransactionService extends ObservableService { const transaction = await this.createTransaction(request) - const [ key, instrument ] = await this.pressKey(transaction) - if (!key) - return + const [key, instrument] = await this.pressKey(transaction) + if (!key) return await this.createCube(key) - setTimeout(() => { - this.releaseKey(key, instrument) - }, 100 + Math.random() * 3_000) + setTimeout( + () => { + this.releaseKey(key, instrument) + }, + 100 + Math.random() * 3_000, + ) } - async createTransaction(request: CreateTransactionRequest): Promise { - const network = await this.netwtorkRepository.find(request.network as NetworkEnum) + async createTransaction( + request: CreateTransactionRequest, + ): Promise { + const network = await this.netwtorkRepository.find( + request.network as NetworkEnum, + ) - if (!network) - throw new Error('Network not found') + if (!network) throw new Error('Network not found') const transaction = Transaction.create({ txType: TxType.create(request.txType as TxTypesEnum), address: Address.create(request.address), network, - timestamp: Timestamp.create(request.timestamp) + timestamp: Timestamp.create(request.timestamp), }) await this.transactionRepository.create(transaction) @@ -68,22 +82,25 @@ export class CreateTransactionService extends ObservableService { this.createCubeService.execute({ x: key.x.value, - color: key.color + color: key.color, }) } - async pressKey(transaction: Transaction): Promise<[Key | undefined, (InstrumentEnum | undefined) ] > { + async pressKey( + transaction: Transaction, + ): Promise<[Key | undefined, InstrumentEnum | undefined]> { const keyboard = this.keyboardRepository.getKeyboard() - if (!keyboard) - return [undefined, undefined] + if (!keyboard) return [undefined, undefined] - const randomKey = keyboard.getRandomWhiteKeyByTxType(transaction.txType.value) + const randomKey = keyboard.getRandomWhiteKeyByTxType( + transaction.txType.value, + ) const { instrument } = await this.pressKeyService.execute({ pitchClass: randomKey.pitch.pitchClass.value, - octave: randomKey.pitch.octave + octave: randomKey.pitch.octave, }) - + return [randomKey, instrument as InstrumentEnum] } @@ -91,7 +108,7 @@ export class CreateTransactionService extends ObservableService { it('should return the correct values', async () => { expect(response).toEqual({ - cubes: [{ - id: 'some-id', - x: 0.5, - y: 0, - color: 'blue', - mirrored: false - }, { - id: 'another-id', - x: 0.1, - y: 0, - color: 'orange', - mirrored: false - }] + cubes: [ + { + id: 'some-id', + x: 0.5, + y: 0, + color: 'blue', + mirrored: false, + }, + { + id: 'another-id', + x: 0.1, + y: 0, + color: 'orange', + mirrored: false, + }, + ], }) }) }) diff --git a/apps/web/src/application/services/GetCubes/GetCubesService.ts b/apps/web/src/application/services/GetCubes/GetCubesService.ts index ae8f362..2ac9b54 100644 --- a/apps/web/src/application/services/GetCubes/GetCubesService.ts +++ b/apps/web/src/application/services/GetCubes/GetCubesService.ts @@ -2,9 +2,12 @@ import { CubeRepository } from '../../../domain/repositories/CubeRepository' import { ObservableService } from '../../ObservableService' import { GetCubesResponseDto } from './GetCubesDtos' -export class GetCubesService extends ObservableService{ +export class GetCubesService extends ObservableService< + void, + GetCubesResponseDto +> { private readonly cubeRepository: CubeRepository - + constructor(cubeRepository: CubeRepository) { super() this.cubeRepository = cubeRepository @@ -20,8 +23,8 @@ export class GetCubesService extends ObservableService{ +export class GetKeyboardService extends ObservableService< + void, + GetKeyboardResponseDto +> { private readonly keyboardRepository: KeyboardRepository - + constructor(keyboardRepository: KeyboardRepository) { super() this.keyboardRepository = keyboardRepository @@ -15,7 +18,7 @@ export class GetKeyboardService extends ObservableService ({ pitch: { class: key.pitch.pitchClass.value, - octave: key.pitch.octave + octave: key.pitch.octave, }, keyShape: key.keyShape.value, x: key.x.value, color: key.color, - pressed: key.pressed - })) + pressed: key.pressed, + })), } } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/GetOptions/GetOptionsDtos.ts b/apps/web/src/application/services/GetOptions/GetOptionsDtos.ts index 1009d77..8d77f35 100644 --- a/apps/web/src/application/services/GetOptions/GetOptionsDtos.ts +++ b/apps/web/src/application/services/GetOptions/GetOptionsDtos.ts @@ -1,4 +1,4 @@ export interface GetOptionsResponseDto { muted: boolean instrument: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/GetOptions/GetOptionsService.ts b/apps/web/src/application/services/GetOptions/GetOptionsService.ts index 8cc3c5c..995a9ae 100644 --- a/apps/web/src/application/services/GetOptions/GetOptionsService.ts +++ b/apps/web/src/application/services/GetOptions/GetOptionsService.ts @@ -2,9 +2,12 @@ import { OptionsRepository } from '../../../domain/repositories/OptionsRepositor import { ObservableService } from '../../ObservableService' import { GetOptionsResponseDto } from './GetOptionsDtos' -export class GetOptionsService extends ObservableService{ +export class GetOptionsService extends ObservableService< + void, + GetOptionsResponseDto +> { private readonly optionsRepository: OptionsRepository - + constructor(optionsRepository: OptionsRepository) { super() this.optionsRepository = optionsRepository @@ -15,7 +18,7 @@ export class GetOptionsService extends ObservableService { +export class GetSelectedNetworkService extends ObservableService< + void, + GetSelectedNetworkResponseDto +> { private readonly netwtorkRepository: NetworkRepository - constructor( - networkRepository: NetworkRepository, - ) { + constructor(networkRepository: NetworkRepository) { super() this.netwtorkRepository = networkRepository } @@ -19,4 +20,4 @@ export class GetSelectedNetworkService extends ObservableService { +export class ListNetworksService extends ObservableService< + void, + ListNetworksResponseDto +> { private readonly networkRepository: NetworkRepository - constructor( - networkRepository: NetworkRepository, - ) { + constructor(networkRepository: NetworkRepository) { super() this.networkRepository = networkRepository } @@ -16,7 +16,7 @@ export class ListNetworksService extends ObservableService { const [networks, selectedNetwork] = await Promise.all([ this.networkRepository.list(), - this.networkRepository.getSelected() + this.networkRepository.getSelected(), ]) return { @@ -24,7 +24,7 @@ export class ListNetworksService extends ObservableService{ +export class ListTransactionsService extends ObservableService< + void, + ListTransactionsResponseDto +> { private readonly transactionRepository: TransactionRepository - - constructor( - transactionRepository: TransactionRepository, - ) { + + constructor(transactionRepository: TransactionRepository) { super() this.transactionRepository = transactionRepository } @@ -20,8 +21,8 @@ export class ListTransactionsService extends ObservableService { +import { + LoadInstrumentRequest, + LoadInstrumentResponse, +} from './LoadInstrumentDtos' +export class LoadInstrumentService extends ObservableService< + LoadInstrumentRequest, + LoadInstrumentResponse +> { private soundService: SoundService constructor(soundService: SoundService) { @@ -11,7 +16,9 @@ export class LoadInstrumentService extends ObservableService { + protected async process({ + instrument, + }: LoadInstrumentRequest): Promise { await this.soundService.loadInstrument(instrument) return } diff --git a/apps/web/src/application/services/PlaySound/PlaySoundDtos.ts b/apps/web/src/application/services/PlaySound/PlaySoundDtos.ts index ba48bf0..2be1440 100644 --- a/apps/web/src/application/services/PlaySound/PlaySoundDtos.ts +++ b/apps/web/src/application/services/PlaySound/PlaySoundDtos.ts @@ -5,4 +5,4 @@ export interface PlaySoundRequest { export interface PlaySoundResponse { instrument?: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/PlaySound/PlaySoundService.ts b/apps/web/src/application/services/PlaySound/PlaySoundService.ts index 05e1ff2..981483d 100644 --- a/apps/web/src/application/services/PlaySound/PlaySoundService.ts +++ b/apps/web/src/application/services/PlaySound/PlaySoundService.ts @@ -5,27 +5,38 @@ import { PitchClass } from '../../../domain/valueObjects/PitchClass' import { ObservableService } from '../../ObservableService' import { PlaySoundRequest, PlaySoundResponse } from './PlaySoundDtos' -export class PlaySoundService extends ObservableService{ - +export class PlaySoundService extends ObservableService< + PlaySoundRequest, + PlaySoundResponse +> { private soundService: SoundService private optionsRepository: OptionsRepository - constructor(soundService: SoundService, optionsRepository: OptionsRepository) { + constructor( + soundService: SoundService, + optionsRepository: OptionsRepository, + ) { super() this.soundService = soundService this.optionsRepository = optionsRepository } - protected async process(request: PlaySoundRequest): Promise { + protected async process( + request: PlaySoundRequest, + ): Promise { const options = await this.optionsRepository.getOptions() - if(options.muted) { + if (options.muted) { return {} } const pitchClass = PitchClass.create(request.pitchClass as PitchClassEnum) - await this.soundService.playSound(pitchClass.value, request.octave, options.instrument.name) + await this.soundService.playSound( + pitchClass.value, + request.octave, + options.instrument.name, + ) return { instrument: options.instrument.name } } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/PressKey/PressKeyDtos.ts b/apps/web/src/application/services/PressKey/PressKeyDtos.ts index 85bceab..fa041b8 100644 --- a/apps/web/src/application/services/PressKey/PressKeyDtos.ts +++ b/apps/web/src/application/services/PressKey/PressKeyDtos.ts @@ -5,4 +5,4 @@ export interface PressKeyRequest { export interface PressKeyResponse { instrument?: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/PressKey/PressKeyService.ts b/apps/web/src/application/services/PressKey/PressKeyService.ts index c8fa215..d50046c 100644 --- a/apps/web/src/application/services/PressKey/PressKeyService.ts +++ b/apps/web/src/application/services/PressKey/PressKeyService.ts @@ -3,12 +3,17 @@ import { ObservableService } from '../../ObservableService' import { PlaySoundService } from '../PlaySound/PlaySoundService' import { PressKeyRequest, PressKeyResponse } from './PressKeyDtos' -export class PressKeyService extends ObservableService{ - +export class PressKeyService extends ObservableService< + PressKeyRequest, + PressKeyResponse +> { private keyboardRepository: KeyboardRepository private playSound: PlaySoundService - constructor(keyboardRepository: KeyboardRepository, playSound: PlaySoundService) { + constructor( + keyboardRepository: KeyboardRepository, + playSound: PlaySoundService, + ) { super() this.keyboardRepository = keyboardRepository this.playSound = playSound @@ -16,20 +21,18 @@ export class PressKeyService extends ObservableService { const keyboard = this.keyboardRepository.getKeyboard() - if (!keyboard) - return {} + if (!keyboard) return {} const key = keyboard.findKey(request.pitchClass, request.octave) - if (!key) - return {} + if (!key) return {} key.press() const { instrument } = await this.playSound.execute({ pitchClass: request.pitchClass, - octave: request.octave + octave: request.octave, }) return { instrument } } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/RecalculateCubePositions/RecalculateCubePositionsService.ts b/apps/web/src/application/services/RecalculateCubePositions/RecalculateCubePositionsService.ts index 0607a52..48c90e2 100644 --- a/apps/web/src/application/services/RecalculateCubePositions/RecalculateCubePositionsService.ts +++ b/apps/web/src/application/services/RecalculateCubePositions/RecalculateCubePositionsService.ts @@ -1,25 +1,33 @@ import { CubeRepository } from '../../../domain/repositories/CubeRepository' import { ObservableService } from '../../ObservableService' -import { RecalculateCubePositionsRequestDto, RecalculateCubePositionsResponseDto } from './RecalculateCubePositionsDtos' +import { + RecalculateCubePositionsRequestDto, + RecalculateCubePositionsResponseDto, +} from './RecalculateCubePositionsDtos' -export class RecalculateCubePositionsService extends ObservableService{ +export class RecalculateCubePositionsService extends ObservableService< + RecalculateCubePositionsRequestDto, + RecalculateCubePositionsResponseDto +> { private readonly cubeRepository: CubeRepository - + constructor(cubeRepository: CubeRepository) { super() this.cubeRepository = cubeRepository } - protected async process(request: RecalculateCubePositionsRequestDto): Promise { + protected async process( + request: RecalculateCubePositionsRequestDto, + ): Promise { const cubes = await this.cubeRepository.list() - const response: RecalculateCubePositionsResponseDto = { + const response: RecalculateCubePositionsResponseDto = { deletedCubes: 0, - updatedCubes: 0 + updatedCubes: 0, } - for await(const cube of cubes) { + for await (const cube of cubes) { cube.recalculateYByAge(request.maxAge) - if(cube.isOnTop) { + if (cube.isOnTop) { await this.cubeRepository.delete(cube.uuid) response.deletedCubes++ } else { @@ -27,8 +35,7 @@ export class RecalculateCubePositionsService extends ObservableService{ - +export class ReleaseKeyService extends ObservableService< + ReleaseKeyRequest, + void +> { private keyboardRepository: KeyboardRepository private stopSound: StopSoundService - constructor(keyboardRepository: KeyboardRepository, stopSound: StopSoundService) { + constructor( + keyboardRepository: KeyboardRepository, + stopSound: StopSoundService, + ) { super() this.keyboardRepository = keyboardRepository this.stopSound = stopSound @@ -17,23 +22,21 @@ export class ReleaseKeyService extends ObservableService { const keyboard = this.keyboardRepository.getKeyboard() - if (!keyboard) - return + if (!keyboard) return const key = keyboard.findKey(request.pitchClass, request.octave) - if (!key) - return + if (!key) return key.release() - - if(!request.instrument) { + + if (!request.instrument) { return } this.stopSound.execute({ pitchClass: request.pitchClass, octave: request.octave, - instrument: request.instrument as InstrumentEnum + instrument: request.instrument as InstrumentEnum, }) } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/SetInstrument/SetInstrumentDtos.ts b/apps/web/src/application/services/SetInstrument/SetInstrumentDtos.ts index 246324f..c49f32b 100644 --- a/apps/web/src/application/services/SetInstrument/SetInstrumentDtos.ts +++ b/apps/web/src/application/services/SetInstrument/SetInstrumentDtos.ts @@ -1,3 +1,3 @@ export interface SetInstrumentRequest { instrument: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/SetInstrument/SetInstrumentService.ts b/apps/web/src/application/services/SetInstrument/SetInstrumentService.ts index 41723b5..427b342 100644 --- a/apps/web/src/application/services/SetInstrument/SetInstrumentService.ts +++ b/apps/web/src/application/services/SetInstrument/SetInstrumentService.ts @@ -5,8 +5,10 @@ import { Options } from '../../../domain/valueObjects/Options' import { ObservableService } from '../../ObservableService' import { SetInstrumentRequest } from './SetInstrumentDtos' -export class SetInstrumentService extends ObservableService{ - +export class SetInstrumentService extends ObservableService< + SetInstrumentRequest, + void +> { private optionsRepository: OptionsRepository constructor(optionsRepository: OptionsRepository) { @@ -19,8 +21,10 @@ export class SetInstrumentService extends ObservableService{ - +export class SetMutedService extends ObservableService { private optionsRepository: OptionsRepository constructor(optionsRepository: OptionsRepository) { @@ -17,8 +16,8 @@ export class SetMutedService extends ObservableService{ this.optionsRepository.setOptions( Options.create({ instrument: options.instrument, - muted: request.muted - }) + muted: request.muted, + }), ) } } diff --git a/apps/web/src/application/services/StopSound/StopSoundDtos.ts b/apps/web/src/application/services/StopSound/StopSoundDtos.ts index 223bd57..9746608 100644 --- a/apps/web/src/application/services/StopSound/StopSoundDtos.ts +++ b/apps/web/src/application/services/StopSound/StopSoundDtos.ts @@ -2,4 +2,4 @@ export interface StopSoundRequest { pitchClass: string octave: number instrument: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/StopSound/StopSoundService.ts b/apps/web/src/application/services/StopSound/StopSoundService.ts index c8ab2e6..eb29819 100644 --- a/apps/web/src/application/services/StopSound/StopSoundService.ts +++ b/apps/web/src/application/services/StopSound/StopSoundService.ts @@ -4,8 +4,10 @@ import { PitchClass } from '../../../domain/valueObjects/PitchClass' import { ObservableService } from '../../ObservableService' import { StopSoundRequest } from './StopSoundDtos' -export class StopSoundService extends ObservableService{ - +export class StopSoundService extends ObservableService< + StopSoundRequest, + void +> { private soundService: SoundService constructor(soundService: SoundService) { @@ -15,6 +17,10 @@ export class StopSoundService extends ObservableService{ protected async process(request: StopSoundRequest): Promise { const pitchClass = PitchClass.create(request.pitchClass as PitchClassEnum) - await this.soundService.stopSound(pitchClass.value, request.octave, request.instrument) + await this.soundService.stopSound( + pitchClass.value, + request.octave, + request.instrument, + ) } -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/SwitchNetwork/SwitchNetworkRequestDto.ts b/apps/web/src/application/services/SwitchNetwork/SwitchNetworkRequestDto.ts index 281ef9a..4b48cfe 100644 --- a/apps/web/src/application/services/SwitchNetwork/SwitchNetworkRequestDto.ts +++ b/apps/web/src/application/services/SwitchNetwork/SwitchNetworkRequestDto.ts @@ -1,3 +1,3 @@ export interface SwitchNetworkRequestDto { networkName: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/SwitchNetwork/SwitchNetworkResponseDto.ts b/apps/web/src/application/services/SwitchNetwork/SwitchNetworkResponseDto.ts index 46cd0ac..c728dd7 100644 --- a/apps/web/src/application/services/SwitchNetwork/SwitchNetworkResponseDto.ts +++ b/apps/web/src/application/services/SwitchNetwork/SwitchNetworkResponseDto.ts @@ -1,4 +1,4 @@ export interface SwitchNetworkResponseDto { - networkName: string, + networkName: string networkWsUrl: string -} \ No newline at end of file +} diff --git a/apps/web/src/application/services/SwitchNetwork/SwitchNetworkService.ts b/apps/web/src/application/services/SwitchNetwork/SwitchNetworkService.ts index dd20dd4..6489f17 100644 --- a/apps/web/src/application/services/SwitchNetwork/SwitchNetworkService.ts +++ b/apps/web/src/application/services/SwitchNetwork/SwitchNetworkService.ts @@ -7,7 +7,10 @@ import { ObservableService } from '../../ObservableService' import { SwitchNetworkRequestDto } from './SwitchNetworkRequestDto' import { SwitchNetworkResponseDto } from './SwitchNetworkResponseDto' -export class SwitchNetworkService extends ObservableService { +export class SwitchNetworkService extends ObservableService< + SwitchNetworkRequestDto, + SwitchNetworkResponseDto +> { private readonly transactionRepository: TransactionRepository private readonly networkRepository: NetworkRepository private readonly cubeRepository: CubeRepository @@ -15,7 +18,7 @@ export class SwitchNetworkService extends ObservableService { + protected async process( + request: SwitchNetworkRequestDto, + ): Promise { const network = await this.validateNetwork(request.networkName) await Promise.all([ this.transactionRepository.clear(), this.cubeRepository.clear(), - this.networkRepository.select(network.name) + this.networkRepository.select(network.name), ]) return { networkName: network.name, - networkWsUrl: network.wsUrl + networkWsUrl: network.wsUrl, } } @@ -42,7 +47,9 @@ export class SwitchNetworkService extends ObservableService { private constructor(color: TransactionColor, x: UnitInterval, uuid: Uuid) { - super({ - color, - x: x, - y: UnitInterval.create(0), - creation: Date.now(), - mirrored: Cube.randomMirrored() - }, uuid) + super( + { + color, + x: x, + y: UnitInterval.create(0), + creation: Date.now(), + mirrored: Cube.randomMirrored(), + }, + uuid, + ) } private static randomMirrored() { @@ -43,7 +46,10 @@ export class Cube extends Entity { return new Cube(color, x, Uuid.create()) } - private static isValidPosition(color: TransactionColor, x: UnitInterval): boolean { + private static isValidPosition( + color: TransactionColor, + x: UnitInterval, + ): boolean { const range = CubeXRange.get(color.value) if (!range) { diff --git a/apps/web/src/domain/entities/Network.spec.ts b/apps/web/src/domain/entities/Network.spec.ts index 9b4fd64..3a37057 100644 --- a/apps/web/src/domain/entities/Network.spec.ts +++ b/apps/web/src/domain/entities/Network.spec.ts @@ -1,39 +1,39 @@ -import { describe, it, expect } from 'vitest'; -import { Network } from './Network'; -import { NetworkEnum } from '../../../../../packages/shared/src/domain/enums/NetworkEnum'; -import { Uuid } from '../../../../../packages/shared/src/domain/valueObjects/Uuid'; +import { describe, it, expect } from 'vitest' +import { Network } from './Network' +import { NetworkEnum } from '../../../../../packages/shared/src/domain/enums/NetworkEnum' +import { Uuid } from '../../../../../packages/shared/src/domain/valueObjects/Uuid' describe('Network', () => { const mockProps = { name: NetworkEnum.MAINNET, explorerUrl: 'https://explorer.mainnet.com', wsUrl: 'wss://mainnet.ws.com', - }; + } it('should create a Network instance with default UUID', () => { - const network = Network.create(mockProps); - expect(network).toBeInstanceOf(Network); - expect(network.name).toBe(NetworkEnum.MAINNET); - expect(network.explorerUrl).toBe(mockProps.explorerUrl); - expect(network.wsUrl).toBe(mockProps.wsUrl); - expect(network.transactionUrl).toBe(`${mockProps.explorerUrl}/tx`); - expect(network.blockUrl).toBe(`${mockProps.explorerUrl}/block`); - }); + const network = Network.create(mockProps) + expect(network).toBeInstanceOf(Network) + expect(network.name).toBe(NetworkEnum.MAINNET) + expect(network.explorerUrl).toBe(mockProps.explorerUrl) + expect(network.wsUrl).toBe(mockProps.wsUrl) + expect(network.transactionUrl).toBe(`${mockProps.explorerUrl}/tx`) + expect(network.blockUrl).toBe(`${mockProps.explorerUrl}/block`) + }) it('should create a Network instance with a provided UUID', () => { - const uuid = Uuid.create(); - const network = Network.create(mockProps, uuid); - expect(network).toBeInstanceOf(Network); - expect(network.name).toBe(NetworkEnum.MAINNET); - expect(network.explorerUrl).toBe(mockProps.explorerUrl); - expect(network.wsUrl).toBe(mockProps.wsUrl); - expect(network.transactionUrl).toBe(`${mockProps.explorerUrl}/tx`); - expect(network.blockUrl).toBe(`${mockProps.explorerUrl}/block`); - }); + const uuid = Uuid.create() + const network = Network.create(mockProps, uuid) + expect(network).toBeInstanceOf(Network) + expect(network.name).toBe(NetworkEnum.MAINNET) + expect(network.explorerUrl).toBe(mockProps.explorerUrl) + expect(network.wsUrl).toBe(mockProps.wsUrl) + expect(network.transactionUrl).toBe(`${mockProps.explorerUrl}/tx`) + expect(network.blockUrl).toBe(`${mockProps.explorerUrl}/block`) + }) it('should return correct URLs for transactions and blocks', () => { - const network = Network.create(mockProps); - expect(network.transactionUrl).toBe(`${mockProps.explorerUrl}/tx`); - expect(network.blockUrl).toBe(`${mockProps.explorerUrl}/block`); - }); -}); + const network = Network.create(mockProps) + expect(network.transactionUrl).toBe(`${mockProps.explorerUrl}/tx`) + expect(network.blockUrl).toBe(`${mockProps.explorerUrl}/block`) + }) +}) diff --git a/apps/web/src/domain/entities/Network.ts b/apps/web/src/domain/entities/Network.ts index 9aa485d..3fbb69a 100644 --- a/apps/web/src/domain/entities/Network.ts +++ b/apps/web/src/domain/entities/Network.ts @@ -9,13 +9,12 @@ export interface NetworkProps { } export class Network extends Entity { - private constructor(props: NetworkProps, uuid: Uuid) { super(props, uuid) } static create(props: NetworkProps, id?: Uuid) { - const uuid = id ?? Uuid.create(); + const uuid = id ?? Uuid.create() return new Network(props, uuid) } @@ -38,4 +37,4 @@ export class Network extends Entity { get wsUrl() { return this.props.wsUrl } -} \ No newline at end of file +} diff --git a/apps/web/src/domain/entities/Transaction.spec.ts b/apps/web/src/domain/entities/Transaction.spec.ts index 97ebead..a7da207 100644 --- a/apps/web/src/domain/entities/Transaction.spec.ts +++ b/apps/web/src/domain/entities/Transaction.spec.ts @@ -1,62 +1,62 @@ -import { describe, it, expect } from 'vitest'; -import { Transaction } from './Transaction'; -import { NetworkEnum, TxType } from '@cryptochords/shared'; -import { Network } from './Network'; -import { Address } from '@cryptochords/shared'; -import { Timestamp } from '@cryptochords/shared'; +import { describe, it, expect } from 'vitest' +import { Transaction } from './Transaction' +import { NetworkEnum, TxType } from '@cryptochords/shared' +import { Network } from './Network' +import { Address } from '@cryptochords/shared' +import { Timestamp } from '@cryptochords/shared' describe('Transaction', () => { const mockNetwork = Network.create({ name: NetworkEnum.MAINNET, explorerUrl: 'https://explorer.mainnet.com', - wsUrl: 'wss://mainnet.ws.com' - }); + wsUrl: 'wss://mainnet.ws.com', + }) - const mockAddress = { value: '0x1234567890abcdef' } as Address; + const mockAddress = { value: '0x1234567890abcdef' } as Address const mockTimestamp = Timestamp.create(new Date().getTime()) it('should create a Transaction instance', () => { - const txType = { isBlock: false } as TxType; + const txType = { isBlock: false } as TxType const transaction = Transaction.create({ txType, address: mockAddress, network: mockNetwork, - timestamp: mockTimestamp - }); + timestamp: mockTimestamp, + }) - expect(transaction).toBeInstanceOf(Transaction); - expect(transaction.txType).toEqual(txType); - expect(transaction.address).toEqual(mockAddress); - expect(transaction.network).toEqual(mockNetwork); - expect(transaction.timestamp).toEqual(mockTimestamp); - }); + expect(transaction).toBeInstanceOf(Transaction) + expect(transaction.txType).toEqual(txType) + expect(transaction.address).toEqual(mockAddress) + expect(transaction.network).toEqual(mockNetwork) + expect(transaction.timestamp).toEqual(mockTimestamp) + }) it('should return correct block URL when txType.isBlock is true', () => { - const txType = { isBlock: true } as TxType; + const txType = { isBlock: true } as TxType const transaction = Transaction.create({ txType, address: mockAddress, network: mockNetwork, - timestamp: mockTimestamp - }); + timestamp: mockTimestamp, + }) - const expectedUrl = `${mockNetwork.blockUrl}/${mockAddress.value}`; - expect(transaction.url).toBe(expectedUrl); - }); + const expectedUrl = `${mockNetwork.blockUrl}/${mockAddress.value}` + expect(transaction.url).toBe(expectedUrl) + }) it('should return correct transaction URL when txType.isBlock is false', () => { - const txType = { isBlock: false } as TxType; + const txType = { isBlock: false } as TxType const transaction = Transaction.create({ txType, address: mockAddress, network: mockNetwork, - timestamp: mockTimestamp - }); + timestamp: mockTimestamp, + }) - const expectedUrl = `${mockNetwork.transactionUrl}/${mockAddress.value}`; - expect(transaction.url).toBe(expectedUrl); - }); -}); \ No newline at end of file + const expectedUrl = `${mockNetwork.transactionUrl}/${mockAddress.value}` + expect(transaction.url).toBe(expectedUrl) + }) +}) diff --git a/apps/web/src/domain/entities/Transaction.ts b/apps/web/src/domain/entities/Transaction.ts index 8c7f742..4e3f460 100644 --- a/apps/web/src/domain/entities/Transaction.ts +++ b/apps/web/src/domain/entities/Transaction.ts @@ -36,10 +36,10 @@ export class Transaction { } get url() { - if(this.props.txType.isBlock) { + if (this.props.txType.isBlock) { return `${this.props.network.blockUrl}/${this.props.address.value}` } return `${this.props.network.transactionUrl}/${this.props.address.value}` } -} \ No newline at end of file +} diff --git a/apps/web/src/domain/enum/InstrumentEnum.ts b/apps/web/src/domain/enum/InstrumentEnum.ts index 929dee4..a2660f3 100644 --- a/apps/web/src/domain/enum/InstrumentEnum.ts +++ b/apps/web/src/domain/enum/InstrumentEnum.ts @@ -40,4 +40,4 @@ export const instrumentLabels: Map = new Map([ [InstrumentEnum.Tuba, 'Tuba'], [InstrumentEnum.Violin, 'Violin'], [InstrumentEnum.Xylophone, 'Xylophone'], -]) \ No newline at end of file +]) diff --git a/apps/web/src/domain/enum/KeyShapeEnum.ts b/apps/web/src/domain/enum/KeyShapeEnum.ts index 3412eec..b9e0f4c 100644 --- a/apps/web/src/domain/enum/KeyShapeEnum.ts +++ b/apps/web/src/domain/enum/KeyShapeEnum.ts @@ -3,5 +3,5 @@ export enum KeyShapeEnum { White = 'white', WhiteMiddle = 'white-middle', WhiteLeft = 'white-left', - WhiteRight = 'white-right' -} \ No newline at end of file + WhiteRight = 'white-right', +} diff --git a/apps/web/src/domain/enum/PitchClassEnum.ts b/apps/web/src/domain/enum/PitchClassEnum.ts index 0f8bfc2..7a56940 100644 --- a/apps/web/src/domain/enum/PitchClassEnum.ts +++ b/apps/web/src/domain/enum/PitchClassEnum.ts @@ -10,5 +10,5 @@ export enum PitchClassEnum { G_SHARP = 'G#', A = 'A', A_SHARP = 'A#', - B = 'B' + B = 'B', } diff --git a/apps/web/src/domain/enum/TransactionColorEnum.ts b/apps/web/src/domain/enum/TransactionColorEnum.ts index f4d415c..9973488 100644 --- a/apps/web/src/domain/enum/TransactionColorEnum.ts +++ b/apps/web/src/domain/enum/TransactionColorEnum.ts @@ -2,5 +2,5 @@ export enum TransactionColorEnum { Orange = 'orange', Blue = 'blue', Purple = 'purple', - Green = 'green' + Green = 'green', } diff --git a/apps/web/src/domain/events/TransactionCreatedEvent.ts b/apps/web/src/domain/events/TransactionCreatedEvent.ts index 6be77c8..39b9f11 100644 --- a/apps/web/src/domain/events/TransactionCreatedEvent.ts +++ b/apps/web/src/domain/events/TransactionCreatedEvent.ts @@ -2,7 +2,7 @@ import { Event } from '@cryptochords/shared' import { Transaction } from '../entities/Transaction' export class TransactionCreatedEvent extends Event { - static eventKey:symbol = Symbol('TransactionCreatedEvent') + static eventKey: symbol = Symbol('TransactionCreatedEvent') private readonly transaction: Transaction @@ -18,4 +18,4 @@ export class TransactionCreatedEvent extends Event { getTransaction(): Transaction { return this.transaction } -} \ No newline at end of file +} diff --git a/apps/web/src/domain/factories/KeyboardFactory.test.ts b/apps/web/src/domain/factories/KeyboardFactory.test.ts index ffac32a..3b1a231 100644 --- a/apps/web/src/domain/factories/KeyboardFactory.test.ts +++ b/apps/web/src/domain/factories/KeyboardFactory.test.ts @@ -12,7 +12,7 @@ describe('src/domain/factories/KeyboardFactory', () => { const keyboard = KeyboardFactory.create({ initialOctave: 0, numberOfKeys: 88, - initialPitchClass: PitchClassEnum.A + initialPitchClass: PitchClassEnum.A, }) expect(keyboard.keys.length).toBe(88) }) @@ -21,7 +21,7 @@ describe('src/domain/factories/KeyboardFactory', () => { const keyboard = KeyboardFactory.create({ initialOctave: 1, numberOfKeys: 12, - initialPitchClass: PitchClassEnum.F + initialPitchClass: PitchClassEnum.F, }) expect(keyboard.keys[0].pitch.pitchClass.value).toBe(PitchClassEnum.F) expect(keyboard.keys[0].pitch.octave).toBe(1) @@ -30,43 +30,45 @@ describe('src/domain/factories/KeyboardFactory', () => { expect(keyboard.keys[1].pitch.pitchClass.value).toBe(PitchClassEnum.F_SHARP) expect(keyboard.keys[1].pitch.octave).toBe(1) expect(keyboard.keys[1].keyShape.value).toBe(KeyShapeEnum.Black) - + expect(keyboard.keys[2].pitch.pitchClass.value).toBe(PitchClassEnum.G) expect(keyboard.keys[2].pitch.octave).toBe(1) expect(keyboard.keys[2].keyShape.value).toBe(KeyShapeEnum.WhiteMiddle) - + expect(keyboard.keys[3].pitch.pitchClass.value).toBe(PitchClassEnum.G_SHARP) expect(keyboard.keys[3].pitch.octave).toBe(1) expect(keyboard.keys[3].keyShape.value).toBe(KeyShapeEnum.Black) - + expect(keyboard.keys[4].pitch.pitchClass.value).toBe(PitchClassEnum.A) expect(keyboard.keys[4].pitch.octave).toBe(1) expect(keyboard.keys[4].keyShape.value).toBe(KeyShapeEnum.WhiteMiddle) - + expect(keyboard.keys[5].pitch.pitchClass.value).toBe(PitchClassEnum.A_SHARP) expect(keyboard.keys[5].pitch.octave).toBe(1) expect(keyboard.keys[5].keyShape.value).toBe(KeyShapeEnum.Black) - + expect(keyboard.keys[6].pitch.pitchClass.value).toBe(PitchClassEnum.B) expect(keyboard.keys[6].pitch.octave).toBe(1) expect(keyboard.keys[6].keyShape.value).toBe(KeyShapeEnum.WhiteRight) - + expect(keyboard.keys[7].pitch.pitchClass.value).toBe(PitchClassEnum.C) expect(keyboard.keys[7].pitch.octave).toBe(2) expect(keyboard.keys[7].keyShape.value).toBe(KeyShapeEnum.WhiteLeft) - + expect(keyboard.keys[8].pitch.pitchClass.value).toBe(PitchClassEnum.C_SHARP) expect(keyboard.keys[8].pitch.octave).toBe(2) expect(keyboard.keys[8].keyShape.value).toBe(KeyShapeEnum.Black) - + expect(keyboard.keys[9].pitch.pitchClass.value).toBe(PitchClassEnum.D) expect(keyboard.keys[9].pitch.octave).toBe(2) expect(keyboard.keys[9].keyShape.value).toBe(KeyShapeEnum.WhiteMiddle) - - expect(keyboard.keys[10].pitch.pitchClass.value).toBe(PitchClassEnum.D_SHARP) + + expect(keyboard.keys[10].pitch.pitchClass.value).toBe( + PitchClassEnum.D_SHARP, + ) expect(keyboard.keys[10].pitch.octave).toBe(2) expect(keyboard.keys[10].keyShape.value).toBe(KeyShapeEnum.Black) - + expect(keyboard.keys[11].pitch.pitchClass.value).toBe(PitchClassEnum.E) expect(keyboard.keys[11].pitch.octave).toBe(2) expect(keyboard.keys[11].keyShape.value).toBe(KeyShapeEnum.WhiteRight) @@ -76,7 +78,7 @@ describe('src/domain/factories/KeyboardFactory', () => { const keyboard = KeyboardFactory.create({ initialOctave: 1, numberOfKeys: 2, - initialPitchClass: PitchClassEnum.B + initialPitchClass: PitchClassEnum.B, }) expect(keyboard.keys[1].keyShape.value).toBe(KeyShapeEnum.White) }) @@ -85,7 +87,7 @@ describe('src/domain/factories/KeyboardFactory', () => { const keyboard = KeyboardFactory.create({ initialOctave: 1, numberOfKeys: 2, - initialPitchClass: PitchClassEnum.E + initialPitchClass: PitchClassEnum.E, }) expect(keyboard.keys[1].keyShape.value).toBe(KeyShapeEnum.White) }) @@ -94,16 +96,16 @@ describe('src/domain/factories/KeyboardFactory', () => { const keyboard = KeyboardFactory.create({ initialOctave: 1, numberOfKeys: 3, - initialPitchClass: PitchClassEnum.C + initialPitchClass: PitchClassEnum.C, }) expect(keyboard.keys[2].keyShape.value).toBe(KeyShapeEnum.WhiteRight) }) - + it('should return a keyboard with a white-right last key when it finish with a G key', () => { const keyboard = KeyboardFactory.create({ initialOctave: 1, numberOfKeys: 3, - initialPitchClass: PitchClassEnum.F + initialPitchClass: PitchClassEnum.F, }) expect(keyboard.keys[2].keyShape.value).toBe(KeyShapeEnum.WhiteRight) }) @@ -112,7 +114,7 @@ describe('src/domain/factories/KeyboardFactory', () => { const keyboard = KeyboardFactory.create({ initialOctave: 1, numberOfKeys: 3, - initialPitchClass: PitchClassEnum.G + initialPitchClass: PitchClassEnum.G, }) expect(keyboard.keys[2].keyShape.value).toBe(KeyShapeEnum.WhiteRight) }) diff --git a/apps/web/src/domain/factories/KeyboardFactory.ts b/apps/web/src/domain/factories/KeyboardFactory.ts index ee67555..fe7ff3f 100644 --- a/apps/web/src/domain/factories/KeyboardFactory.ts +++ b/apps/web/src/domain/factories/KeyboardFactory.ts @@ -8,11 +8,10 @@ import { Pitch } from '../valueObjects/Pitch' import { PitchClass } from '../valueObjects/PitchClass' import { UnitInterval } from '../valueObjects/UnitInterval' - enum KeyPosition { Beginning, Middle, - End + End, } export interface CreateProps { @@ -23,14 +22,23 @@ export interface CreateProps { export class KeyboardFactory { static create(props: CreateProps) { - const keysProps: Pick[] = [] + const keysProps: Pick< + KeyProps, + 'color' | 'keyShape' | 'pitch' | 'pressed' + >[] = [] let pitchClassValue = props.initialPitchClass let octave = props.initialOctave for (let i = 0; i < props.numberOfKeys; i++) { const isLastKey = i === props.numberOfKeys - 1 const isFirstKey = i === 0 - const position = isFirstKey ? KeyPosition.Beginning : isLastKey ? KeyPosition.End : KeyPosition.Middle - const keyShape = KeyShape.create(KeyboardFactory.getKeyShape(pitchClassValue, position)) + const position = isFirstKey + ? KeyPosition.Beginning + : isLastKey + ? KeyPosition.End + : KeyPosition.Middle + const keyShape = KeyShape.create( + KeyboardFactory.getKeyShape(pitchClassValue, position), + ) const pitchClass = PitchClass.create(pitchClassValue) const pitch = Pitch.create({ pitchClass: pitchClass, octave }) const color = KeyboardFactory.getColor(props.numberOfKeys, i) @@ -55,8 +63,12 @@ export class KeyboardFactory { return Object.values(TransactionColorEnum)[keyQuarter] } - private static calculatePositions(keysProps: Pick[]) { - const whiteKeysCount = keysProps.filter(key => key.keyShape.value !== KeyShapeEnum.Black).length + private static calculatePositions( + keysProps: Pick[], + ) { + const whiteKeysCount = keysProps.filter( + key => key.keyShape.value !== KeyShapeEnum.Black, + ).length const minKeyCenterSpacing = 1 / (whiteKeysCount * 2 - 1) const result = [] let x = 0 @@ -64,7 +76,10 @@ export class KeyboardFactory { const shape = keysProps[i].keyShape.value if (i === 0) { x += minKeyCenterSpacing * 0.75 - } else if (shape === KeyShapeEnum.WhiteLeft || shape === KeyShapeEnum.White) { + } else if ( + shape === KeyShapeEnum.WhiteLeft || + shape === KeyShapeEnum.White + ) { x += minKeyCenterSpacing * 2 } else { x += minKeyCenterSpacing @@ -103,7 +118,10 @@ export class KeyboardFactory { } } - private static getKeyShape(pitchClass: PitchClassEnum, position: KeyPosition): KeyShapeEnum { + private static getKeyShape( + pitchClass: PitchClassEnum, + position: KeyPosition, + ): KeyShapeEnum { if (pitchClass === PitchClassEnum.C || pitchClass === PitchClassEnum.F) { if (position === KeyPosition.End) { return KeyShapeEnum.White @@ -118,7 +136,11 @@ export class KeyboardFactory { return KeyShapeEnum.WhiteRight } - if (pitchClass === PitchClassEnum.D || pitchClass === PitchClassEnum.G || pitchClass === PitchClassEnum.A) { + if ( + pitchClass === PitchClassEnum.D || + pitchClass === PitchClassEnum.G || + pitchClass === PitchClassEnum.A + ) { if (position === KeyPosition.End) { return KeyShapeEnum.WhiteRight } diff --git a/apps/web/src/domain/repositories/CubeRepository.ts b/apps/web/src/domain/repositories/CubeRepository.ts index ff531b4..218f334 100644 --- a/apps/web/src/domain/repositories/CubeRepository.ts +++ b/apps/web/src/domain/repositories/CubeRepository.ts @@ -7,4 +7,4 @@ export interface CubeRepository { update: (cube: Cube) => Promise delete: (id: Uuid) => Promise clear: () => Promise -} \ No newline at end of file +} diff --git a/apps/web/src/domain/repositories/KeyboardRepository.ts b/apps/web/src/domain/repositories/KeyboardRepository.ts index 063e7a4..62515c3 100644 --- a/apps/web/src/domain/repositories/KeyboardRepository.ts +++ b/apps/web/src/domain/repositories/KeyboardRepository.ts @@ -3,4 +3,4 @@ import { Keyboard } from '../valueObjects/Keyboard' export interface KeyboardRepository { setKeyboard: (keyboard: Keyboard) => void getKeyboard: () => Keyboard | undefined -} \ No newline at end of file +} diff --git a/apps/web/src/domain/repositories/NetworkRepository.ts b/apps/web/src/domain/repositories/NetworkRepository.ts index f560b66..c6cbd93 100644 --- a/apps/web/src/domain/repositories/NetworkRepository.ts +++ b/apps/web/src/domain/repositories/NetworkRepository.ts @@ -6,4 +6,4 @@ export interface NetworkRepository { find: (name: NetworkEnum) => Promise select: (name: NetworkEnum) => Promise getSelected: () => Promise -} \ No newline at end of file +} diff --git a/apps/web/src/domain/repositories/OptionsRepository.ts b/apps/web/src/domain/repositories/OptionsRepository.ts index 9ae5f43..1940116 100644 --- a/apps/web/src/domain/repositories/OptionsRepository.ts +++ b/apps/web/src/domain/repositories/OptionsRepository.ts @@ -3,4 +3,4 @@ import { Options } from '../valueObjects/Options' export interface OptionsRepository { setOptions: (options: Options) => Promise getOptions: () => Promise -} \ No newline at end of file +} diff --git a/apps/web/src/domain/repositories/TransactionRepository.ts b/apps/web/src/domain/repositories/TransactionRepository.ts index 0cd9486..c7398f1 100644 --- a/apps/web/src/domain/repositories/TransactionRepository.ts +++ b/apps/web/src/domain/repositories/TransactionRepository.ts @@ -4,4 +4,4 @@ export interface TransactionRepository { create(block: Transaction): Promise list(): Promise clear(): Promise -} \ No newline at end of file +} diff --git a/apps/web/src/domain/services/SoundService.ts b/apps/web/src/domain/services/SoundService.ts index 5ae5fb3..fa1d0c2 100644 --- a/apps/web/src/domain/services/SoundService.ts +++ b/apps/web/src/domain/services/SoundService.ts @@ -2,6 +2,14 @@ import { PitchClassEnum } from '../enum/PitchClassEnum' export interface SoundService { loadInstrument(instrument: string): Promise - playSound(pitch: PitchClassEnum, octave: number, instrument: string): Promise - stopSound(pitch: PitchClassEnum, octave: number, instrument: string): Promise + playSound( + pitch: PitchClassEnum, + octave: number, + instrument: string, + ): Promise + stopSound( + pitch: PitchClassEnum, + octave: number, + instrument: string, + ): Promise } diff --git a/apps/web/src/domain/valueObjects/Instrument.ts b/apps/web/src/domain/valueObjects/Instrument.ts index 83639ae..586f2b7 100644 --- a/apps/web/src/domain/valueObjects/Instrument.ts +++ b/apps/web/src/domain/valueObjects/Instrument.ts @@ -5,7 +5,7 @@ export interface InstrumentProps { name: InstrumentEnum } -export class Instrument extends ValueObject{ +export class Instrument extends ValueObject { private constructor(props: InstrumentProps) { super(props) } diff --git a/apps/web/src/domain/valueObjects/KeyShape.ts b/apps/web/src/domain/valueObjects/KeyShape.ts index 742906f..bab6c26 100644 --- a/apps/web/src/domain/valueObjects/KeyShape.ts +++ b/apps/web/src/domain/valueObjects/KeyShape.ts @@ -5,7 +5,7 @@ interface KeyShapeProps { value: KeyShapeEnum } -export class KeyShape extends ValueObject{ +export class KeyShape extends ValueObject { private constructor(shape: KeyShapeEnum) { super({ value: shape }) } @@ -17,4 +17,4 @@ export class KeyShape extends ValueObject{ get value() { return this.props.value } -} \ No newline at end of file +} diff --git a/apps/web/src/domain/valueObjects/Keyboard.ts b/apps/web/src/domain/valueObjects/Keyboard.ts index eeaa035..956db3b 100644 --- a/apps/web/src/domain/valueObjects/Keyboard.ts +++ b/apps/web/src/domain/valueObjects/Keyboard.ts @@ -20,9 +20,10 @@ export class Keyboard extends ValueObject { } findKey(pitchClass: string, octave: number): Key | undefined { - return this.props.keys.find((key) => - key.pitch.pitchClass.value === pitchClass - && key.pitch.octave === octave + return this.props.keys.find( + key => + key.pitch.pitchClass.value === pitchClass && + key.pitch.octave === octave, ) } @@ -30,10 +31,19 @@ export class Keyboard extends ValueObject { if (txType === TxTypesEnum.Btc) return this.getRandomWhiteKey(0, Math.floor(this.props.keys.length / 4)) if (txType === TxTypesEnum.Eth) - return this.getRandomWhiteKey(Math.floor(this.props.keys.length / 4), Math.floor(this.props.keys.length / 2)) + return this.getRandomWhiteKey( + Math.floor(this.props.keys.length / 4), + Math.floor(this.props.keys.length / 2), + ) if (txType === TxTypesEnum.Pop) - return this.getRandomWhiteKey(Math.floor(this.props.keys.length / 2), Math.floor(this.props.keys.length * 3 / 4)) - return this.getRandomWhiteKey(Math.floor(this.props.keys.length * 3 / 4), this.props.keys.length) + return this.getRandomWhiteKey( + Math.floor(this.props.keys.length / 2), + Math.floor((this.props.keys.length * 3) / 4), + ) + return this.getRandomWhiteKey( + Math.floor((this.props.keys.length * 3) / 4), + this.props.keys.length, + ) } getRandomKey(min: number, max: number): Key { @@ -43,7 +53,9 @@ export class Keyboard extends ValueObject { getRandomWhiteKey(min: number, max: number): Key { const keysInTheRange = this.props.keys.slice(min, max) - const whiteKeys = keysInTheRange.filter((key) => key.keyShape.value !== KeyShapeEnum.Black ) + const whiteKeys = keysInTheRange.filter( + key => key.keyShape.value !== KeyShapeEnum.Black, + ) const randomIndex = Math.floor(Math.random() * whiteKeys.length) return whiteKeys[randomIndex] } diff --git a/apps/web/src/domain/valueObjects/Options.ts b/apps/web/src/domain/valueObjects/Options.ts index 4759d96..a5ee7ac 100644 --- a/apps/web/src/domain/valueObjects/Options.ts +++ b/apps/web/src/domain/valueObjects/Options.ts @@ -12,7 +12,7 @@ export interface OptionsJSON { instrument: string } -export class Options extends ValueObject{ +export class Options extends ValueObject { private constructor(props: OptionsProps) { super(props) } @@ -32,14 +32,16 @@ export class Options extends ValueObject{ toJSON(): OptionsJSON { return { muted: this.props.muted, - instrument: this.props.instrument.name + instrument: this.props.instrument.name, } } static fromJSON(json: OptionsJSON) { return Options.create({ muted: json.muted, - instrument: Instrument.create({ name: json.instrument as InstrumentEnum}) + instrument: Instrument.create({ + name: json.instrument as InstrumentEnum, + }), }) } } diff --git a/apps/web/src/domain/valueObjects/Pitch.ts b/apps/web/src/domain/valueObjects/Pitch.ts index 1d03500..e56526f 100644 --- a/apps/web/src/domain/valueObjects/Pitch.ts +++ b/apps/web/src/domain/valueObjects/Pitch.ts @@ -27,7 +27,8 @@ export class Pitch extends ValueObject { const nextClass = this.props.pitchClass.next return Pitch.create({ pitchClass: nextClass, - octave: nextClass.value === 'C' ? this.props.octave + 1 : this.props.octave + octave: + nextClass.value === 'C' ? this.props.octave + 1 : this.props.octave, }) } @@ -35,39 +36,51 @@ export class Pitch extends ValueObject { const previousClass = this.props.pitchClass.previous return Pitch.create({ pitchClass: previousClass, - octave: previousClass.value === 'B' ? this.props.octave - 1 : this.props.octave + octave: + previousClass.value === 'B' ? this.props.octave - 1 : this.props.octave, }) } isBefore(pitch: Pitch) { - return this.props.octave < pitch.octave || (this.props.octave === pitch.octave && this.props.pitchClass.value < pitch.pitchClass.value) + return ( + this.props.octave < pitch.octave || + (this.props.octave === pitch.octave && + this.props.pitchClass.value < pitch.pitchClass.value) + ) } isAfter(pitch: Pitch) { - return this.props.octave > pitch.octave || (this.props.octave === pitch.octave && this.props.pitchClass.value > pitch.pitchClass.value) + return ( + this.props.octave > pitch.octave || + (this.props.octave === pitch.octave && + this.props.pitchClass.value > pitch.pitchClass.value) + ) } equals(pitch: Pitch) { - return this.props.octave === pitch.octave && this.props.pitchClass.value === pitch.pitchClass.value + return ( + this.props.octave === pitch.octave && + this.props.pitchClass.value === pitch.pitchClass.value + ) } toString() { return `${this.props.pitchClass.value}${this.props.octave}` - } - - static random(min: Pitch, max: Pitch) { + } + + static random(min: Pitch, max: Pitch) { const range = Pitch.range(min, max) return range[Math.floor(Math.random() * range.length)] } static range(min: Pitch, max: Pitch): Pitch[] { - if(min.isAfter(max)) { + if (min.isAfter(max)) { throw new Error('Min pitch must be before max pitch') } - - const possibleValues:Pitch[] = [] + + const possibleValues: Pitch[] = [] let current = min - while(!current.isAfter(max)) { + while (!current.isAfter(max)) { possibleValues.push(current) current = current.next } diff --git a/apps/web/src/domain/valueObjects/PitchClass.spec.ts b/apps/web/src/domain/valueObjects/PitchClass.spec.ts index b185194..d2b3b3d 100644 --- a/apps/web/src/domain/valueObjects/PitchClass.spec.ts +++ b/apps/web/src/domain/valueObjects/PitchClass.spec.ts @@ -18,4 +18,4 @@ describe('src/domain/valueObjects/PitchClass', () => { PitchClass.create('Z' as PitchClassEnum) }).toThrow(InvalidPitchClassError) }) -}) \ No newline at end of file +}) diff --git a/apps/web/src/domain/valueObjects/PitchClass.ts b/apps/web/src/domain/valueObjects/PitchClass.ts index 23e4fd1..b67caa1 100644 --- a/apps/web/src/domain/valueObjects/PitchClass.ts +++ b/apps/web/src/domain/valueObjects/PitchClass.ts @@ -27,12 +27,18 @@ export class PitchClass extends ValueObject { } get next() { - const nextIndex = (Object.values(PitchClassEnum).indexOf(this.props.value) + 1) % Object.values(PitchClassEnum).length + const nextIndex = + (Object.values(PitchClassEnum).indexOf(this.props.value) + 1) % + Object.values(PitchClassEnum).length return PitchClass.create(Object.values(PitchClassEnum)[nextIndex]) } get previous() { - const previousIndex = (Object.values(PitchClassEnum).indexOf(this.props.value) - 1 + Object.values(PitchClassEnum).length) % Object.values(PitchClassEnum).length + const previousIndex = + (Object.values(PitchClassEnum).indexOf(this.props.value) - + 1 + + Object.values(PitchClassEnum).length) % + Object.values(PitchClassEnum).length return PitchClass.create(Object.values(PitchClassEnum)[previousIndex]) } } diff --git a/apps/web/src/domain/valueObjects/TransactionColor.spec.ts b/apps/web/src/domain/valueObjects/TransactionColor.spec.ts index 8dcd138..939f8f1 100644 --- a/apps/web/src/domain/valueObjects/TransactionColor.spec.ts +++ b/apps/web/src/domain/valueObjects/TransactionColor.spec.ts @@ -22,5 +22,4 @@ describe('src/domain/valueObjects/CubeColor', () => { expect(color.value).toBeDefined() }) }) - -}) \ No newline at end of file +}) diff --git a/apps/web/src/domain/valueObjects/UnitInterval.ts b/apps/web/src/domain/valueObjects/UnitInterval.ts index 46d67ec..a81e44e 100644 --- a/apps/web/src/domain/valueObjects/UnitInterval.ts +++ b/apps/web/src/domain/valueObjects/UnitInterval.ts @@ -1,7 +1,6 @@ import { ValueObject } from '@cryptochords/shared' import { InvalidUnitIntervalError } from '../errors/InvalidUnitIntervalError' - interface UnitIntevalProps { value: number } @@ -12,9 +11,7 @@ export class UnitInterval extends ValueObject { } static create(value: number) { - if (typeof value !== 'number' || - value < 0 || - value > 1) { + if (typeof value !== 'number' || value < 0 || value > 1) { throw new InvalidUnitIntervalError() } @@ -33,7 +30,9 @@ export class UnitInterval extends ValueObject { const minSanitized = Math.max(0, min) const maxSanitized = Math.min(1, max) - return this.create(Math.random() * (maxSanitized - minSanitized) + minSanitized) + return this.create( + Math.random() * (maxSanitized - minSanitized) + minSanitized, + ) } increment(value: number) { diff --git a/apps/web/src/domain/valueObjects/UnitIntervalRange.ts b/apps/web/src/domain/valueObjects/UnitIntervalRange.ts index 1d18d9b..42fa9f5 100644 --- a/apps/web/src/domain/valueObjects/UnitIntervalRange.ts +++ b/apps/web/src/domain/valueObjects/UnitIntervalRange.ts @@ -1,9 +1,8 @@ import { ValueObject } from '@cryptochords/shared' import { UnitInterval } from './UnitInterval' - interface UnitIntevalRangeProps { - min: UnitInterval, + min: UnitInterval max: UnitInterval } @@ -13,7 +12,10 @@ export class UnitIntervalRange extends ValueObject { } static create(min: number, max: number) { - return new UnitIntervalRange(UnitInterval.create(min), UnitInterval.create(max)) + return new UnitIntervalRange( + UnitInterval.create(min), + UnitInterval.create(max), + ) } get min() { @@ -23,5 +25,4 @@ export class UnitIntervalRange extends ValueObject { get max() { return this.props.max.value } - } diff --git a/apps/web/src/infrastructure/repositories/LimitedInMemoryTransactionRepository.ts b/apps/web/src/infrastructure/repositories/LimitedInMemoryTransactionRepository.ts index 894334f..b0b1bb4 100644 --- a/apps/web/src/infrastructure/repositories/LimitedInMemoryTransactionRepository.ts +++ b/apps/web/src/infrastructure/repositories/LimitedInMemoryTransactionRepository.ts @@ -1,7 +1,9 @@ import { Transaction } from '../../domain/entities/Transaction' import { TransactionRepository } from '../../domain/repositories/TransactionRepository' -export class LimitedInMemoryTransactionRepository implements TransactionRepository { +export class LimitedInMemoryTransactionRepository + implements TransactionRepository +{ private transactions: Transaction[] = [] constructor(private max = 99) { @@ -26,4 +28,4 @@ export class LimitedInMemoryTransactionRepository implements TransactionReposito async clear(): Promise { this.transactions = [] } -} \ No newline at end of file +} diff --git a/apps/web/src/infrastructure/repositories/LocalStorageOptionsRepository.ts b/apps/web/src/infrastructure/repositories/LocalStorageOptionsRepository.ts index f090a2d..e69d739 100644 --- a/apps/web/src/infrastructure/repositories/LocalStorageOptionsRepository.ts +++ b/apps/web/src/infrastructure/repositories/LocalStorageOptionsRepository.ts @@ -16,7 +16,7 @@ export class LocalStorageOptionsRepository implements OptionsRepository { return Options.create({ muted: true, - instrument: Instrument.create({ name: InstrumentEnum.Piano }) + instrument: Instrument.create({ name: InstrumentEnum.Piano }), }) } } diff --git a/apps/web/src/infrastructure/repositories/StaticNetworksRepository.ts b/apps/web/src/infrastructure/repositories/StaticNetworksRepository.ts index 0622d30..00eb899 100644 --- a/apps/web/src/infrastructure/repositories/StaticNetworksRepository.ts +++ b/apps/web/src/infrastructure/repositories/StaticNetworksRepository.ts @@ -3,23 +3,19 @@ import { NetworkRepository } from '../../domain/repositories/NetworkRepository' import { Network } from '../../domain/entities/Network' export class StaticNetworksRepository implements NetworkRepository { - private readonly testnet = Network.create({ name: NetworkEnum.TESTNET, explorerUrl: HemiTestnet.blockExplorers.default.url, - wsUrl: StaticNetworksRepository.buildWebserviceUrl(NetworkEnum.TESTNET) + wsUrl: StaticNetworksRepository.buildWebserviceUrl(NetworkEnum.TESTNET), }) private readonly mainnet = Network.create({ name: NetworkEnum.MAINNET, explorerUrl: HemiMainnet.blockExplorers.default.url, - wsUrl: StaticNetworksRepository.buildWebserviceUrl(NetworkEnum.MAINNET) + wsUrl: StaticNetworksRepository.buildWebserviceUrl(NetworkEnum.MAINNET), }) - private networks = [ - this.mainnet, - this.testnet - ] + private networks = [this.mainnet, this.testnet] private selectedNetwork: Network = this.testnet @@ -28,9 +24,8 @@ export class StaticNetworksRepository implements NetworkRepository { } async find(name: NetworkEnum) { - if(!name) - return undefined - + if (!name) return undefined + return this.networks.find(network => network.name === name) } @@ -53,8 +48,7 @@ export class StaticNetworksRepository implements NetworkRepository { async select(name: NetworkEnum) { const network = await this.find(name) - if(!network) - throw new Error('Network not found') + if (!network) throw new Error('Network not found') this.selectedNetwork = network } @@ -62,4 +56,4 @@ export class StaticNetworksRepository implements NetworkRepository { async getSelected() { return this.selectedNetwork } -} \ No newline at end of file +} diff --git a/apps/web/src/infrastructure/services/ToneJs.ts b/apps/web/src/infrastructure/services/ToneJs.ts index f7a4db1..94aa902 100644 --- a/apps/web/src/infrastructure/services/ToneJs.ts +++ b/apps/web/src/infrastructure/services/ToneJs.ts @@ -1,4 +1,7 @@ -import { Instrument, InstrumentOptions } from 'tone/build/esm/instrument/Instrument' +import { + Instrument, + InstrumentOptions, +} from 'tone/build/esm/instrument/Instrument' import InstrumentBassElectricMp3 from 'tonejs-instrument-bass-electric-mp3' import InstrumentBassoonMp3 from 'tonejs-instrument-bassoon-mp3' import InstrumentClarinetMp3 from 'tonejs-instrument-clarinet-mp3' @@ -22,11 +25,12 @@ import InstrumentXylophoneMp3 from 'tonejs-instrument-xylophone-mp3' import { InstrumentEnum } from '../../domain/enum/InstrumentEnum' import { SoundService } from '../../domain/services/SoundService' - export class ToneJS implements SoundService { static instruments: Map> = new Map() - static async getInstrument(instrumentId: string): Promise> { + static async getInstrument( + instrumentId: string, + ): Promise> { let instrument = ToneJS.instruments.get(instrumentId) if (instrument) { return instrument @@ -34,38 +38,58 @@ export class ToneJS implements SoundService { const InstrumentClass = ToneJS.getInstrumentClass(instrumentId) - return new Promise>((resolve) => { + return new Promise>(resolve => { instrument = new InstrumentClass({ onload: () => { ToneJS.instruments.set(instrumentId, instrument!) resolve(ToneJS.getInstrument(instrumentId)) - } + }, }).toDestination() }) } static getInstrumentClass(instrumentId: string) { switch (instrumentId) { - case InstrumentEnum.BassElectric: return InstrumentBassElectricMp3 - case InstrumentEnum.Bassoon: return InstrumentBassoonMp3 - case InstrumentEnum.Clarinet: return InstrumentClarinetMp3 - case InstrumentEnum.Contrabass: return InstrumentContrabassMp3 - case InstrumentEnum.Flute: return InstrumentFluteMp3 - case InstrumentEnum.FrenchHorn: return InstrumentFrenchHornMp3 - case InstrumentEnum.GuitarAcoustic: return InstrumentGuitarAcousticMp3 - case InstrumentEnum.GuitarElectric: return InstrumentGuitarElectricMp3 - case InstrumentEnum.GuitarNylon: return InstrumentGuitarNylonMp3 - case InstrumentEnum.Harmonium: return InstrumentHarmoniumMp3 - case InstrumentEnum.Harp: return InstrumentHarpMp3 - case InstrumentEnum.Organ: return InstrumentOrganMp3 - case InstrumentEnum.Piano: return InstrumentPianoMp3 - case InstrumentEnum.Saxophone: return InstrumentSaxophoneMp3 - case InstrumentEnum.Trombone: return InstrumentTromboneMp3 - case InstrumentEnum.Trumpet: return InstrumentTrumpetMp3 - case InstrumentEnum.Tuba: return InstrumentTubaMp3 - case InstrumentEnum.Violin: return InstrumentViolinMp3 - case InstrumentEnum.Xylophone: return InstrumentXylophoneMp3 - default: throw new Error(`Instrument ${instrumentId} not found`) + case InstrumentEnum.BassElectric: + return InstrumentBassElectricMp3 + case InstrumentEnum.Bassoon: + return InstrumentBassoonMp3 + case InstrumentEnum.Clarinet: + return InstrumentClarinetMp3 + case InstrumentEnum.Contrabass: + return InstrumentContrabassMp3 + case InstrumentEnum.Flute: + return InstrumentFluteMp3 + case InstrumentEnum.FrenchHorn: + return InstrumentFrenchHornMp3 + case InstrumentEnum.GuitarAcoustic: + return InstrumentGuitarAcousticMp3 + case InstrumentEnum.GuitarElectric: + return InstrumentGuitarElectricMp3 + case InstrumentEnum.GuitarNylon: + return InstrumentGuitarNylonMp3 + case InstrumentEnum.Harmonium: + return InstrumentHarmoniumMp3 + case InstrumentEnum.Harp: + return InstrumentHarpMp3 + case InstrumentEnum.Organ: + return InstrumentOrganMp3 + case InstrumentEnum.Piano: + return InstrumentPianoMp3 + case InstrumentEnum.Saxophone: + return InstrumentSaxophoneMp3 + case InstrumentEnum.Trombone: + return InstrumentTromboneMp3 + case InstrumentEnum.Trumpet: + return InstrumentTrumpetMp3 + case InstrumentEnum.Tuba: + return InstrumentTubaMp3 + case InstrumentEnum.Violin: + return InstrumentViolinMp3 + case InstrumentEnum.Xylophone: + return InstrumentXylophoneMp3 + default: + throw new Error(`Instrument ${instrumentId} not found`) } } @@ -73,11 +97,21 @@ export class ToneJS implements SoundService { await ToneJS.getInstrument(instrument) } - async playSound(pitch: string, octave: number, instrument: string): Promise { - (await ToneJS.getInstrument(instrument)).triggerAttack(`${pitch}${octave}`) + async playSound( + pitch: string, + octave: number, + instrument: string, + ): Promise { + const instrumentInstance = await ToneJS.getInstrument(instrument) + instrumentInstance.triggerAttack(`${pitch}${octave}`) } - async stopSound(pitch: string, octave: number, instrument: string): Promise { - (await ToneJS.getInstrument(instrument)).triggerRelease(`${pitch}${octave}`) + async stopSound( + pitch: string, + octave: number, + instrument: string, + ): Promise { + const instrumentInstance = await ToneJS.getInstrument(instrument) + instrumentInstance.triggerRelease(`${pitch}${octave}`) } -} \ No newline at end of file +} diff --git a/apps/web/src/presentation/common/base/Presenter.ts b/apps/web/src/presentation/common/base/Presenter.ts index 815dee3..861e44d 100644 --- a/apps/web/src/presentation/common/base/Presenter.ts +++ b/apps/web/src/presentation/common/base/Presenter.ts @@ -1,3 +1,3 @@ import { StateController } from './StateController' -export abstract class Presenter extends StateController {} \ No newline at end of file +export abstract class Presenter extends StateController {} diff --git a/apps/web/src/presentation/common/base/StateController.spec.ts b/apps/web/src/presentation/common/base/StateController.spec.ts index 041e70b..a5d41d5 100644 --- a/apps/web/src/presentation/common/base/StateController.spec.ts +++ b/apps/web/src/presentation/common/base/StateController.spec.ts @@ -55,7 +55,6 @@ describe('src/presentation/common/base/StateController', () => { }) describe('unsubscribe', () => { - it('should unsubscribe subscribers correctly', () => { const subscriber1 = vi.fn() const subscriber2 = vi.fn() diff --git a/apps/web/src/presentation/common/base/StateController.ts b/apps/web/src/presentation/common/base/StateController.ts index 8e0e801..77fbef9 100644 --- a/apps/web/src/presentation/common/base/StateController.ts +++ b/apps/web/src/presentation/common/base/StateController.ts @@ -15,7 +15,7 @@ export abstract class StateController { protected changeState(state: Partial): void { this.setState({ ...this.internalState, - ...state + ...state, }) } @@ -34,7 +34,10 @@ export abstract class StateController { listener(this.state) } - public subscribe(listener: Subscription, sendCurrentStatus = false): void { + public subscribe( + listener: Subscription, + sendCurrentStatus = false, + ): void { this.listeners.push(listener) if (sendCurrentStatus) { this.notifyStateChangeToListener(listener) diff --git a/apps/web/src/presentation/common/context/domainServices.ts b/apps/web/src/presentation/common/context/domainServices.ts index 07e18f9..ff5f28c 100644 --- a/apps/web/src/presentation/common/context/domainServices.ts +++ b/apps/web/src/presentation/common/context/domainServices.ts @@ -6,5 +6,5 @@ export interface DomainServices { } export const domainServices: DomainServices = { - soundService: new ToneJS() + soundService: new ToneJS(), } diff --git a/apps/web/src/presentation/common/context/presenters.ts b/apps/web/src/presentation/common/context/presenters.ts index 5432c2b..4bc1c0b 100644 --- a/apps/web/src/presentation/common/context/presenters.ts +++ b/apps/web/src/presentation/common/context/presenters.ts @@ -16,45 +16,38 @@ export interface Presenters { const cubesPresenter = new CubesPresenter( services.getCubes, - services.recalculateCubePosition + services.recalculateCubePosition, ) const appPresenter = new AppPresenter( services.createTransaction, services.switchNetwork, services.getSelectedNetwork, - services.listNetworks + services.listNetworks, ) const keyboardPresenter = new KeyboardPresenter( services.createKeyboard, services.getKeyboard, - new ObservableSet( - services.pressKey, - services.releaseKey - ) + new ObservableSet(services.pressKey, services.releaseKey), ) const transactionsPresenter = new TransactionsPresenter( services.listTransactions, - new ObservableSet( - services.createTransaction, - services.switchNetwork - ) + new ObservableSet(services.createTransaction, services.switchNetwork), ) const optionsPresenter = new OptionsPresenter( services.getOptions, services.setMuted, services.setInstrument, - services.loadInstrument + services.loadInstrument, ) - export const presenters: Presenters = { cubesPresenter, appPresenter, keyboardPresenter, transactionsPresenter, - optionsPresenter + optionsPresenter, } diff --git a/apps/web/src/presentation/common/context/repositories.ts b/apps/web/src/presentation/common/context/repositories.ts index b5b00d3..2b07703 100644 --- a/apps/web/src/presentation/common/context/repositories.ts +++ b/apps/web/src/presentation/common/context/repositories.ts @@ -11,9 +11,9 @@ import { StaticNetworksRepository } from '../../../infrastructure/repositories/S export interface Repositories { cubeRepository: CubeRepository - transactionRepository: TransactionRepository, - keyboardRepository: KeyboardRepository, - optionsRepository: OptionsRepository, + transactionRepository: TransactionRepository + keyboardRepository: KeyboardRepository + optionsRepository: OptionsRepository networkRepository: NetworkRepository } @@ -22,5 +22,5 @@ export const repositories: Repositories = { transactionRepository: new LimitedInMemoryTransactionRepository(), keyboardRepository: new InMemoryKeyboardRepository(), optionsRepository: new LocalStorageOptionsRepository(), - networkRepository: new StaticNetworksRepository() + networkRepository: new StaticNetworksRepository(), } diff --git a/apps/web/src/presentation/common/context/services.ts b/apps/web/src/presentation/common/context/services.ts index 13c5f6f..b7be09a 100644 --- a/apps/web/src/presentation/common/context/services.ts +++ b/apps/web/src/presentation/common/context/services.ts @@ -33,38 +33,55 @@ export interface Services { setMuted: SetMutedService setInstrument: SetInstrumentService loadInstrument: LoadInstrumentService - switchNetwork: SwitchNetworkService, + switchNetwork: SwitchNetworkService getSelectedNetwork: GetSelectedNetworkService listNetworks: ListNetworksService } const getCubes = new GetCubesService(repositories.cubeRepository) const createCube = new CreateCubeService(repositories.cubeRepository) -const recalculateCubePosition = new RecalculateCubePositionsService(repositories.cubeRepository) -const createKeyboard = new CreateKeyboardService(repositories.keyboardRepository) -const playSound = new PlaySoundService(domainServices.soundService, repositories.optionsRepository) +const recalculateCubePosition = new RecalculateCubePositionsService( + repositories.cubeRepository, +) +const createKeyboard = new CreateKeyboardService( + repositories.keyboardRepository, +) +const playSound = new PlaySoundService( + domainServices.soundService, + repositories.optionsRepository, +) const stopSound = new StopSoundService(domainServices.soundService) const pressKey = new PressKeyService(repositories.keyboardRepository, playSound) -const releaseKey = new ReleaseKeyService(repositories.keyboardRepository, stopSound) +const releaseKey = new ReleaseKeyService( + repositories.keyboardRepository, + stopSound, +) const createTransaction = new CreateTransactionService( repositories.keyboardRepository, repositories.transactionRepository, createCube, pressKey, releaseKey, - repositories.networkRepository + repositories.networkRepository, +) +const listTransactions = new ListTransactionsService( + repositories.transactionRepository, ) -const listTransactions = new ListTransactionsService(repositories.transactionRepository) const getKeyboard = new GetKeyboardService(repositories.keyboardRepository) const getOptions = new GetOptionsService(repositories.optionsRepository) const setMuted = new SetMutedService(repositories.optionsRepository) const setInstrument = new SetInstrumentService(repositories.optionsRepository) const loadInstrument = new LoadInstrumentService(domainServices.soundService) -const switchNetwork = new SwitchNetworkService(repositories.transactionRepository, repositories.networkRepository, repositories.cubeRepository) -const getSelectedNetwork = new GetSelectedNetworkService(repositories.networkRepository) +const switchNetwork = new SwitchNetworkService( + repositories.transactionRepository, + repositories.networkRepository, + repositories.cubeRepository, +) +const getSelectedNetwork = new GetSelectedNetworkService( + repositories.networkRepository, +) const listNetworks = new ListNetworksService(repositories.networkRepository) - export const services: Services = { createTransaction, getCubes, @@ -81,5 +98,5 @@ export const services: Services = { loadInstrument, switchNetwork, getSelectedNetwork, - listNetworks + listNetworks, } diff --git a/apps/web/src/presentation/common/presenter/app/AppPresenter.ts b/apps/web/src/presentation/common/presenter/app/AppPresenter.ts index b0a30cc..74ab669 100644 --- a/apps/web/src/presentation/common/presenter/app/AppPresenter.ts +++ b/apps/web/src/presentation/common/presenter/app/AppPresenter.ts @@ -42,8 +42,7 @@ export class AppPresenter extends Presenter { private async loadNetworkState() { const { networks } = await this.listNetworks.execute() - if (!networks) - return + if (!networks) return const networkNames = networks.map(network => network.name) const selectedNetwork = await this.getSelectedNetworkService.execute() @@ -51,7 +50,7 @@ export class AppPresenter extends Presenter { this.changeState({ networkNames, selectedNetworkName: selectedNetwork.networkName, - selectedNetworkWsUrl: selectedNetwork.networkWsUrl + selectedNetworkWsUrl: selectedNetwork.networkWsUrl, }) } @@ -69,21 +68,25 @@ export class AppPresenter extends Presenter { async selectNetwork(networkName: string) { const response = await this.switchNetworkService.execute({ - networkName + networkName, }) this.changeState({ selectedNetworkName: response.networkName, - selectedNetworkWsUrl: response.networkWsUrl + selectedNetworkWsUrl: response.networkWsUrl, }) } - async createTransaction(txType: string, address: string, network: string): Promise { + async createTransaction( + txType: string, + address: string, + network: string, + ): Promise { return this.createTransactionService.execute({ txType, address, network, - timestamp: Date.now() + timestamp: Date.now(), }) } } diff --git a/apps/web/src/presentation/common/presenter/app/AppPresenterState.ts b/apps/web/src/presentation/common/presenter/app/AppPresenterState.ts index 1b82589..375b802 100644 --- a/apps/web/src/presentation/common/presenter/app/AppPresenterState.ts +++ b/apps/web/src/presentation/common/presenter/app/AppPresenterState.ts @@ -1,8 +1,7 @@ - export type AppPresenterState = { networkNames: string[] selectedNetworkName: string | null navMenuVisible: boolean selectedNetworkWsUrl: string | null enableMainnet: boolean -} \ No newline at end of file +} diff --git a/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.spec.ts b/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.spec.ts index 98250d2..6da78d4 100644 --- a/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.spec.ts +++ b/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.spec.ts @@ -17,28 +17,32 @@ describe('src/presentation/common/CubesPresenter', () => { let createKeyboardService: CreateKeyboardService let createCubeService: CreateCubeService - const options = { timeToFirstBlock: 100, maxCubeCreationInterval: 10, tickInterval: 5, - cubeStep: 0.001 + cubeStep: 0.001, } - beforeEach(async () => { keyboardRepository = new InMemoryKeyboardRepository() createKeyboardService = new CreateKeyboardService(keyboardRepository) await createKeyboardService.execute({ numberOfKeys: 88, initialPitchClass: 'A', - initialOctave: 1 + initialOctave: 1, }) cubeRepository = new InMemoryCubeRepository() getCubesService = new GetCubesService(cubeRepository) - recalculateCubePositionsService = new RecalculateCubePositionsService(cubeRepository) + recalculateCubePositionsService = new RecalculateCubePositionsService( + cubeRepository, + ) createCubeService = new CreateCubeService(cubeRepository) - presenter = new CubesPresenter(getCubesService, recalculateCubePositionsService, options) + presenter = new CubesPresenter( + getCubesService, + recalculateCubePositionsService, + options, + ) await createCubeService.execute({ color: 'blue', x: 0.5 }) }) @@ -55,11 +59,15 @@ describe('src/presentation/common/CubesPresenter', () => { it('should stop creating cubes when stopped', async () => { presenter.run() - await new Promise(resolve => setTimeout(resolve, options.maxCubeCreationInterval)) + await new Promise(resolve => + setTimeout(resolve, options.maxCubeCreationInterval), + ) presenter.stop() const cubes = await cubeRepository.list() const cubesCount = cubes.length - await new Promise(resolve => setTimeout(resolve, options.maxCubeCreationInterval)) + await new Promise(resolve => + setTimeout(resolve, options.maxCubeCreationInterval), + ) expect(cubes.length).toBe(cubesCount) }) diff --git a/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.ts b/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.ts index 7b8f4b3..62bb8ca 100644 --- a/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.ts +++ b/apps/web/src/presentation/common/presenter/cubes/CubesPresenter.ts @@ -16,31 +16,28 @@ interface CubesPresenterOptions { const defaultOptions: CubesPresenterOptions = { maxCubeCreationInterval: 300, tickInterval: 20, - maxCubeAge: 17_000 + maxCubeAge: 17_000, } export class CubesPresenter extends Presenter { - private readonly getCubesService: GetCubesService private readonly recalculateCubePositions: RecalculateCubePositionsService - private options: CubesPresenterOptions private tickLoop: NodeJS.Timeout | undefined - constructor( getCubesService: GetCubesService, recalculateCubePositions: RecalculateCubePositionsService, - options?: Partial + options?: Partial, ) { super(initalState) this.getCubesService = getCubesService this.recalculateCubePositions = recalculateCubePositions this.options = { ...defaultOptions, - ...options + ...options, } this.run() @@ -77,14 +74,14 @@ export class CubesPresenter extends Presenter { private async moveAllCubesUp() { await this.recalculateCubePositions.execute({ - maxAge: this.options.maxCubeAge + maxAge: this.options.maxCubeAge, }) } private async syncState() { const getCubesResponse = await this.getCubesService.execute() this.changeState({ - cubes: getCubesResponse.cubes + cubes: getCubesResponse.cubes, }) } } diff --git a/apps/web/src/presentation/common/presenter/cubes/CubesPresenterState.ts b/apps/web/src/presentation/common/presenter/cubes/CubesPresenterState.ts index ce796cc..f40c4f3 100644 --- a/apps/web/src/presentation/common/presenter/cubes/CubesPresenterState.ts +++ b/apps/web/src/presentation/common/presenter/cubes/CubesPresenterState.ts @@ -1,4 +1,3 @@ - export type CubesPresenterState = { cubes: { id: string diff --git a/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.spec.ts b/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.spec.ts index 012acd7..480474b 100644 --- a/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.spec.ts +++ b/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.spec.ts @@ -9,10 +9,13 @@ const createKeyboardService = new CreateKeyboardService(keyboardRepository) await createKeyboardService.execute({ numberOfKeys: 88, initialPitchClass: 'A', - initialOctave: 1 + initialOctave: 1, }) const getKeyboardService = new GetKeyboardService(keyboardRepository) -const keyboardPresenter = new KeyboardPresenter(createKeyboardService, getKeyboardService) +const keyboardPresenter = new KeyboardPresenter( + createKeyboardService, + getKeyboardService, +) describe('src/presentation/common/presenter/keyboard/KeyboardPresenter', () => { it('should be defined', () => { @@ -22,5 +25,4 @@ describe('src/presentation/common/presenter/keyboard/KeyboardPresenter', () => { it('should have 88 keys', () => { expect(keyboardPresenter.state.keys.length).toBe(88) }) - -}) \ No newline at end of file +}) diff --git a/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.ts b/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.ts index 9da55da..cf5af31 100644 --- a/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.ts +++ b/apps/web/src/presentation/common/presenter/keyboard/KeyboardPresenter.ts @@ -5,7 +5,7 @@ import { Presenter } from '../../base/Presenter' import { KeyboardPresenterState } from './KeyboardPresenterState' const initalState: KeyboardPresenterState = { - keys: [] + keys: [], } interface KeyboardPresenterOptions { @@ -17,11 +17,10 @@ interface KeyboardPresenterOptions { const defaultOptions: KeyboardPresenterOptions = { numberOfKeys: 88, initialPitchClass: 'A', - initialOctave: 0 + initialOctave: 0, } export class KeyboardPresenter extends Presenter { - private options: KeyboardPresenterOptions private createKeyboardService: CreateKeyboardService private getKeyboardService: GetKeyboardService @@ -30,7 +29,7 @@ export class KeyboardPresenter extends Presenter { createKeyboardService: CreateKeyboardService, getKeyboardService: GetKeyboardService, keyboardChangesObserver?: Observable, - options?: Partial + options?: Partial, ) { super(initalState) @@ -39,13 +38,13 @@ export class KeyboardPresenter extends Presenter { this.options = { ...defaultOptions, - ...options + ...options, } this.createKeyboard() this.refresh() - if(keyboardChangesObserver) { + if (keyboardChangesObserver) { keyboardChangesObserver.listen(this.refresh.bind(this)) } } @@ -61,12 +60,12 @@ export class KeyboardPresenter extends Presenter { keys: keyboard.keys.map(key => ({ pitch: { class: key.pitch.class, - octave: key.pitch.octave + octave: key.pitch.octave, }, keyShape: key.keyShape, x: key.x, - color: key.pressed ? key.color : undefined - })) + color: key.pressed ? key.color : undefined, + })), }) } @@ -74,8 +73,7 @@ export class KeyboardPresenter extends Presenter { await this.createKeyboardService.execute({ numberOfKeys: this.options.numberOfKeys, initialPitchClass: this.options.initialPitchClass, - initialOctave: this.options.initialOctave + initialOctave: this.options.initialOctave, }) } - } diff --git a/apps/web/src/presentation/common/presenter/options/OptionsPresenter.ts b/apps/web/src/presentation/common/presenter/options/OptionsPresenter.ts index b91f01d..0530d2a 100644 --- a/apps/web/src/presentation/common/presenter/options/OptionsPresenter.ts +++ b/apps/web/src/presentation/common/presenter/options/OptionsPresenter.ts @@ -2,27 +2,29 @@ import { GetOptionsService } from '../../../../application/services/GetOptions/G import { LoadInstrumentService } from '../../../../application/services/LoadInstrument/LoadInstrumentService' import { SetInstrumentService } from '../../../../application/services/SetInstrument/SetInstrumentService' import { SetMutedService } from '../../../../application/services/SetMuted/SetMutedService' -import { InstrumentEnum, instrumentLabels } from '../../../../domain/enum/InstrumentEnum' +import { + InstrumentEnum, + instrumentLabels, +} from '../../../../domain/enum/InstrumentEnum' import { Presenter } from '../../base/Presenter' import { OptionsPresenterState } from './OptionsPresenterState' -const instruments: { label: string, value: string }[] = Object.values(InstrumentEnum).map( - (instrument) => ({ - label: instrumentLabels.get(instrument) ?? instrument, - value: instrument - }) -) +const instruments: { label: string; value: string }[] = Object.values( + InstrumentEnum, +).map(instrument => ({ + label: instrumentLabels.get(instrument) ?? instrument, + value: instrument, +})) const initalState: OptionsPresenterState = { muted: true, selectedInstrument: '', instruments, displayLoadingMessage: false, - displayInstrumentPicker: false + displayInstrumentPicker: false, } export class OptionsPresenter extends Presenter { - private getOptionsService: GetOptionsService private setMutedService: SetMutedService private setInstrumentService: SetInstrumentService @@ -32,7 +34,7 @@ export class OptionsPresenter extends Presenter { getOptionsService: GetOptionsService, setMutedService: SetMutedService, setInstrumentService: SetInstrumentService, - loadInstrumentService: LoadInstrumentService + loadInstrumentService: LoadInstrumentService, ) { super(initalState) @@ -51,11 +53,10 @@ export class OptionsPresenter extends Presenter { selectedInstrument: options.instrument, instruments, displayLoadingMessage: false, - displayInstrumentPicker: true + displayInstrumentPicker: true, }) } - async setMuted(muted: boolean) { if (!muted) { await this.loadInstrument(this.state.selectedInstrument) @@ -71,8 +72,14 @@ export class OptionsPresenter extends Presenter { } private async loadInstrument(instrument: string) { - this.changeState({ displayLoadingMessage: true, displayInstrumentPicker: false }) + this.changeState({ + displayLoadingMessage: true, + displayInstrumentPicker: false, + }) await this.loadInstrumentService.execute({ instrument }) - this.changeState({ displayLoadingMessage: false, displayInstrumentPicker: true }) + this.changeState({ + displayLoadingMessage: false, + displayInstrumentPicker: true, + }) } } diff --git a/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenter.ts b/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenter.ts index 278d31f..7368164 100644 --- a/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenter.ts +++ b/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenter.ts @@ -11,36 +11,38 @@ const titleMap: Map = new Map([ [TxTypesEnum.Block, 'New Block'], [TxTypesEnum.Eth, 'ETH'], [TxTypesEnum.Pop, 'PoP'], - [TxTypesEnum.Btc, 'BTC'] + [TxTypesEnum.Btc, 'BTC'], ]) const rgbMap: Map = new Map([ [TxTypesEnum.Block, '#10FF2A'], [TxTypesEnum.Eth, '#00D3FF'], [TxTypesEnum.Pop, '#DC53FF'], - [TxTypesEnum.Btc, '#FFB200'] + [TxTypesEnum.Btc, '#FFB200'], ]) const messageMap: Map = new Map([ [TxTypesEnum.Block, 'created by'], [TxTypesEnum.Eth, 'transaction by'], [TxTypesEnum.Pop, 'transaction by'], - [TxTypesEnum.Btc, 'transaction by'] + [TxTypesEnum.Btc, 'transaction by'], ]) export class TransactionsPresenter extends Presenter { - private listTransactions: ListTransactionsService - constructor(listTransactions: ListTransactionsService, transactionsChangeObserver?: Observable) { + constructor( + listTransactions: ListTransactionsService, + transactionsChangeObserver?: Observable, + ) { super(initalState) this.listTransactions = listTransactions - if(transactionsChangeObserver) { + if (transactionsChangeObserver) { transactionsChangeObserver.listen(this.refresh.bind(this)) } } - async refresh() { + async refresh() { const response = await this.listTransactions.execute() this.changeState({ transactions: response.transactions.map(transaction => ({ @@ -49,8 +51,8 @@ export class TransactionsPresenter extends Presenter message: messageMap.get(transaction.txType as TxTypesEnum) ?? ' ', id: transaction.address, at: this.formatDate(transaction.timestamp), - url: transaction.url - })) + url: transaction.url, + })), }) } @@ -62,7 +64,7 @@ export class TransactionsPresenter extends Presenter hour: 'numeric', minute: 'numeric', second: 'numeric', - hour12: true + hour12: true, }) return formattedDate.replace(/,/g, '') diff --git a/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenterState.ts b/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenterState.ts index b07f631..689604f 100644 --- a/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenterState.ts +++ b/apps/web/src/presentation/common/presenter/transactions/TransactionsPresenterState.ts @@ -1,4 +1,3 @@ - export type TransactionsPresenterState = { transactions: { type: string diff --git a/apps/web/src/presentation/react/App.tsx b/apps/web/src/presentation/react/App.tsx index f0eb8ea..b47cc00 100644 --- a/apps/web/src/presentation/react/App.tsx +++ b/apps/web/src/presentation/react/App.tsx @@ -10,26 +10,26 @@ import { usePresenter } from './hooks/usePresenter' function App() { const { appPresenter } = useContext(presenters) - const { - navMenuVisible, - } = usePresenter(appPresenter) + const { navMenuVisible } = usePresenter( + appPresenter, + ) return ( <> -
appPresenter.navButtonClicked()} - networks={appPresenter.state.networkNames} - selectedNetwork={appPresenter.state.selectedNetworkName} - selectNetwork={(networkName) => appPresenter.selectNetwork(networkName)} +
appPresenter.navButtonClicked()} + networks={appPresenter.state.networkNames} + selectedNetwork={appPresenter.state.selectedNetworkName} + selectNetwork={networkName => appPresenter.selectNetwork(networkName)} enableMainnet={appPresenter.state.enableMainnet} /> - appPresenter.closeButtonClicked()} - className={`${navMenuVisible ? '' : 'hidden'} md:hidden`} - networks={appPresenter.state.networkNames} - selectedNetwork={appPresenter.state.selectedNetworkName} - selectNetwork={(networkName) => appPresenter.selectNetwork(networkName)} + appPresenter.closeButtonClicked()} + className={`${navMenuVisible ? '' : 'hidden'} md:hidden`} + networks={appPresenter.state.networkNames} + selectedNetwork={appPresenter.state.selectedNetworkName} + selectNetwork={networkName => appPresenter.selectNetwork(networkName)} enableMainnet={appPresenter.state.enableMainnet} /> diff --git a/apps/web/src/presentation/react/components/Card.tsx b/apps/web/src/presentation/react/components/Card.tsx index 0cdded2..e1d0439 100644 --- a/apps/web/src/presentation/react/components/Card.tsx +++ b/apps/web/src/presentation/react/components/Card.tsx @@ -3,7 +3,9 @@ export const Card = function (props: { children?: React.ReactNode }) { return ( -
+
{props.children}
) diff --git a/apps/web/src/presentation/react/components/Checkbox.tsx b/apps/web/src/presentation/react/components/Checkbox.tsx index 652fb9f..3e02815 100644 --- a/apps/web/src/presentation/react/components/Checkbox.tsx +++ b/apps/web/src/presentation/react/components/Checkbox.tsx @@ -5,10 +5,15 @@ export const Checkbox = function (props: { }) { return ( <> - {}}/> + {}} + />
- + `} + >
+ ) } diff --git a/apps/web/src/presentation/react/components/Cube.tsx b/apps/web/src/presentation/react/components/Cube.tsx index d0ebf91..12a16d2 100644 --- a/apps/web/src/presentation/react/components/Cube.tsx +++ b/apps/web/src/presentation/react/components/Cube.tsx @@ -6,7 +6,8 @@ export const Cube = function (props: { mirrored?: boolean }) { return ( - (cubesPresenter) + const { cubes } = usePresenter( + cubesPresenter, + ) return (
- {cubes - .map((cube) => { - return ( - - ) - })} + {cubes.map(cube => { + return ( + + ) + })}
) } diff --git a/apps/web/src/presentation/react/components/Header.tsx b/apps/web/src/presentation/react/components/Header.tsx index 7c7a001..b835936 100644 --- a/apps/web/src/presentation/react/components/Header.tsx +++ b/apps/web/src/presentation/react/components/Header.tsx @@ -1,16 +1,16 @@ -import { JoinCommunityButton } from './JoinCommunityButton'; -import { Logo } from './Logo'; -import { NavButton } from './NavButton'; -import { NavItems } from './NavItems'; -import { NetworkSwitch } from './NetworkSwitch'; +import { JoinCommunityButton } from './JoinCommunityButton' +import { Logo } from './Logo' +import { NavButton } from './NavButton' +import { NavItems } from './NavItems' +import { NetworkSwitch } from './NetworkSwitch' export const Header = (props: { - className?: string; - onNavButtonClick?: () => void; - networks: string[]; - selectedNetwork: string | null; - selectNetwork: (networkName: string) => void; - enableMainnet: boolean; + className?: string + onNavButtonClick?: () => void + networks: string[] + selectedNetwork: string | null + selectNetwork: (networkName: string) => void + enableMainnet: boolean }) => { return (
- ); -}; \ No newline at end of file + ) +} diff --git a/apps/web/src/presentation/react/components/JoinCommunityButton.tsx b/apps/web/src/presentation/react/components/JoinCommunityButton.tsx index f93ab02..1d61ed8 100644 --- a/apps/web/src/presentation/react/components/JoinCommunityButton.tsx +++ b/apps/web/src/presentation/react/components/JoinCommunityButton.tsx @@ -1,6 +1,4 @@ -export const JoinCommunityButton = (props: { - className?: string -}) => { +export const JoinCommunityButton = (props: { className?: string }) => { return ( Join Community + target="_blank" + > + Join Community + ) } diff --git a/apps/web/src/presentation/react/components/Key.tsx b/apps/web/src/presentation/react/components/Key.tsx index b1d0329..c85ef5f 100644 --- a/apps/web/src/presentation/react/components/Key.tsx +++ b/apps/web/src/presentation/react/components/Key.tsx @@ -4,7 +4,7 @@ export interface KeyProps { index: number color?: string } -export const Key = function ({ keyShape, x, index, color}: KeyProps) { +export const Key = function ({ keyShape, x, index, color }: KeyProps) { const isBlack = keyShape === 'black' const zIndex = isBlack ? 1 : 0 const left = `${x * 100}%` @@ -13,11 +13,12 @@ export const Key = function ({ keyShape, x, index, color}: KeyProps) { return ( (keyboardPresenter) + const { keys } = usePresenter( + keyboardPresenter, + ) return (
-
- { - keys.map( - ({ keyShape, x, color, pitch }, index) => - - ) - } +
+ {keys.map(({ keyShape, x, color, pitch }, index) => ( + + ))}
) diff --git a/apps/web/src/presentation/react/components/Logo.tsx b/apps/web/src/presentation/react/components/Logo.tsx index 0594daf..3f2ee41 100644 --- a/apps/web/src/presentation/react/components/Logo.tsx +++ b/apps/web/src/presentation/react/components/Logo.tsx @@ -3,13 +3,14 @@ import logo from '/image/crypto-chords.svg' export const Logo = function () { const url = import.meta.env.VITE_LOGO_URL return ( - + `} + > Crypto Chords ) diff --git a/apps/web/src/presentation/react/components/MainContent.tsx b/apps/web/src/presentation/react/components/MainContent.tsx index deb5669..8297db1 100644 --- a/apps/web/src/presentation/react/components/MainContent.tsx +++ b/apps/web/src/presentation/react/components/MainContent.tsx @@ -14,25 +14,33 @@ const CUBES_Y_MULTIPLIER = 2 // to hide the creation of the cubes avoiding a visual glitch const CUBES_BOTTOM_OFFSET = 0.15 -export function MainContent(props: { - className: string -}) { +export function MainContent(props: { className: string }) { return (
- - -
- - -
- + + +
+ + +
+ - +
- +
) diff --git a/apps/web/src/presentation/react/components/NavButton.tsx b/apps/web/src/presentation/react/components/NavButton.tsx index bc1c067..8f0d149 100644 --- a/apps/web/src/presentation/react/components/NavButton.tsx +++ b/apps/web/src/presentation/react/components/NavButton.tsx @@ -4,6 +4,10 @@ export const NavButton = function (props: { onClick?: () => void }) { return ( - + ) } diff --git a/apps/web/src/presentation/react/components/NavItem.tsx b/apps/web/src/presentation/react/components/NavItem.tsx index a744532..c935b90 100644 --- a/apps/web/src/presentation/react/components/NavItem.tsx +++ b/apps/web/src/presentation/react/components/NavItem.tsx @@ -3,7 +3,8 @@ export const NavItem = function (options: { children: React.ReactNode }) { return ( - diff --git a/apps/web/src/presentation/react/components/NavItems.tsx b/apps/web/src/presentation/react/components/NavItems.tsx index 724f9be..4041889 100644 --- a/apps/web/src/presentation/react/components/NavItems.tsx +++ b/apps/web/src/presentation/react/components/NavItems.tsx @@ -1,8 +1,6 @@ import { NavItem } from './NavItem' -export const NavItems = function (props: { - className?: string -}) { +export const NavItems = function (props: { className?: string }) { return (