diff --git a/GPU/.gitignore b/GPU/.gitignore new file mode 100644 index 0000000..f18b582 --- /dev/null +++ b/GPU/.gitignore @@ -0,0 +1,4 @@ + +settings/Mainnet.toml +settings/Testnet.toml +history.txt diff --git a/GPU/.vscode/settings.json b/GPU/.vscode/settings.json new file mode 100644 index 0000000..02e21eb --- /dev/null +++ b/GPU/.vscode/settings.json @@ -0,0 +1,4 @@ + +{ + "deno.enable": true, +} diff --git a/GPU/.vscode/tasks.json b/GPU/.vscode/tasks.json new file mode 100644 index 0000000..22af91c --- /dev/null +++ b/GPU/.vscode/tasks.json @@ -0,0 +1,18 @@ + +{ + "version": "2.0.0", + "tasks": [ + { + "label": "check contracts", + "group": "test", + "type": "shell", + "command": "clarinet check" + }, + { + "label": "test contracts", + "group": "test", + "type": "shell", + "command": "clarinet test" + } + ] +} diff --git a/GPU/Clarinet.toml b/GPU/Clarinet.toml new file mode 100644 index 0000000..80b8afd --- /dev/null +++ b/GPU/Clarinet.toml @@ -0,0 +1,11 @@ +[project] +name = "GPU" +authors = [] +description = "" +telemetry = true +requirements = [] +analysis = ["check_checker"] +costs_version = 2 +[contracts.farm] +path = "contracts/farm.clar" +depends_on = [] diff --git a/GPU/contracts/farm.clar b/GPU/contracts/farm.clar new file mode 100644 index 0000000..c1ea9fc --- /dev/null +++ b/GPU/contracts/farm.clar @@ -0,0 +1,258 @@ +;; Decentralized GPU Rendering Farm - Clarity Smart Contract +;; Features: Provider management, Job handling, Payments, Reputation system + +;; ============ DATA STRUCTURES ============ + +;; GPU Provider information +(define-map gpu-providers principal + { + owner: principal, + gpu-count: uint, + memory-gb: uint, + reputation: uint, + active: bool, + total-jobs: uint, + successful-jobs: uint + } +) + +;; Rendering Jobs +(define-map rendering-jobs uint + { + requester: principal, + provider: (optional principal), + status: (string-ascii 20), + complexity: uint, + reward: uint, + timestamp: uint, + completion-time: (optional uint) + } +) + +;; Job assignments +(define-map job-assignments uint + { + provider: principal, + start-time: uint, + estimated-duration: uint + } +) + +;; Payment ledger +(define-map payments principal uint) + +;; ============ CONSTANTS ============ + +(define-constant CONTRACT-OWNER tx-sender) +(define-constant MIN-STAKE u1000000) +(define-constant MIN-REPUTATION u1) +(define-constant JOB-TIMEOUT u86400) +(define-constant PLATFORM-FEE u2) + +;; ============ STATE VARIABLES ============ + +(define-data-var next-job-id uint u1) +(define-data-var total-volume uint u0) + +;; ============ ERRORS ============ + +(define-constant ERR-NOT-OWNER u101) +(define-constant ERR-INVALID-PROVIDER u102) +(define-constant ERR-INSUFFICIENT-STAKE u103) +(define-constant ERR-JOB-NOT-FOUND u104) +(define-constant ERR-INVALID-STATUS u105) +(define-constant ERR-LOW-REPUTATION u106) +(define-constant ERR-UNAUTHORIZED u107) + +;; ============ PROVIDER MANAGEMENT ============ + +(define-public (register-gpu-provider (gpu-count uint) (memory-gb uint)) + (begin + (asserts! (> gpu-count u0) (err ERR-INVALID-PROVIDER)) + (asserts! (> memory-gb u0) (err ERR-INVALID-PROVIDER)) + (map-set gpu-providers tx-sender + { + owner: tx-sender, + gpu-count: gpu-count, + memory-gb: memory-gb, + reputation: u100, + active: true, + total-jobs: u0, + successful-jobs: u0 + } + ) + (ok true) + ) +) + +(define-public (update-provider-resources (gpu-count uint) (memory-gb uint)) + (let ((provider (map-get? gpu-providers tx-sender))) + (asserts! (is-some provider) (err ERR-INVALID-PROVIDER)) + (map-set gpu-providers tx-sender + (merge (unwrap-panic provider) {gpu-count: gpu-count, memory-gb: memory-gb}) + ) + (ok true) + ) +) + +(define-public (toggle-provider-status) + (let ((provider (map-get? gpu-providers tx-sender))) + (asserts! (is-some provider) (err ERR-INVALID-PROVIDER)) + (let ((data (unwrap-panic provider))) + (map-set gpu-providers tx-sender + (merge data {active: (not (get active data))}) + ) + ) + (ok true) + ) +) + +;; ============ JOB MANAGEMENT ============ + +(define-public (create-job (complexity uint) (reward uint)) + (let ((job-id (var-get next-job-id))) + (asserts! (> reward u0) (err ERR-INVALID-STATUS)) + (asserts! (> complexity u0) (err ERR-INVALID-STATUS)) + (map-set rendering-jobs job-id + { + requester: tx-sender, + provider: none, + status: "pending", + complexity: complexity, + reward: reward, + timestamp: block-height, + completion-time: none + } + ) + (var-set next-job-id (+ job-id u1)) + (ok job-id) + ) +) + +(define-public (assign-job (job-id uint) (provider principal) (duration uint)) + (let ((job (map-get? rendering-jobs job-id)) + (prov (map-get? gpu-providers provider))) + (asserts! (is-some job) (err ERR-JOB-NOT-FOUND)) + (asserts! (is-some prov) (err ERR-INVALID-PROVIDER)) + (let ((job-data (unwrap-panic job)) + (prov-data (unwrap-panic prov))) + ;; replaced equal? with is-eq for Clarity built-in function + (asserts! (is-eq (get status job-data) "pending") (err ERR-INVALID-STATUS)) + (asserts! (get active prov-data) (err ERR-INVALID-PROVIDER)) + (asserts! (>= (get reputation prov-data) MIN-REPUTATION) (err ERR-LOW-REPUTATION)) + (map-set rendering-jobs job-id + (merge job-data {provider: (some provider), status: "in-progress"}) + ) + (map-set job-assignments job-id + {provider: provider, start-time: block-height, estimated-duration: duration} + ) + (ok true) + ) + ) +) + +(define-public (complete-job (job-id uint)) + (let ((job (map-get? rendering-jobs job-id)) + (assignment (map-get? job-assignments job-id))) + (asserts! (is-some job) (err ERR-JOB-NOT-FOUND)) + (asserts! (is-some assignment) (err ERR-INVALID-STATUS)) + (let ((job-data (unwrap-panic job)) + (assign-data (unwrap-panic assignment)) + (provider (get provider assign-data))) + ;; replaced equal? with is-eq for Clarity built-in function + (asserts! (is-eq (get status job-data) "in-progress") (err ERR-INVALID-STATUS)) + (asserts! (is-eq tx-sender provider) (err ERR-UNAUTHORIZED)) + (map-set rendering-jobs job-id + (merge job-data {status: "completed", completion-time: (some block-height)}) + ) + (let ((prov (map-get? gpu-providers provider))) + (if (is-some prov) + (let ((p (unwrap-panic prov))) + (map-set gpu-providers provider + (merge p { + total-jobs: (+ (get total-jobs p) u1), + successful-jobs: (+ (get successful-jobs p) u1), + reputation: (+ (get reputation p) u10) + }) + ) + ) + true + ) + ) + (ok true) + ) + ) +) + +;; ============ PAYMENT SYSTEM ============ + +(define-public (claim-reward (job-id uint)) + (let ((job (map-get? rendering-jobs job-id)) + (assignment (map-get? job-assignments job-id))) + (asserts! (is-some job) (err ERR-JOB-NOT-FOUND)) + (let ((job-data (unwrap-panic job))) + ;; replaced equal? with is-eq for Clarity built-in function + (asserts! (is-eq (get status job-data) "completed") (err ERR-INVALID-STATUS)) + (let ((provider (get provider job-data)) + (reward (get reward job-data))) + (asserts! (is-eq provider (some tx-sender)) (err ERR-UNAUTHORIZED)) + (let ((current-balance (default-to u0 (map-get? payments tx-sender)))) + (map-set payments tx-sender (+ current-balance reward)) + (var-set total-volume (+ (var-get total-volume) reward)) + (ok reward) + ) + ) + ) + ) +) + +(define-public (withdraw-earnings (amount uint)) + (let ((balance (default-to u0 (map-get? payments tx-sender)))) + (asserts! (>= balance amount) (err ERR-INSUFFICIENT-STAKE)) + (begin + (map-set payments tx-sender (- balance amount)) + (ok amount) + ) + ) +) + +;; ============ VIEW FUNCTIONS ============ + +(define-read-only (get-provider-info (provider principal)) + (map-get? gpu-providers provider) +) + +(define-read-only (get-job-info (job-id uint)) + (map-get? rendering-jobs job-id) +) + +(define-read-only (get-provider-balance (provider principal)) + (default-to u0 (map-get? payments provider)) +) + +(define-read-only (get-next-job-id) + (var-get next-job-id) +) + +(define-read-only (get-total-volume) + (var-get total-volume) +) + +(define-read-only (get-provider-stats (provider principal)) + (let ((prov (map-get? gpu-providers provider))) + (if (is-some prov) + (let ((data (unwrap-panic prov))) + (ok { + total-jobs: (get total-jobs data), + successful-jobs: (get successful-jobs data), + success-rate: (if (> (get total-jobs data) u0) + (/ (* (get successful-jobs data) u100) (get total-jobs data)) + u0 + ), + reputation: (get reputation data) + }) + ) + (err ERR-INVALID-PROVIDER) + ) + ) +) diff --git a/GPU/settings/Devnet.toml b/GPU/settings/Devnet.toml new file mode 100644 index 0000000..8a5ff75 --- /dev/null +++ b/GPU/settings/Devnet.toml @@ -0,0 +1,127 @@ +[network] +name = "devnet" +deployment_fee_rate = 10 + +[accounts.deployer] +mnemonic = "twice kind fence tip hidden tilt action fragile skin nothing glory cousin green tomorrow spring wrist shed math olympic multiply hip blue scout claw" +balance = 100_000_000_000_000 +# secret_key: 753b7cc01a1a2e86221266a154af739463fce51219d97e4f856cd7200c3bd2a601 +# stx_address: ST1PQHQKV0RJXZFY1DGX8MNSNYVE3VGZJSRTPGZGM +# btc_address: mqVnk6NPRdhntvfm4hh9vvjiRkFDUuSYsH + +[accounts.wallet_1] +mnemonic = "sell invite acquire kitten bamboo drastic jelly vivid peace spawn twice guilt pave pen trash pretty park cube fragile unaware remain midnight betray rebuild" +balance = 100_000_000_000_000 +# secret_key: 7287ba251d44a4d3fd9276c88ce34c5c52a038955511cccaf77e61068649c17801 +# stx_address: ST1SJ3DTE5DN7X54YDH5D64R3BCB6A2AG2ZQ8YPD5 +# btc_address: mr1iPkD9N3RJZZxXRk7xF9d36gffa6exNC + +[accounts.wallet_2] +mnemonic = "hold excess usual excess ring elephant install account glad dry fragile donkey gaze humble truck breeze nation gasp vacuum limb head keep delay hospital" +balance = 100_000_000_000_000 +# secret_key: 530d9f61984c888536871c6573073bdfc0058896dc1adfe9a6a10dfacadc209101 +# stx_address: ST2CY5V39NHDPWSXMW9QDT3HC3GD6Q6XX4CFRK9AG +# btc_address: muYdXKmX9bByAueDe6KFfHd5Ff1gdN9ErG + +[accounts.wallet_3] +mnemonic = "cycle puppy glare enroll cost improve round trend wrist mushroom scorpion tower claim oppose clever elephant dinosaur eight problem before frozen dune wagon high" +balance = 100_000_000_000_000 +# secret_key: d655b2523bcd65e34889725c73064feb17ceb796831c0e111ba1a552b0f31b3901 +# stx_address: ST2JHG361ZXG51QTKY2NQCVBPPRRE2KZB1HR05NNC +# btc_address: mvZtbibDAAA3WLpY7zXXFqRa3T4XSknBX7 + +[accounts.wallet_4] +mnemonic = "board list obtain sugar hour worth raven scout denial thunder horse logic fury scorpion fold genuine phrase wealth news aim below celery when cabin" +balance = 100_000_000_000_000 +# secret_key: f9d7206a47f14d2870c163ebab4bf3e70d18f5d14ce1031f3902fbbc894fe4c701 +# stx_address: ST2NEB84ASENDXKYGJPQW86YXQCEFEX2ZQPG87ND +# btc_address: mg1C76bNTutiCDV3t9nWhZs3Dc8LzUufj8 + +[accounts.wallet_5] +mnemonic = "hurry aunt blame peanut heavy update captain human rice crime juice adult scale device promote vast project quiz unit note reform update climb purchase" +balance = 100_000_000_000_000 +# secret_key: 3eccc5dac8056590432db6a35d52b9896876a3d5cbdea53b72400bc9c2099fe801 +# stx_address: ST2REHHS5J3CERCRBEPMGH7921Q6PYKAADT7JP2VB +# btc_address: mweN5WVqadScHdA81aATSdcVr4B6dNokqx + +[accounts.wallet_6] +mnemonic = "area desk dutch sign gold cricket dawn toward giggle vibrant indoor bench warfare wagon number tiny universe sand talk dilemma pottery bone trap buddy" +balance = 100_000_000_000_000 +# secret_key: 7036b29cb5e235e5fd9b09ae3e8eec4404e44906814d5d01cbca968a60ed4bfb01 +# stx_address: ST3AM1A56AK2C1XAFJ4115ZSV26EB49BVQ10MGCS0 +# btc_address: mzxXgV6e4BZSsz8zVHm3TmqbECt7mbuErt + +[accounts.wallet_7] +mnemonic = "prevent gallery kind limb income control noise together echo rival record wedding sense uncover school version force bleak nuclear include danger skirt enact arrow" +balance = 100_000_000_000_000 +# secret_key: b463f0df6c05d2f156393eee73f8016c5372caa0e9e29a901bb7171d90dc4f1401 +# stx_address: ST3PF13W7Z0RRM42A8VZRVFQ75SV1K26RXEP8YGKJ +# btc_address: n37mwmru2oaVosgfuvzBwgV2ysCQRrLko7 + +[accounts.wallet_8] +mnemonic = "female adjust gallery certain visit token during great side clown fitness like hurt clip knife warm bench start reunion globe detail dream depend fortune" +balance = 100_000_000_000_000 +# secret_key: 6a1a754ba863d7bab14adbbc3f8ebb090af9e871ace621d3e5ab634e1422885e01 +# stx_address: ST3NBRSFKX28FQ2ZJ1MAKX58HKHSDGNV5N7R21XCP +# btc_address: n2v875jbJ4RjBnTjgbfikDfnwsDV5iUByw + +[accounts.wallet_9] +mnemonic = "shadow private easily thought say logic fault paddle word top book during ignore notable orange flight clock image wealth health outside kitten belt reform" +balance = 100_000_000_000_000 +# secret_key: de433bdfa14ec43aa1098d5be594c8ffb20a31485ff9de2923b2689471c401b801 +# stx_address: STNHKEPYEPJ8ET55ZZ0M5A34J0R3N5FM2CMMMAZ6 +# btc_address: mjSrB3wS4xab3kYqFktwBzfTdPg367ZJ2d + +[devnet] +disable_bitcoin_explorer = true +# disable_stacks_explorer = true +# disable_stacks_api = true +# working_dir = "tmp/devnet" +# stacks_node_events_observers = ["host.docker.internal:8002"] +# miner_mnemonic = "twice kind fence tip hidden tilt action fragile skin nothing glory cousin green tomorrow spring wrist shed math olympic multiply hip blue scout claw" +# miner_derivation_path = "m/44'/5757'/0'/0/0" +# orchestrator_port = 20445 +# bitcoin_node_p2p_port = 18444 +# bitcoin_node_rpc_port = 18443 +# bitcoin_node_username = "devnet" +# bitcoin_node_password = "devnet" +# bitcoin_controller_port = 18442 +# bitcoin_controller_block_time = 30_000 +# stacks_node_rpc_port = 20443 +# stacks_node_p2p_port = 20444 +# stacks_api_port = 3999 +# stacks_api_events_port = 3700 +# bitcoin_explorer_port = 8001 +# stacks_explorer_port = 8000 +# postgres_port = 5432 +# postgres_username = "postgres" +# postgres_password = "postgres" +# postgres_database = "postgres" +# bitcoin_node_image_url = "quay.io/hirosystems/bitcoind:devnet" +# stacks_node_image_url = "localhost:5000/stacks-node:devnet" +# stacks_api_image_url = "blockstack/stacks-blockchain-api:latest" +# stacks_explorer_image_url = "blockstack/explorer:latest" +# bitcoin_explorer_image_url = "quay.io/hirosystems/bitcoin-explorer:devnet" +# postgres_image_url = "postgres:alpine" + +# Send some stacking orders +[[devnet.pox_stacking_orders]] +start_at_cycle = 3 +duration = 12 +wallet = "wallet_1" +slots = 2 +btc_address = "mr1iPkD9N3RJZZxXRk7xF9d36gffa6exNC" + +[[devnet.pox_stacking_orders]] +start_at_cycle = 3 +duration = 12 +wallet = "wallet_2" +slots = 1 +btc_address = "muYdXKmX9bByAueDe6KFfHd5Ff1gdN9ErG" + +[[devnet.pox_stacking_orders]] +start_at_cycle = 3 +duration = 12 +wallet = "wallet_3" +slots = 1 +btc_address = "mvZtbibDAAA3WLpY7zXXFqRa3T4XSknBX7" diff --git a/GPU/tests/farm_test.ts b/GPU/tests/farm_test.ts new file mode 100644 index 0000000..9a18ae0 --- /dev/null +++ b/GPU/tests/farm_test.ts @@ -0,0 +1,26 @@ + +import { Clarinet, Tx, Chain, Account, types } from 'https://deno.land/x/clarinet@v0.14.0/index.ts'; +import { assertEquals } from 'https://deno.land/std@0.90.0/testing/asserts.ts'; + +Clarinet.test({ + name: "Ensure that <...>", + async fn(chain: Chain, accounts: Map) { + let block = chain.mineBlock([ + /* + * Add transactions with: + * Tx.contractCall(...) + */ + ]); + assertEquals(block.receipts.length, 0); + assertEquals(block.height, 2); + + block = chain.mineBlock([ + /* + * Add transactions with: + * Tx.contractCall(...) + */ + ]); + assertEquals(block.receipts.length, 0); + assertEquals(block.height, 3); + }, +});