Source Code

;; Storage
;; pop-id:block-height
(define-map frozen-pops uint uint)

;; Define Constants
(define-constant CONTRACT-OWNER tx-sender)

;; Freeze 1 week min
(define-constant MIN-FREEZING-BLOCKS u1000)
(define-constant ICE-PER-POP-PER-BLOCK u1)

;; Define Errors
(define-constant ERR-NOT-AUTHORIZED (err u401))
(define-constant ERR-NOT-FOUND (err u404))
(define-constant ERR-SWITCHED-OFF (err u500))
(define-constant ERR-TOO-EARLY (err u501))
(define-constant ERR-FATAL (err u999))

;; Define Variables
(define-data-var running bool false)

;; Freeze a Pop
(define-private (freeze (id uint))
  (let ((owner (unwrap! (unwrap! (contract-call? 'SPJW1XE278YMCEYMXB8ZFGJMH8ZVAAEDP2S2PJYG.stacks-pops get-owner id) ERR-FATAL) ERR-NOT-FOUND)))
    (asserts! (var-get running) ERR-SWITCHED-OFF)
    (asserts! (is-eq owner tx-sender) ERR-NOT-AUTHORIZED)
    (asserts! (map-insert frozen-pops id block-height) ERR-FATAL)
    (try! (push-to-vault id))
    (contract-call? .frozen-stacks-pops-v2 mint tx-sender id)))

(define-private (push-to-vault (id uint))
  (let ((slot (mod id u4)))
    (and (is-eq slot u0)  (try! (contract-call? .stacks-pops-vault-0-v1 push-pop id)))
    (and (is-eq slot u1)  (try! (contract-call? .stacks-pops-vault-1-v1 push-pop id)))
    (and (is-eq slot u2)  (try! (contract-call? .stacks-pops-vault-2-v1 push-pop id)))
    (and (is-eq slot u3)  (try! (contract-call? .stacks-pops-vault-3-v1 push-pop id)))
    (ok true)))

(define-private (pull-from-vault (id uint) (owner principal))
  (let ((slot (mod id u4)))
    (and (is-eq slot u0)  (try! (as-contract (contract-call? .stacks-pops-vault-0-v1 pull-pop id owner))))
    (and (is-eq slot u1)  (try! (as-contract (contract-call? .stacks-pops-vault-1-v1 pull-pop id owner))))
    (and (is-eq slot u2)  (try! (as-contract (contract-call? .stacks-pops-vault-2-v1 pull-pop id owner))))
    (and (is-eq slot u3)  (try! (as-contract (contract-call? .stacks-pops-vault-3-v1 pull-pop id owner))))
    (ok true)))

(define-public (freeze-three (id1 uint) (id2 uint) (id3 uint))
  (begin
    (try! (freeze id1))
    (try! (freeze id2))
    (try! (freeze id3))
    (ok true)))

(define-private (check-err (item (response bool uint)) (result (response bool uint)))
  (if (is-err item)
    item
    result))

(define-public (freeze-many (ids (list 25 uint)))
  (begin
    (asserts! (>= (len ids) u3) ERR-FATAL)
    (fold check-err (map freeze ids) (ok true))))

(define-private (defrost (id uint))
  (let 
    (
      (freeze-bh (unwrap! (map-get? frozen-pops id) ERR-NOT-FOUND))
      (ice-cubes (* ICE-PER-POP-PER-BLOCK (- block-height freeze-bh)))
      (owner (unwrap! (unwrap! (contract-call? .frozen-stacks-pops-v2 get-owner id) ERR-FATAL) ERR-NOT-FOUND))
    )
    (asserts! (is-eq owner tx-sender) ERR-NOT-AUTHORIZED)
    (asserts! (>= block-height (+ freeze-bh MIN-FREEZING-BLOCKS)) ERR-TOO-EARLY)
    (map-delete frozen-pops id)
    (try! (contract-call? .frozen-stacks-pops-v2 burn id tx-sender))
    (try! (pull-from-vault id owner))
    (match (as-contract (contract-call? .stacks-pops-ice-v2 transfer ice-cubes tx-sender owner (some 0x646566726f737420726577617264)))
      okValue (ok okValue)
      ;; We ignore the error since we want the user to still be able to defrost if the machine doesn't have enough $ICE
      errValue (ok true) 
    )
  )
)

(define-public (defrost-three (id1 uint) (id2 uint) (id3 uint))
  (begin
    (try! (defrost id1))
    (try! (defrost id2))
    (try! (defrost id3))
    (ok true)))

(define-public (defrost-many (ids (list 25 uint)))
  (fold check-err (map defrost ids) (ok true)))

;; Switch power on or off
(define-public (flip-power-switch)
  (begin
    (asserts! (is-eq contract-caller CONTRACT-OWNER) ERR-NOT-AUTHORIZED)
    (var-set running (not (var-get running)))
    (ok (var-get running))))

;; Get pop freeze height
(define-read-only (get-freeze-block-height (id uint))
  (default-to u0
    (map-get? frozen-pops id)))

;; Get machine state
(define-read-only (get-machine-state)
  (ok (var-get running)))

;; Get the machine ice balance
(define-read-only (get-machine-ice-balance)
  (unwrap! (as-contract (contract-call? .stacks-pops-ice-v2 get-caller-balance)) u0))

;; Swap v1 tokens to v2 tokens 
(define-public (swap)
  (let ((owner tx-sender)
    (ice-v1-balance (unwrap! (contract-call? 'SP277HZA8AGXV42MZKDW5B2NNN61RHQ42MTAHVNB1.stacks-pops-ice-v1 get-caller-balance) ERR-FATAL)))
    (try! (as-contract (contract-call? .stacks-pops-ice-v2 transfer ice-v1-balance tx-sender owner (some 0x746F6B656E2073776170))))
    (try! (contract-call? 'SP277HZA8AGXV42MZKDW5B2NNN61RHQ42MTAHVNB1.stacks-pops-ice-v1 transfer ice-v1-balance owner 'SP277HZA8AGXV42MZKDW5B2NNN61RHQ42MTAHVNB1.stacks-pops-ice-machine-v1 (some 0x6275726E)))
    (ok true)))

;; set mint address for frozen-pops
(as-contract (contract-call? .frozen-stacks-pops-v2 set-mint-address))
(as-contract (contract-call? .stacks-pops-ice-v2 set-ice-machine tx-sender))

(as-contract (contract-call? .stacks-pops-vault-0-v1 set-ice-machine-address))
(as-contract (contract-call? .stacks-pops-vault-1-v1 set-ice-machine-address))
(as-contract (contract-call? .stacks-pops-vault-2-v1 set-ice-machine-address))
(as-contract (contract-call? .stacks-pops-vault-3-v1 set-ice-machine-address))

Functions (14)

FunctionAccessArgs
freezeprivateid: uint
push-to-vaultprivateid: uint
pull-from-vaultprivateid: uint, owner: principal
freeze-threepublicid1: uint, id2: uint, id3: uint
check-errprivateitem: (response bool uint
freeze-manypublicids: (list 25 uint
defrostprivateid: uint
defrost-threepublicid1: uint, id2: uint, id3: uint
defrost-manypublicids: (list 25 uint
flip-power-switchpublic
get-freeze-block-heightread-onlyid: uint
get-machine-stateread-only
get-machine-ice-balanceread-only
swappublic