Source Code

(impl-trait 'SP2PABAF9FTAJYNFZH93XENAJ8FVY99RRM50D2JG9.nft-trait.nft-trait)

(define-non-fungible-token rock uint)

(define-data-var fee-per-rock uint u100000000)
(define-data-var number-of-rocks uint u0)

;; storage of mint-address and admin
(define-map mint-address bool principal)
;; storage of marketplace-address
(define-map marketplace-address bool principal)
;; storage of admin-address
(define-map admin bool principal)

(define-map approvals {owner: principal, operator: principal, id: uint} bool)
(define-map approvals-all {owner: principal, operator: principal} bool)

(define-read-only (get-last-token-id)
  (ok u50))

(define-read-only (get-number-of-rocks)
  (var-get number-of-rocks))

(define-read-only (get-floor)
  (* (var-get number-of-rocks) (var-get fee-per-rock)))

(define-read-only (get-fee-per-rock)
  (var-get fee-per-rock))

(define-public (get-token-uri (id uint))
  (ok (some "ipfs://Qmeq7Z5vdhJwfKauf2XWL44YqgCTpHmAgrzrb1BKSEeY7x/{id}.json")))

(define-public (get-owner (id uint))
  (ok (nft-get-owner? rock id)))

;;
;; transfer functions
;;
(define-private (floor-stx-transfer (id uint))
  (match (nft-get-owner? rock id)
  rock-owner
    (if (not (is-eq tx-sender rock-owner))
      (stx-transfer? (var-get fee-per-rock) tx-sender rock-owner)
      (ok true))
  (ok true)))

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

(define-private (trnsfr (id uint) (sender principal) (recipient principal))
  (let ((floor-result (map floor-stx-transfer indices)))
    (try! (fold check-err floor-result (ok true)))
    (nft-transfer? rock id sender recipient)))

 ;; SIP009: Transfer token to a specified principal
(define-public (transfer (id uint) (sender principal) (recipient principal))
  (let ((owner (unwrap! (nft-get-owner? rock id) err-not-found)))
    (asserts! (is-eq sender owner) err-invalid-sender)
    (asserts! (is-approved-with-owner id contract-caller owner) err-not-authorized)
    (trnsfr id sender recipient)))

(define-public (transfer-memo (id uint) (sender principal) (recipient principal) (memo (buff 34)))
    (let ((result (transfer id sender recipient)))
      (print memo)
      result))
;;
;; operable functions
;;
(define-private (is-approved-with-owner (id uint) (operator principal) (owner principal))
  (or
    (is-eq owner operator)
    (default-to (default-to
      (is-eq operator (get-marketplace))
        (map-get? approvals-all {owner: owner, operator: operator}))
          (map-get? approvals {owner: owner, operator: operator, id: id}))))

(define-read-only (is-approved (id uint) (operator principal))
  (let ((owner (unwrap! (nft-get-owner? rock id) err-not-found)))
    (ok (is-approved-with-owner id operator owner))))

(define-public (set-approved (id uint) (operator principal) (approved bool))
  (let ((owner (unwrap! (nft-get-owner? rock id) err-not-found)))
	  (ok (map-set approvals {owner: contract-caller, operator: operator, id: id} approved))))

(define-public (set-approved-all (operator principal) (approved bool))
	  (ok (map-set approvals-all {owner: contract-caller, operator: operator} approved)))

;; upgrade btc rock - can only be called from mint contract
(define-public (upgrade (id uint))
  (begin
    (asserts! (is-eq contract-caller (unwrap! (map-get? mint-address true) err-fatal)) err-not-authorized)
    (var-set number-of-rocks (+ u1 (var-get number-of-rocks)))
    (nft-mint? rock id tx-sender)))

;;
;; admin function
;;
;; can only be called once
(define-public (set-mint)
  (let ((the-mint (map-get? mint-address true)))
    (asserts! (and (is-none the-mint)
              (map-insert mint-address true contract-caller))
                err-fatal)
    (print contract-caller)
    (ok contract-caller)))

;; can only be called once
(define-public (set-marketplace)
  (let ((marketplace (map-get? marketplace-address true)))
    (asserts! (and
                (is-none marketplace)
                (map-insert marketplace-address true contract-caller))
              err-fatal)
    (print contract-caller)
    (ok contract-caller)))

(define-read-only (get-marketplace)
  (unwrap! (map-get? marketplace-address true) 'SP000000000000000000002Q6VF78))

(define-public (set-fee-per-rock (ustx uint))
  (begin
    (asserts! (is-eq (unwrap! (map-get? admin true) err-fatal) contract-caller) err-not-authorized)
    (ok (var-set fee-per-rock ustx))))

(define-public (set-admin (user principal))
  (begin
    (asserts! (is-eq (unwrap! (map-get? admin true) err-fatal) contract-caller) err-not-authorized)
    (ok (map-set admin true user))))

(define-constant indices
  (list
    u1 u2 u3 u4 u5 u6 u7 u8 u9 u10
    u11 u12 u13 u14 u15 u16 u17 u18 u19 u20
    u21 u22 u23 u24 u25 u26 u27 u28 u29 u30
    u31 u32 u33 u34 u35 u36 u37 u38 u39 u40
    u41 u42 u43 u44 u45 u46 u47 u48 u49 u50))

(define-constant err-not-authorized (err u403))
(define-constant err-not-found (err u404))
(define-constant err-invalid-sender (err u503))
(define-constant err-fatal (err u999))

Functions (21)

FunctionAccessArgs
get-last-token-idread-only
get-number-of-rocksread-only
get-floorread-only
get-fee-per-rockread-only
get-token-uripublicid: uint
get-ownerpublicid: uint
floor-stx-transferprivateid: uint
check-errprivatecurrent: (response bool uint
trnsfrprivateid: uint, sender: principal, recipient: principal
transferpublicid: uint, sender: principal, recipient: principal
transfer-memopublicid: uint, sender: principal, recipient: principal, memo: (buff 34
is-approved-with-ownerprivateid: uint, operator: principal, owner: principal
is-approvedread-onlyid: uint, operator: principal
set-approvedpublicid: uint, operator: principal, approved: bool
set-approved-allpublicoperator: principal, approved: bool
upgradepublicid: uint
set-mintpublic
set-marketplacepublic
get-marketplaceread-only
set-fee-per-rockpublicustx: uint
set-adminpublicuser: principal