Compare commits
No commits in common. "3da6a57d509eba80209415bcbefec39875e33da7" and "9ab31a6d9bdeea04dd24552d2d2fa6003f48a3c0" have entirely different histories.
3da6a57d50
...
9ab31a6d9b
4 changed files with 84 additions and 172 deletions
|
@ -33,6 +33,7 @@ import Database.Persist
|
||||||
import Lens.Micro ((&), (+~), (.~), (?~), (^.), set)
|
import Lens.Micro ((&), (+~), (.~), (?~), (^.), set)
|
||||||
import Lens.Micro.TH
|
import Lens.Micro.TH
|
||||||
import Monomer
|
import Monomer
|
||||||
|
|
||||||
import qualified Monomer.Lens as L
|
import qualified Monomer.Lens as L
|
||||||
import System.Directory (getHomeDirectory)
|
import System.Directory (getHomeDirectory)
|
||||||
import System.FilePath ((</>))
|
import System.FilePath ((</>))
|
||||||
|
@ -68,16 +69,15 @@ import Zenith.Types hiding (ZcashAddress(..))
|
||||||
import Zenith.Utils
|
import Zenith.Utils
|
||||||
( displayAmount
|
( displayAmount
|
||||||
, getChainTip
|
, getChainTip
|
||||||
|
, getZcashPrice
|
||||||
, isRecipientValidGUI
|
, isRecipientValidGUI
|
||||||
, isValidString
|
, isValidString
|
||||||
, isZecAddressValid
|
, isZecAddressValid
|
||||||
, jsonNumber
|
, jsonNumber
|
||||||
, padWithZero
|
, padWithZero
|
||||||
|
, parseZcashPayment
|
||||||
, showAddress
|
, showAddress
|
||||||
, validBarValue
|
, validBarValue
|
||||||
, parseZcashPayment
|
|
||||||
, getZcashPrice
|
|
||||||
, createZip321
|
|
||||||
)
|
)
|
||||||
|
|
||||||
data VkTypeDef
|
data VkTypeDef
|
||||||
|
@ -96,6 +96,7 @@ data AppEvent
|
||||||
| AccountClicked
|
| AccountClicked
|
||||||
| MenuClicked
|
| MenuClicked
|
||||||
| NewClicked
|
| NewClicked
|
||||||
|
| ViewingKeysClicked
|
||||||
| NewAddress !(Maybe (Entity ZcashAccount))
|
| NewAddress !(Maybe (Entity ZcashAccount))
|
||||||
| NewInternalAddress !(Maybe (Entity ZcashAccount))
|
| NewInternalAddress !(Maybe (Entity ZcashAccount))
|
||||||
| NewAccount !(Maybe (Entity ZcashWallet))
|
| NewAccount !(Maybe (Entity ZcashWallet))
|
||||||
|
@ -161,21 +162,15 @@ data AppEvent
|
||||||
| ShowFIATBalance
|
| ShowFIATBalance
|
||||||
| DisplayFIATBalance Double Double
|
| DisplayFIATBalance Double Double
|
||||||
| CloseFIATBalance
|
| CloseFIATBalance
|
||||||
| ViewingKeysClicked
|
|
||||||
| PrepareViewingKey !VkTypeDef !(Maybe (Entity ZcashAccount))
|
| PrepareViewingKey !VkTypeDef !(Maybe (Entity ZcashAccount))
|
||||||
| ShowViewingKey !VkTypeDef !T.Text
|
| ShowViewingKey !VkTypeDef !T.Text
|
||||||
| CopyViewingKey !T.Text !T.Text
|
| CopyViewingKey !T.Text !T.Text
|
||||||
| CloseShowVK
|
| CloseShowVK
|
||||||
| DisplayPaymentURIForm !T.Text
|
| DisplayPaymentURI
|
||||||
| ClosePaymentURIForm
|
| ClosePaymentURI
|
||||||
| PrepareURIString
|
|
||||||
| CloseShowURIOverlay
|
|
||||||
| ShowURIOverlay !T.Text
|
|
||||||
| CopyURIString !T.Text
|
|
||||||
| DisplayPayUsingURI
|
| DisplayPayUsingURI
|
||||||
| ClosePayUsingURI
|
| ClosePayUsingURI
|
||||||
| ProcIfValidURI
|
| ProcIfValidURI
|
||||||
| PreparePaymentURIForm
|
|
||||||
deriving (Eq, Show)
|
deriving (Eq, Show)
|
||||||
|
|
||||||
data AppModel = AppModel
|
data AppModel = AppModel
|
||||||
|
@ -243,10 +238,8 @@ data AppModel = AppModel
|
||||||
, _vkTypeName :: !T.Text
|
, _vkTypeName :: !T.Text
|
||||||
, _vkData :: !T.Text
|
, _vkData :: !T.Text
|
||||||
, _paymentURIDisplay :: !Bool
|
, _paymentURIDisplay :: !Bool
|
||||||
, _showURIDisplay :: !Bool
|
|
||||||
, _usepmtURIOverlay :: !Bool
|
, _usepmtURIOverlay :: !Bool
|
||||||
, _uriString :: !T.Text
|
, _uriString :: !T.Text
|
||||||
, _uriAddr :: !T.Text
|
|
||||||
} deriving (Eq, Show)
|
} deriving (Eq, Show)
|
||||||
|
|
||||||
makeLenses ''AppModel
|
makeLenses ''AppModel
|
||||||
|
@ -290,19 +283,18 @@ buildUI wenv model = widgetTree
|
||||||
, modalOverlay `nodeVisible` isJust (model ^. modalMsg)
|
, modalOverlay `nodeVisible` isJust (model ^. modalMsg)
|
||||||
, adrbookOverlay `nodeVisible` model ^. showAdrBook
|
, adrbookOverlay `nodeVisible` model ^. showAdrBook
|
||||||
, newAdrBkOverlay `nodeVisible` model ^. newAdrBkEntry
|
, newAdrBkOverlay `nodeVisible` model ^. newAdrBkEntry
|
||||||
|
, dfBalOverlay `nodeVisible` model ^. displayFIATBalance
|
||||||
, showABAddressOverlay (model ^. abdescrip) (model ^. abaddress) `nodeVisible`
|
, showABAddressOverlay (model ^. abdescrip) (model ^. abaddress) `nodeVisible`
|
||||||
model ^.
|
model ^.
|
||||||
showABAddress
|
showABAddress
|
||||||
, updateABAddressOverlay (model ^. abdescrip) (model ^. abaddress) `nodeVisible`
|
, updateABAddressOverlay (model ^. abdescrip) (model ^. abaddress) `nodeVisible`
|
||||||
model ^.
|
model ^.
|
||||||
updateABAddress
|
updateABAddress
|
||||||
, shieldOverlay `nodeVisible` model ^. shieldZec
|
|
||||||
, deShieldOverlay `nodeVisible` model ^. deShieldZec
|
|
||||||
, dfBalOverlay `nodeVisible` model ^. displayFIATBalance
|
|
||||||
, showVKOverlay `nodeVisible` model ^. viewingKeyDisplay
|
, showVKOverlay `nodeVisible` model ^. viewingKeyDisplay
|
||||||
, paymentURIOverlay `nodeVisible` model ^. paymentURIDisplay
|
, paymentURIOverlay `nodeVisible` model ^. paymentURIDisplay
|
||||||
, showURIOverlay `nodeVisible` model ^. showURIDisplay
|
|
||||||
, pmtUsingURIOverlay `nodeVisible` model ^. usepmtURIOverlay
|
, pmtUsingURIOverlay `nodeVisible` model ^. usepmtURIOverlay
|
||||||
|
, shieldOverlay `nodeVisible` model ^. shieldZec
|
||||||
|
, deShieldOverlay `nodeVisible` model ^. deShieldZec
|
||||||
, msgAdrBookOverlay `nodeVisible` isJust (model ^. msgAB)
|
, msgAdrBookOverlay `nodeVisible` isJust (model ^. msgAB)
|
||||||
]
|
]
|
||||||
mainWindow =
|
mainWindow =
|
||||||
|
@ -395,7 +387,7 @@ buildUI wenv model = widgetTree
|
||||||
("Balance in " <>
|
("Balance in " <>
|
||||||
T.toUpper (c_currencyCode (model ^. configuration)))) `styleBasic`
|
T.toUpper (c_currencyCode (model ^. configuration)))) `styleBasic`
|
||||||
[bgColor white, borderB 1 gray, padding 3]
|
[bgColor white, borderB 1 gray, padding 3]
|
||||||
, box_ [alignLeft, onClick PreparePaymentURIForm] (label "Create URI") `styleBasic`
|
, box_ [alignLeft, onClick DisplayPaymentURI] (label "Create URI") `styleBasic`
|
||||||
[bgColor white, borderB 1 gray, padding 3]
|
[bgColor white, borderB 1 gray, padding 3]
|
||||||
, box_
|
, box_
|
||||||
[alignLeft, onClick DisplayPayUsingURI]
|
[alignLeft, onClick DisplayPayUsingURI]
|
||||||
|
@ -1249,68 +1241,66 @@ buildUI wenv model = widgetTree
|
||||||
, filler
|
, filler
|
||||||
]) `styleBasic`
|
]) `styleBasic`
|
||||||
[bgColor (white & L.a .~ 0.5)]
|
[bgColor (white & L.a .~ 0.5)]
|
||||||
--
|
|
||||||
paymentURIOverlay =
|
paymentURIOverlay =
|
||||||
alert ClosePaymentURIForm $
|
box
|
||||||
vstack
|
(vstack
|
||||||
[ box_
|
[ filler
|
||||||
[]
|
|
||||||
(label "Create a Payment URI" `styleBasic`
|
|
||||||
[textFont "Bold", textColor white, textSize 10, padding 3]) `styleBasic`
|
|
||||||
[bgColor btnColor, radius 2, padding 3]
|
|
||||||
, spacer
|
|
||||||
, hstack
|
, hstack
|
||||||
[ filler
|
[ filler
|
||||||
, label "Current Address:" `styleBasic` [textFont "Bold"]
|
, box_
|
||||||
, spacer
|
[]
|
||||||
, label_ (txtWrapN (model ^. uriAddr) 64) [multiline]
|
(vstack
|
||||||
, filler
|
[ box_
|
||||||
]
|
[alignMiddle]
|
||||||
|
(label "Create URI" `styleBasic`
|
||||||
|
[textColor white, textFont "Bold", textSize 12]) `styleBasic`
|
||||||
|
[bgColor btnColor]
|
||||||
|
, separatorLine `styleBasic` [fgColor btnColor]
|
||||||
, spacer
|
, spacer
|
||||||
, hstack
|
, hstack
|
||||||
[ label "Amount : " `styleBasic` [textFont "Bold"]
|
[ label "Amount:" `styleBasic`
|
||||||
, numericField_ sendAmount
|
[width 50, textFont "Bold"]
|
||||||
[ decimals 8 ]
|
, spacer
|
||||||
`nodeKey` "floatInput"
|
, numericField_
|
||||||
`styleBasic`
|
sendAmount
|
||||||
|
[ decimals 8
|
||||||
|
, minValue 0.0
|
||||||
|
, maxValue
|
||||||
|
(fromIntegral (model ^. balance) / 100000000.0)
|
||||||
|
, validInput amountValid
|
||||||
|
, onChange CheckAmount
|
||||||
|
] `styleBasic`
|
||||||
[ width 150
|
[ width 150
|
||||||
, styleIf (model ^. sendAmount <= 0.0) (textColor red)
|
, styleIf
|
||||||
|
(not $ model ^. amountValid)
|
||||||
|
(textColor red)
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
, spacer
|
|
||||||
, hstack
|
, hstack
|
||||||
[ label "Memo: " `styleBasic` [textFont "Bold"]
|
[ label "Memo:" `styleBasic`
|
||||||
|
[width 50, textFont "Bold"]
|
||||||
, spacer
|
, spacer
|
||||||
, textField_ sendMemo [] `styleBasic` [width 300]
|
, textArea sendMemo `styleBasic`
|
||||||
|
[width 150, height 40]
|
||||||
]
|
]
|
||||||
, spacer
|
, spacer
|
||||||
, hstack
|
, box_
|
||||||
[ filler
|
[alignMiddle]
|
||||||
, mainButton "Create URI" PrepareURIString `nodeEnabled`
|
(hstack
|
||||||
(model ^. sendAmount > 0.0)
|
[ spacer
|
||||||
|
, mainButton "Create URI" NotImplemented `nodeEnabled`
|
||||||
|
True
|
||||||
|
, spacer
|
||||||
|
, button "Cancel" ClosePaymentURI
|
||||||
|
, spacer
|
||||||
|
])
|
||||||
|
]) `styleBasic`
|
||||||
|
[radius 4, border 2 btnColor, bgColor white, padding 4]
|
||||||
, filler
|
, filler
|
||||||
]
|
]
|
||||||
] `styleBasic` [bgColor (white & L.a .~ 0.5)]
|
|
||||||
--
|
|
||||||
showURIOverlay =
|
|
||||||
alert CloseShowURIOverlay $
|
|
||||||
vstack
|
|
||||||
[ box_
|
|
||||||
[]
|
|
||||||
(label "Payment URI" `styleBasic`
|
|
||||||
[textFont "Bold", textColor white, textSize 12, padding 3]) `styleBasic`
|
|
||||||
[bgColor btnColor, radius 2, padding 3]
|
|
||||||
, spacer
|
|
||||||
, hstack
|
|
||||||
[filler, label_ (txtWrapN (model ^. uriString ) 64) [multiline], filler]
|
|
||||||
, spacer
|
|
||||||
, hstack
|
|
||||||
[ filler
|
|
||||||
, button "Copy to Clipboard" $
|
|
||||||
CopyURIString (model ^. uriString)
|
|
||||||
, filler
|
, filler
|
||||||
]
|
]) `styleBasic`
|
||||||
]
|
[bgColor (white & L.a .~ 0.5)]
|
||||||
pmtUsingURIOverlay =
|
pmtUsingURIOverlay =
|
||||||
box
|
box
|
||||||
(vstack
|
(vstack
|
||||||
|
@ -1704,7 +1694,6 @@ handleEvent wenv node model evt =
|
||||||
model & amountValid .~
|
model & amountValid .~
|
||||||
(i < (fromIntegral (model ^. balance) / 100000000.0))
|
(i < (fromIntegral (model ^. balance) / 100000000.0))
|
||||||
]
|
]
|
||||||
--
|
|
||||||
ShowTxId tx -> [Model $ model & showId ?~ tx & modalMsg .~ Nothing]
|
ShowTxId tx -> [Model $ model & showId ?~ tx & modalMsg .~ Nothing]
|
||||||
-- |
|
-- |
|
||||||
-- | Address Book Events
|
-- | Address Book Events
|
||||||
|
@ -1763,11 +1752,6 @@ handleEvent wenv node model evt =
|
||||||
, setClipboardData $ ClipboardText v
|
, setClipboardData $ ClipboardText v
|
||||||
, Event $ ShowMessage (t <> " viewing key copied!!")
|
, Event $ ShowMessage (t <> " viewing key copied!!")
|
||||||
]
|
]
|
||||||
CopyURIString u ->
|
|
||||||
[ setClipboardData ClipboardEmpty
|
|
||||||
, setClipboardData $ ClipboardText u
|
|
||||||
, Event $ ShowMessage "URI string copied to clipboard!!"
|
|
||||||
]
|
|
||||||
DeleteABEntry a ->
|
DeleteABEntry a ->
|
||||||
[ Task $ deleteAdrBook (model ^. configuration) a
|
[ Task $ deleteAdrBook (model ^. configuration) a
|
||||||
, Model $
|
, Model $
|
||||||
|
@ -1830,31 +1814,11 @@ handleEvent wenv node model evt =
|
||||||
--
|
--
|
||||||
-- Display PaymentURI Form
|
-- Display PaymentURI Form
|
||||||
--
|
--
|
||||||
PreparePaymentURIForm ->
|
DisplayPaymentURI ->
|
||||||
[ Task $ getCurrentAddress currentAddress ]
|
|
||||||
--
|
|
||||||
DisplayPaymentURIForm ua->
|
|
||||||
[ Model $
|
[ Model $
|
||||||
model & uriString .~ ""
|
model & paymentURIDisplay .~ True & uriString .~ "" & menuPopup .~ False
|
||||||
& uriAddr .~ ua
|
|
||||||
& amountValid .~ False
|
|
||||||
& sendAmount .~ 0.0
|
|
||||||
& sendMemo .~ ""
|
|
||||||
& paymentURIDisplay .~ True
|
|
||||||
& menuPopup .~ False
|
|
||||||
]
|
]
|
||||||
ClosePaymentURIForm -> [Model $ model & paymentURIDisplay .~ False]
|
ClosePaymentURI -> [Model $ model & paymentURIDisplay .~ False]
|
||||||
--
|
|
||||||
-- Generate URI
|
|
||||||
--
|
|
||||||
PrepareURIString -> [ Task $ genURIString (model ^. uriAddr) (model ^. sendAmount) (model ^. sendMemo) ]
|
|
||||||
ShowURIOverlay uStr ->
|
|
||||||
[ Model $
|
|
||||||
model & uriString .~ uStr
|
|
||||||
& paymentURIDisplay .~ False
|
|
||||||
& showURIDisplay .~ True
|
|
||||||
]
|
|
||||||
CloseShowURIOverlay -> [ Model $ model & showURIDisplay .~ False & uriString .~ "" ]
|
|
||||||
--
|
--
|
||||||
-- Display Pay using URI Form
|
-- Display Pay using URI Form
|
||||||
--
|
--
|
||||||
|
@ -1877,8 +1841,9 @@ handleEvent wenv node model evt =
|
||||||
T.pack (uriAddress p) &
|
T.pack (uriAddress p) &
|
||||||
sendAmount .~
|
sendAmount .~
|
||||||
realToFrac a &
|
realToFrac a &
|
||||||
sendMemo .~ (uriMemo p)
|
sendMemo .~
|
||||||
, Event $ ClosePaymentURIForm
|
(uriMemo p)
|
||||||
|
, Event $ ClosePaymentURI
|
||||||
]
|
]
|
||||||
Nothing ->
|
Nothing ->
|
||||||
[ Model $
|
[ Model $
|
||||||
|
@ -2097,28 +2062,6 @@ handleEvent wenv node model evt =
|
||||||
let tsk = getTranSK $ zcashAccountTPrivateKey $ entityVal acc
|
let tsk = getTranSK $ zcashAccountTPrivateKey $ entityVal acc
|
||||||
ivk <- deriveUivk n osk ssk tsk
|
ivk <- deriveUivk n osk ssk tsk
|
||||||
return $ ShowViewingKey VkIncoming ivk
|
return $ ShowViewingKey VkIncoming ivk
|
||||||
--
|
|
||||||
-- Get curret zcash address
|
|
||||||
--
|
|
||||||
getCurrentAddress :: Maybe (Entity WalletAddress) -> IO AppEvent
|
|
||||||
getCurrentAddress a = do
|
|
||||||
let ua = case model ^. selPool of
|
|
||||||
OrchardPool -> maybe "None" (getUA . walletAddressUAddress . entityVal) a
|
|
||||||
SaplingPool -> fromMaybe "None" $ (getSaplingFromUA . E.encodeUtf8 . getUA . walletAddressUAddress . entityVal) =<< a
|
|
||||||
SproutPool -> "None"
|
|
||||||
TransparentPool -> maybe "None" (encodeTransparentReceiver (model ^. network)) $
|
|
||||||
t_rec =<< (isValidUnifiedAddress . E.encodeUtf8 . getUA . walletAddressUAddress . entityVal) =<< a
|
|
||||||
return $ DisplayPaymentURIForm ua
|
|
||||||
--
|
|
||||||
-- Gen URI String
|
|
||||||
--
|
|
||||||
genURIString :: T.Text -> Float -> T.Text -> IO AppEvent
|
|
||||||
genURIString addr mAmt mMemo = do
|
|
||||||
let mM = case mMemo of
|
|
||||||
"" -> Nothing
|
|
||||||
_ -> Just (T.unpack mMemo)
|
|
||||||
let uriSt = createZip321 (T.unpack addr) (Just (realToFrac mAmt)) mM
|
|
||||||
return $ ShowURIOverlay (T.pack uriSt)
|
|
||||||
|
|
||||||
scanZebra ::
|
scanZebra ::
|
||||||
T.Text
|
T.Text
|
||||||
|
@ -2472,8 +2415,6 @@ runZenithGUI config = do
|
||||||
""
|
""
|
||||||
False
|
False
|
||||||
False
|
False
|
||||||
False
|
|
||||||
""
|
|
||||||
""
|
""
|
||||||
startApp model handleEvent buildUI (params hD)
|
startApp model handleEvent buildUI (params hD)
|
||||||
Left _e -> print "Zebra not available"
|
Left _e -> print "Zebra not available"
|
||||||
|
|
|
@ -22,7 +22,6 @@ import qualified Data.Text as T
|
||||||
import qualified Data.Text.Encoding as E
|
import qualified Data.Text.Encoding as E
|
||||||
import qualified Data.Text.Encoding as TE
|
import qualified Data.Text.Encoding as TE
|
||||||
import Network.HTTP.Simple
|
import Network.HTTP.Simple
|
||||||
import Network.URI (escapeURIString, isUnreserved)
|
|
||||||
import System.Directory
|
import System.Directory
|
||||||
import System.Process (createProcess_, shell)
|
import System.Process (createProcess_, shell)
|
||||||
import Text.Printf (printf)
|
import Text.Printf (printf)
|
||||||
|
@ -335,22 +334,3 @@ padBase64 bs = bs <> BC.replicate paddingLength '='
|
||||||
-- Function to decode a base64 un-padded string
|
-- Function to decode a base64 un-padded string
|
||||||
decodeBase64Unpadded :: BC.ByteString -> Either String BC.ByteString
|
decodeBase64Unpadded :: BC.ByteString -> Either String BC.ByteString
|
||||||
decodeBase64Unpadded = B64.decode . padBase64
|
decodeBase64Unpadded = B64.decode . padBase64
|
||||||
|
|
||||||
-- Function to encode memo as un-padded Base64
|
|
||||||
encodeBase64Memo :: String -> String
|
|
||||||
encodeBase64Memo = BC.unpack . BC.takeWhile (/= '=') . B64.encode . BC.pack
|
|
||||||
|
|
||||||
-- Function to drop trailing zeros
|
|
||||||
dropTrailingZeros :: String -> String
|
|
||||||
dropTrailingZeros str =
|
|
||||||
let withoutZeros = reverse (dropWhile (== '0') (reverse str))
|
|
||||||
in if last withoutZeros == '.'
|
|
||||||
then withoutZeros ++ "0" -- Ensure at least one decimal place
|
|
||||||
else withoutZeros
|
|
||||||
|
|
||||||
-- Function to create a ZIP-321 URI
|
|
||||||
createZip321 :: String -> Maybe Double -> Maybe String -> String
|
|
||||||
createZip321 address mAmount mMemo =
|
|
||||||
"zcash:" ++ address
|
|
||||||
++ maybe "" (\amount -> "?amount=" ++ dropTrailingZeros (printf "%.8f" amount) ) mAmount
|
|
||||||
++ maybe "" (\memo -> "&memo=" ++ escapeURIString isUnreserved (encodeBase64Memo memo)) mMemo
|
|
||||||
|
|
|
@ -1138,11 +1138,3 @@ main = do
|
||||||
print p
|
print p
|
||||||
(uriAmount p) `shouldBe` Just 100.0
|
(uriAmount p) `shouldBe` Just 100.0
|
||||||
Left e -> assertFailure $ "Error: " ++ e
|
Left e -> assertFailure $ "Error: " ++ e
|
||||||
describe "Create a ZIP-321 URI payment string " $ do
|
|
||||||
it "Creating an URI using a valid Zcash address, an amount, and a memo " $ do
|
|
||||||
let address = "ztestsapling10yy2ex5dcqkclhc7z7yrnjq2z6feyjad56ptwlfgmy77dmaqqrl9gyhprdx59qgmsnyfska2kez"
|
|
||||||
let amount = Just 1.2345
|
|
||||||
let memo = Just "This is a simple memo."
|
|
||||||
let uriString = createZip321 address amount memo
|
|
||||||
print uriString
|
|
||||||
uriString `shouldBe` "zcash:ztestsapling10yy2ex5dcqkclhc7z7yrnjq2z6feyjad56ptwlfgmy77dmaqqrl9gyhprdx59qgmsnyfska2kez?amount=1.2345&memo=VGhpcyBpcyBhIHNpbXBsZSBtZW1vLg"
|
|
||||||
|
|
|
@ -97,7 +97,6 @@ library
|
||||||
, word-wrap
|
, word-wrap
|
||||||
, zcash-haskell
|
, zcash-haskell
|
||||||
, unordered-containers
|
, unordered-containers
|
||||||
, network-uri
|
|
||||||
--pkgconfig-depends: rustzcash_wrapper
|
--pkgconfig-depends: rustzcash_wrapper
|
||||||
default-language: Haskell2010
|
default-language: Haskell2010
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue