Initial working Hydra demo
This commit is contained in:
		
						commit
						a2e4baf299
					
				
							
								
								
									
										37
									
								
								default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										37
									
								
								default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,37 @@ | |||||||
|  | # default.nix | ||||||
|  | { system ? builtins.currentSystem }: | ||||||
|  | let | ||||||
|  |   nix-thunk = import ./dep/nix-thunk { }; | ||||||
|  |   sources = nix-thunk.mapSubdirectories nix-thunk.thunkSource ./dep; | ||||||
|  |   reflex-platform = import sources.reflex-platform { inherit system; }; | ||||||
|  |   hydra-poc = import sources.hydra-poc { }; | ||||||
|  | in reflex-platform.project ({ pkgs, ... }: | ||||||
|  |   let haskellLib = pkgs.haskell.lib; | ||||||
|  |   in { | ||||||
|  |     packages = { hydra-head-demo = ./.; }; | ||||||
|  | 
 | ||||||
|  |     shellToolOverrides = ghc: super: { | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     overrides = self: super: { | ||||||
|  |       hydra-head-demo = haskellLib.overrideCabal super.hydra-head-demo (drv: { | ||||||
|  |         librarySystemDepends = (drv.librarySystemDepends or [ ]) ++ [ | ||||||
|  |           hydra-poc.cardano-node.cardano-node | ||||||
|  |           hydra-poc.cardano-node.cardano-cli | ||||||
|  |           hydra-poc.hsPkgs.hydra-node.components.exes.hydra-node | ||||||
|  |           hydra-poc.hsPkgs.hydra-node.components.exes.hydra-tools | ||||||
|  |           pkgs.jq | ||||||
|  |           pkgs.coreutils | ||||||
|  |         ]; | ||||||
|  |       }); | ||||||
|  |     }; | ||||||
|  | 
 | ||||||
|  |     useWarp = true; | ||||||
|  | 
 | ||||||
|  |     withHoogle = true; | ||||||
|  | 
 | ||||||
|  |     shells = { | ||||||
|  |       ghc = [ "hydra-head-demo" ]; | ||||||
|  |       ghcjs = [ ]; | ||||||
|  |     }; | ||||||
|  |   }) | ||||||
							
								
								
									
										203
									
								
								demo/LICENSE
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										203
									
								
								demo/LICENSE
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,203 @@ | |||||||
|  | [Original license of https://github.com/input-output-hk/hydra-poc below.] | ||||||
|  | 
 | ||||||
|  |                                  Apache License | ||||||
|  |                            Version 2.0, January 2004 | ||||||
|  |                         http://www.apache.org/licenses/ | ||||||
|  | 
 | ||||||
|  |    TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION | ||||||
|  | 
 | ||||||
|  |    1. Definitions. | ||||||
|  | 
 | ||||||
|  |       "License" shall mean the terms and conditions for use, reproduction, | ||||||
|  |       and distribution as defined by Sections 1 through 9 of this document. | ||||||
|  | 
 | ||||||
|  |       "Licensor" shall mean the copyright owner or entity authorized by | ||||||
|  |       the copyright owner that is granting the License. | ||||||
|  | 
 | ||||||
|  |       "Legal Entity" shall mean the union of the acting entity and all | ||||||
|  |       other entities that control, are controlled by, or are under common | ||||||
|  |       control with that entity. For the purposes of this definition, | ||||||
|  |       "control" means (i) the power, direct or indirect, to cause the | ||||||
|  |       direction or management of such entity, whether by contract or | ||||||
|  |       otherwise, or (ii) ownership of fifty percent (50%) or more of the | ||||||
|  |       outstanding shares, or (iii) beneficial ownership of such entity. | ||||||
|  | 
 | ||||||
|  |       "You" (or "Your") shall mean an individual or Legal Entity | ||||||
|  |       exercising permissions granted by this License. | ||||||
|  | 
 | ||||||
|  |       "Source" form shall mean the preferred form for making modifications, | ||||||
|  |       including but not limited to software source code, documentation | ||||||
|  |       source, and configuration files. | ||||||
|  | 
 | ||||||
|  |       "Object" form shall mean any form resulting from mechanical | ||||||
|  |       transformation or translation of a Source form, including but | ||||||
|  |       not limited to compiled object code, generated documentation, | ||||||
|  |       and conversions to other media types. | ||||||
|  | 
 | ||||||
|  |       "Work" shall mean the work of authorship, whether in Source or | ||||||
|  |       Object form, made available under the License, as indicated by a | ||||||
|  |       copyright notice that is included in or attached to the work | ||||||
|  |       (an example is provided in the Appendix below). | ||||||
|  | 
 | ||||||
|  |       "Derivative Works" shall mean any work, whether in Source or Object | ||||||
|  |       form, that is based on (or derived from) the Work and for which the | ||||||
|  |       editorial revisions, annotations, elaborations, or other modifications | ||||||
|  |       represent, as a whole, an original work of authorship. For the purposes | ||||||
|  |       of this License, Derivative Works shall not include works that remain | ||||||
|  |       separable from, or merely link (or bind by name) to the interfaces of, | ||||||
|  |       the Work and Derivative Works thereof. | ||||||
|  | 
 | ||||||
|  |       "Contribution" shall mean any work of authorship, including | ||||||
|  |       the original version of the Work and any modifications or additions | ||||||
|  |       to that Work or Derivative Works thereof, that is intentionally | ||||||
|  |       submitted to Licensor for inclusion in the Work by the copyright owner | ||||||
|  |       or by an individual or Legal Entity authorized to submit on behalf of | ||||||
|  |       the copyright owner. For the purposes of this definition, "submitted" | ||||||
|  |       means any form of electronic, verbal, or written communication sent | ||||||
|  |       to the Licensor or its representatives, including but not limited to | ||||||
|  |       communication on electronic mailing lists, source code control systems, | ||||||
|  |       and issue tracking systems that are managed by, or on behalf of, the | ||||||
|  |       Licensor for the purpose of discussing and improving the Work, but | ||||||
|  |       excluding communication that is conspicuously marked or otherwise | ||||||
|  |       designated in writing by the copyright owner as "Not a Contribution." | ||||||
|  | 
 | ||||||
|  |       "Contributor" shall mean Licensor and any individual or Legal Entity | ||||||
|  |       on behalf of whom a Contribution has been received by Licensor and | ||||||
|  |       subsequently incorporated within the Work. | ||||||
|  | 
 | ||||||
|  |    2. Grant of Copyright License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       copyright license to reproduce, prepare Derivative Works of, | ||||||
|  |       publicly display, publicly perform, sublicense, and distribute the | ||||||
|  |       Work and such Derivative Works in Source or Object form. | ||||||
|  | 
 | ||||||
|  |    3. Grant of Patent License. Subject to the terms and conditions of | ||||||
|  |       this License, each Contributor hereby grants to You a perpetual, | ||||||
|  |       worldwide, non-exclusive, no-charge, royalty-free, irrevocable | ||||||
|  |       (except as stated in this section) patent license to make, have made, | ||||||
|  |       use, offer to sell, sell, import, and otherwise transfer the Work, | ||||||
|  |       where such license applies only to those patent claims licensable | ||||||
|  |       by such Contributor that are necessarily infringed by their | ||||||
|  |       Contribution(s) alone or by combination of their Contribution(s) | ||||||
|  |       with the Work to which such Contribution(s) was submitted. If You | ||||||
|  |       institute patent litigation against any entity (including a | ||||||
|  |       cross-claim or counterclaim in a lawsuit) alleging that the Work | ||||||
|  |       or a Contribution incorporated within the Work constitutes direct | ||||||
|  |       or contributory patent infringement, then any patent licenses | ||||||
|  |       granted to You under this License for that Work shall terminate | ||||||
|  |       as of the date such litigation is filed. | ||||||
|  | 
 | ||||||
|  |    4. Redistribution. You may reproduce and distribute copies of the | ||||||
|  |       Work or Derivative Works thereof in any medium, with or without | ||||||
|  |       modifications, and in Source or Object form, provided that You | ||||||
|  |       meet the following conditions: | ||||||
|  | 
 | ||||||
|  |       (a) You must give any other recipients of the Work or | ||||||
|  |           Derivative Works a copy of this License; and | ||||||
|  | 
 | ||||||
|  |       (b) You must cause any modified files to carry prominent notices | ||||||
|  |           stating that You changed the files; and | ||||||
|  | 
 | ||||||
|  |       (c) You must retain, in the Source form of any Derivative Works | ||||||
|  |           that You distribute, all copyright, patent, trademark, and | ||||||
|  |           attribution notices from the Source form of the Work, | ||||||
|  |           excluding those notices that do not pertain to any part of | ||||||
|  |           the Derivative Works; and | ||||||
|  | 
 | ||||||
|  |       (d) If the Work includes a "NOTICE" text file as part of its | ||||||
|  |           distribution, then any Derivative Works that You distribute must | ||||||
|  |           include a readable copy of the attribution notices contained | ||||||
|  |           within such NOTICE file, excluding those notices that do not | ||||||
|  |           pertain to any part of the Derivative Works, in at least one | ||||||
|  |           of the following places: within a NOTICE text file distributed | ||||||
|  |           as part of the Derivative Works; within the Source form or | ||||||
|  |           documentation, if provided along with the Derivative Works; or, | ||||||
|  |           within a display generated by the Derivative Works, if and | ||||||
|  |           wherever such third-party notices normally appear. The contents | ||||||
|  |           of the NOTICE file are for informational purposes only and | ||||||
|  |           do not modify the License. You may add Your own attribution | ||||||
|  |           notices within Derivative Works that You distribute, alongside | ||||||
|  |           or as an addendum to the NOTICE text from the Work, provided | ||||||
|  |           that such additional attribution notices cannot be construed | ||||||
|  |           as modifying the License. | ||||||
|  | 
 | ||||||
|  |       You may add Your own copyright statement to Your modifications and | ||||||
|  |       may provide additional or different license terms and conditions | ||||||
|  |       for use, reproduction, or distribution of Your modifications, or | ||||||
|  |       for any such Derivative Works as a whole, provided Your use, | ||||||
|  |       reproduction, and distribution of the Work otherwise complies with | ||||||
|  |       the conditions stated in this License. | ||||||
|  | 
 | ||||||
|  |    5. Submission of Contributions. Unless You explicitly state otherwise, | ||||||
|  |       any Contribution intentionally submitted for inclusion in the Work | ||||||
|  |       by You to the Licensor shall be under the terms and conditions of | ||||||
|  |       this License, without any additional terms or conditions. | ||||||
|  |       Notwithstanding the above, nothing herein shall supersede or modify | ||||||
|  |       the terms of any separate license agreement you may have executed | ||||||
|  |       with Licensor regarding such Contributions. | ||||||
|  | 
 | ||||||
|  |    6. Trademarks. This License does not grant permission to use the trade | ||||||
|  |       names, trademarks, service marks, or product names of the Licensor, | ||||||
|  |       except as required for reasonable and customary use in describing the | ||||||
|  |       origin of the Work and reproducing the content of the NOTICE file. | ||||||
|  | 
 | ||||||
|  |    7. Disclaimer of Warranty. Unless required by applicable law or | ||||||
|  |       agreed to in writing, Licensor provides the Work (and each | ||||||
|  |       Contributor provides its Contributions) on an "AS IS" BASIS, | ||||||
|  |       WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or | ||||||
|  |       implied, including, without limitation, any warranties or conditions | ||||||
|  |       of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A | ||||||
|  |       PARTICULAR PURPOSE. You are solely responsible for determining the | ||||||
|  |       appropriateness of using or redistributing the Work and assume any | ||||||
|  |       risks associated with Your exercise of permissions under this License. | ||||||
|  | 
 | ||||||
|  |    8. Limitation of Liability. In no event and under no legal theory, | ||||||
|  |       whether in tort (including negligence), contract, or otherwise, | ||||||
|  |       unless required by applicable law (such as deliberate and grossly | ||||||
|  |       negligent acts) or agreed to in writing, shall any Contributor be | ||||||
|  |       liable to You for damages, including any direct, indirect, special, | ||||||
|  |       incidental, or consequential damages of any character arising as a | ||||||
|  |       result of this License or out of the use or inability to use the | ||||||
|  |       Work (including but not limited to damages for loss of goodwill, | ||||||
|  |       work stoppage, computer failure or malfunction, or any and all | ||||||
|  |       other commercial damages or losses), even if such Contributor | ||||||
|  |       has been advised of the possibility of such damages. | ||||||
|  | 
 | ||||||
|  |    9. Accepting Warranty or Additional Liability. While redistributing | ||||||
|  |       the Work or Derivative Works thereof, You may choose to offer, | ||||||
|  |       and charge a fee for, acceptance of support, warranty, indemnity, | ||||||
|  |       or other liability obligations and/or rights consistent with this | ||||||
|  |       License. However, in accepting such obligations, You may act only | ||||||
|  |       on Your own behalf and on Your sole responsibility, not on behalf | ||||||
|  |       of any other Contributor, and only if You agree to indemnify, | ||||||
|  |       defend, and hold each Contributor harmless for any liability | ||||||
|  |       incurred by, or claims asserted against, such Contributor by reason | ||||||
|  |       of your accepting any such warranty or additional liability. | ||||||
|  | 
 | ||||||
|  |    END OF TERMS AND CONDITIONS | ||||||
|  | 
 | ||||||
|  |    APPENDIX: How to apply the Apache License to your work. | ||||||
|  | 
 | ||||||
|  |       To apply the Apache License to your work, attach the following | ||||||
|  |       boilerplate notice, with the fields enclosed by brackets "[]" | ||||||
|  |       replaced with your own identifying information. (Don't include | ||||||
|  |       the brackets!)  The text should be enclosed in the appropriate | ||||||
|  |       comment syntax for the file format. We also recommend that a | ||||||
|  |       file or class name and description of purpose be included on the | ||||||
|  |       same "printed page" as the copyright notice for easier | ||||||
|  |       identification within third-party archives. | ||||||
|  | 
 | ||||||
|  |    Copyright [2021-2022] [IOG] | ||||||
|  | 
 | ||||||
|  |    Licensed under the Apache License, Version 2.0 (the "License"); | ||||||
|  |    you may not use this file except in compliance with the License. | ||||||
|  |    You may obtain a copy of the License at | ||||||
|  | 
 | ||||||
|  |        http://www.apache.org/licenses/LICENSE-2.0 | ||||||
|  | 
 | ||||||
|  |    Unless required by applicable law or agreed to in writing, software | ||||||
|  |    distributed under the License is distributed on an "AS IS" BASIS, | ||||||
|  |    WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||||
|  |    See the License for the specific language governing permissions and | ||||||
|  |    limitations under the License. | ||||||
							
								
								
									
										5
									
								
								demo/credentials/faucet.sk
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/credentials/faucet.sk
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |     "type": "PaymentSigningKeyShelley_ed25519", | ||||||
|  |     "description": "Payment Signing Key", | ||||||
|  |     "cborHex": "5820a5e4238b67ebb1108c52a01ac850bbce82c915d77bad94331892f3edf612883c" | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								demo/credentials/faucet.vk
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/credentials/faucet.vk
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |     "type": "PaymentVerificationKeyShelley_ed25519", | ||||||
|  |     "description": "Payment Verification Key", | ||||||
|  |     "cborHex": "5820ce13cd433cdcb3dfb00c04e216956aeb622dcd7f282b03304d9fc9de804723b2" | ||||||
|  | } | ||||||
							
								
								
									
										
											BIN
										
									
								
								demo/devnet/byron-delegate.key
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										
											BIN
										
									
								
								demo/devnet/byron-delegate.key
									
									
									
									
									
										Normal file
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										8
									
								
								demo/devnet/byron-delegation.cert
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								demo/devnet/byron-delegation.cert
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { "omega": 0 | ||||||
|  | , "issuerPk": | ||||||
|  |     "NclXQiNNEpaaLdSxP3VFeOPIfSuFqBcNtmv8/7fftBKtgW1Aig7UqHJ/czsywkWFFVmBYPRnGjXspUl3wEMvuQ==" | ||||||
|  | , "delegatePk": | ||||||
|  |     "24ejRK+kCDs1g4f3PcodFEUFVgNFWtfmuoEtVQf8/Ii2j2ruXHebJmZZPrwtAdbJYwDiSEvsHr95+BAF1ifGsA==" | ||||||
|  | , "cert": | ||||||
|  |     "498c72e35ef30cd4657b48bfcc0a84a555a67981e3b6104a0d1708ab84510367d81e1ba3f47619565b1ee1098e31dcb8eb648d8030e061b568de113fdf3d6a09" | ||||||
|  | } | ||||||
							
								
								
									
										79
									
								
								demo/devnet/cardano-node.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										79
									
								
								demo/devnet/cardano-node.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,79 @@ | |||||||
|  | { | ||||||
|  |   "Protocol": "Cardano", | ||||||
|  | 
 | ||||||
|  |   "ByronGenesisFile": "genesis-byron.json", | ||||||
|  |   "ShelleyGenesisFile": "genesis-shelley.json", | ||||||
|  |   "AlonzoGenesisFile": "genesis-alonzo.json", | ||||||
|  | 
 | ||||||
|  |   "ApplicationName": "cardano-sl", | ||||||
|  |   "ApplicationVersion": 1, | ||||||
|  |   "MaxKnownMajorProtocolVersion": 2, | ||||||
|  |   "LastKnownBlockVersion-Alt": 0, | ||||||
|  |   "LastKnownBlockVersion-Major": 6, | ||||||
|  |   "LastKnownBlockVersion-Minor": 0, | ||||||
|  | 
 | ||||||
|  |   "TestShelleyHardForkAtEpoch": 0, | ||||||
|  |   "TestAllegraHardForkAtEpoch": 0, | ||||||
|  |   "TestMaryHardForkAtEpoch": 0, | ||||||
|  |   "TestAlonzoHardForkAtEpoch": 0, | ||||||
|  |   "TestBabbageHardForkAtEpoch": 0, | ||||||
|  | 
 | ||||||
|  |   "RequiresNetworkMagic": "RequiresNoMagic", | ||||||
|  | 
 | ||||||
|  |   "minSeverity": "Info", | ||||||
|  |   "defaultBackends": ["KatipBK"], | ||||||
|  |   "defaultScribes": [[ "StdoutSK", "stdout" ]], | ||||||
|  |   "setupBackends": ["KatipBK"], | ||||||
|  |   "setupScribes": [ | ||||||
|  |     { | ||||||
|  |       "scFormat": "ScJson", | ||||||
|  |       "scKind": "StdoutSK", | ||||||
|  |       "scName": "stdout", | ||||||
|  |       "scRotation": null | ||||||
|  |     } | ||||||
|  |   ], | ||||||
|  | 
 | ||||||
|  |   "TurnOnLogMetrics": true, | ||||||
|  |   "TurnOnLogging": true, | ||||||
|  | 
 | ||||||
|  |   "TracingVerbosity": "NormalVerbosity", | ||||||
|  |   "TraceBlockFetchClient": false, | ||||||
|  |   "TraceBlockFetchDecisions": false, | ||||||
|  |   "TraceBlockFetchProtocol": false, | ||||||
|  |   "TraceBlockFetchProtocolSerialised": false, | ||||||
|  |   "TraceBlockFetchServer": false, | ||||||
|  |   "TraceChainDb": true, | ||||||
|  |   "TraceChainSyncBlockServer": false, | ||||||
|  |   "TraceChainSyncClient": false, | ||||||
|  |   "TraceChainSyncHeaderServer": false, | ||||||
|  |   "TraceChainSyncProtocol": false, | ||||||
|  |   "TraceDNSResolver": false, | ||||||
|  |   "TraceDNSSubscription": false, | ||||||
|  |   "TraceErrorPolicy": false, | ||||||
|  |   "TraceForge": true, | ||||||
|  |   "TraceHandshake": false, | ||||||
|  |   "TraceIpSubscription": false, | ||||||
|  |   "TraceLocalChainSyncProtocol": true, | ||||||
|  |   "TraceLocalErrorPolicy": false, | ||||||
|  |   "TraceLocalHandshake": false, | ||||||
|  |   "TraceLocalTxSubmissionProtocol": true, | ||||||
|  |   "TraceLocalTxSubmissionServer": true, | ||||||
|  |   "TraceMempool": true, | ||||||
|  |   "TraceMux": false, | ||||||
|  |   "TraceTxInbound": false, | ||||||
|  |   "TraceTxOutbound": false, | ||||||
|  |   "TraceTxSubmissionProtocol": false, | ||||||
|  | 
 | ||||||
|  |   "options": { | ||||||
|  |     "mapBackends": { | ||||||
|  |       "cardano.node.metrics": [ "EKGViewBK" ], | ||||||
|  |       "cardano.node.resources": [ "EKGViewBK" ] | ||||||
|  |     }, | ||||||
|  |     "mapSubtrace": { | ||||||
|  |       "cardano.node.metrics": { "subtrace": "Neutral" } | ||||||
|  |     } | ||||||
|  |   }, | ||||||
|  | 
 | ||||||
|  |   "TestEnableDevelopmentHardForkEras": true, | ||||||
|  |   "TestEnableDevelopmentNetworkProtocols": true | ||||||
|  | } | ||||||
							
								
								
									
										365
									
								
								demo/devnet/genesis-alonzo.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										365
									
								
								demo/devnet/genesis-alonzo.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,365 @@ | |||||||
|  | { | ||||||
|  |     "collateralPercentage": 150, | ||||||
|  |     "costModels": { | ||||||
|  |         "PlutusV1": { | ||||||
|  |             "addInteger-cpu-arguments-intercept": 197209, | ||||||
|  |             "addInteger-cpu-arguments-slope": 0, | ||||||
|  |             "addInteger-memory-arguments-intercept": 1, | ||||||
|  |             "addInteger-memory-arguments-slope": 1, | ||||||
|  |             "appendByteString-cpu-arguments-intercept": 396231, | ||||||
|  |             "appendByteString-cpu-arguments-slope": 621, | ||||||
|  |             "appendByteString-memory-arguments-intercept": 0, | ||||||
|  |             "appendByteString-memory-arguments-slope": 1, | ||||||
|  |             "appendString-cpu-arguments-intercept": 150000, | ||||||
|  |             "appendString-cpu-arguments-slope": 1000, | ||||||
|  |             "appendString-memory-arguments-intercept": 0, | ||||||
|  |             "appendString-memory-arguments-slope": 1, | ||||||
|  |             "bData-cpu-arguments": 150000, | ||||||
|  |             "bData-memory-arguments": 32, | ||||||
|  |             "blake2b_256-cpu-arguments-intercept": 2477736, | ||||||
|  |             "blake2b_256-cpu-arguments-slope": 29175, | ||||||
|  |             "blake2b_256-memory-arguments": 4, | ||||||
|  |             "cekApplyCost-exBudgetCPU": 29773, | ||||||
|  |             "cekApplyCost-exBudgetMemory": 100, | ||||||
|  |             "cekBuiltinCost-exBudgetCPU": 29773, | ||||||
|  |             "cekBuiltinCost-exBudgetMemory": 100, | ||||||
|  |             "cekConstCost-exBudgetCPU": 29773, | ||||||
|  |             "cekConstCost-exBudgetMemory": 100, | ||||||
|  |             "cekDelayCost-exBudgetCPU": 29773, | ||||||
|  |             "cekDelayCost-exBudgetMemory": 100, | ||||||
|  |             "cekForceCost-exBudgetCPU": 29773, | ||||||
|  |             "cekForceCost-exBudgetMemory": 100, | ||||||
|  |             "cekLamCost-exBudgetCPU": 29773, | ||||||
|  |             "cekLamCost-exBudgetMemory": 100, | ||||||
|  |             "cekStartupCost-exBudgetCPU": 100, | ||||||
|  |             "cekStartupCost-exBudgetMemory": 100, | ||||||
|  |             "cekVarCost-exBudgetCPU": 29773, | ||||||
|  |             "cekVarCost-exBudgetMemory": 100, | ||||||
|  |             "chooseData-cpu-arguments": 150000, | ||||||
|  |             "chooseData-memory-arguments": 32, | ||||||
|  |             "chooseList-cpu-arguments": 150000, | ||||||
|  |             "chooseList-memory-arguments": 32, | ||||||
|  |             "chooseUnit-cpu-arguments": 150000, | ||||||
|  |             "chooseUnit-memory-arguments": 32, | ||||||
|  |             "consByteString-cpu-arguments-intercept": 150000, | ||||||
|  |             "consByteString-cpu-arguments-slope": 1000, | ||||||
|  |             "consByteString-memory-arguments-intercept": 0, | ||||||
|  |             "consByteString-memory-arguments-slope": 1, | ||||||
|  |             "constrData-cpu-arguments": 150000, | ||||||
|  |             "constrData-memory-arguments": 32, | ||||||
|  |             "decodeUtf8-cpu-arguments-intercept": 150000, | ||||||
|  |             "decodeUtf8-cpu-arguments-slope": 1000, | ||||||
|  |             "decodeUtf8-memory-arguments-intercept": 0, | ||||||
|  |             "decodeUtf8-memory-arguments-slope": 8, | ||||||
|  |             "divideInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "divideInteger-memory-arguments-intercept": 0, | ||||||
|  |             "divideInteger-memory-arguments-minimum": 1, | ||||||
|  |             "divideInteger-memory-arguments-slope": 1, | ||||||
|  |             "encodeUtf8-cpu-arguments-intercept": 150000, | ||||||
|  |             "encodeUtf8-cpu-arguments-slope": 1000, | ||||||
|  |             "encodeUtf8-memory-arguments-intercept": 0, | ||||||
|  |             "encodeUtf8-memory-arguments-slope": 8, | ||||||
|  |             "equalsByteString-cpu-arguments-constant": 150000, | ||||||
|  |             "equalsByteString-cpu-arguments-intercept": 112536, | ||||||
|  |             "equalsByteString-cpu-arguments-slope": 247, | ||||||
|  |             "equalsByteString-memory-arguments": 1, | ||||||
|  |             "equalsData-cpu-arguments-intercept": 150000, | ||||||
|  |             "equalsData-cpu-arguments-slope": 10000, | ||||||
|  |             "equalsData-memory-arguments": 1, | ||||||
|  |             "equalsInteger-cpu-arguments-intercept": 136542, | ||||||
|  |             "equalsInteger-cpu-arguments-slope": 1326, | ||||||
|  |             "equalsInteger-memory-arguments": 1, | ||||||
|  |             "equalsString-cpu-arguments-constant": 1000, | ||||||
|  |             "equalsString-cpu-arguments-intercept": 150000, | ||||||
|  |             "equalsString-cpu-arguments-slope": 1000, | ||||||
|  |             "equalsString-memory-arguments": 1, | ||||||
|  |             "fstPair-cpu-arguments": 150000, | ||||||
|  |             "fstPair-memory-arguments": 32, | ||||||
|  |             "headList-cpu-arguments": 150000, | ||||||
|  |             "headList-memory-arguments": 32, | ||||||
|  |             "iData-cpu-arguments": 150000, | ||||||
|  |             "iData-memory-arguments": 32, | ||||||
|  |             "ifThenElse-cpu-arguments": 1, | ||||||
|  |             "ifThenElse-memory-arguments": 1, | ||||||
|  |             "indexByteString-cpu-arguments": 150000, | ||||||
|  |             "indexByteString-memory-arguments": 1, | ||||||
|  |             "lengthOfByteString-cpu-arguments": 150000, | ||||||
|  |             "lengthOfByteString-memory-arguments": 4, | ||||||
|  |             "lessThanByteString-cpu-arguments-intercept": 103599, | ||||||
|  |             "lessThanByteString-cpu-arguments-slope": 248, | ||||||
|  |             "lessThanByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-intercept": 103599, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-slope": 248, | ||||||
|  |             "lessThanEqualsByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-intercept": 145276, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-slope": 1366, | ||||||
|  |             "lessThanEqualsInteger-memory-arguments": 1, | ||||||
|  |             "lessThanInteger-cpu-arguments-intercept": 179690, | ||||||
|  |             "lessThanInteger-cpu-arguments-slope": 497, | ||||||
|  |             "lessThanInteger-memory-arguments": 1, | ||||||
|  |             "listData-cpu-arguments": 150000, | ||||||
|  |             "listData-memory-arguments": 32, | ||||||
|  |             "mapData-cpu-arguments": 150000, | ||||||
|  |             "mapData-memory-arguments": 32, | ||||||
|  |             "mkCons-cpu-arguments": 150000, | ||||||
|  |             "mkCons-memory-arguments": 32, | ||||||
|  |             "mkNilData-cpu-arguments": 150000, | ||||||
|  |             "mkNilData-memory-arguments": 32, | ||||||
|  |             "mkNilPairData-cpu-arguments": 150000, | ||||||
|  |             "mkNilPairData-memory-arguments": 32, | ||||||
|  |             "mkPairData-cpu-arguments": 150000, | ||||||
|  |             "mkPairData-memory-arguments": 32, | ||||||
|  |             "modInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "modInteger-memory-arguments-intercept": 0, | ||||||
|  |             "modInteger-memory-arguments-minimum": 1, | ||||||
|  |             "modInteger-memory-arguments-slope": 1, | ||||||
|  |             "multiplyInteger-cpu-arguments-intercept": 61516, | ||||||
|  |             "multiplyInteger-cpu-arguments-slope": 11218, | ||||||
|  |             "multiplyInteger-memory-arguments-intercept": 0, | ||||||
|  |             "multiplyInteger-memory-arguments-slope": 1, | ||||||
|  |             "nullList-cpu-arguments": 150000, | ||||||
|  |             "nullList-memory-arguments": 32, | ||||||
|  |             "quotientInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "quotientInteger-memory-arguments-intercept": 0, | ||||||
|  |             "quotientInteger-memory-arguments-minimum": 1, | ||||||
|  |             "quotientInteger-memory-arguments-slope": 1, | ||||||
|  |             "remainderInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "remainderInteger-memory-arguments-intercept": 0, | ||||||
|  |             "remainderInteger-memory-arguments-minimum": 1, | ||||||
|  |             "remainderInteger-memory-arguments-slope": 1, | ||||||
|  |             "sha2_256-cpu-arguments-intercept": 2477736, | ||||||
|  |             "sha2_256-cpu-arguments-slope": 29175, | ||||||
|  |             "sha2_256-memory-arguments": 4, | ||||||
|  |             "sha3_256-cpu-arguments-intercept": 0, | ||||||
|  |             "sha3_256-cpu-arguments-slope": 82363, | ||||||
|  |             "sha3_256-memory-arguments": 4, | ||||||
|  |             "sliceByteString-cpu-arguments-intercept": 150000, | ||||||
|  |             "sliceByteString-cpu-arguments-slope": 5000, | ||||||
|  |             "sliceByteString-memory-arguments-intercept": 0, | ||||||
|  |             "sliceByteString-memory-arguments-slope": 1, | ||||||
|  |             "sndPair-cpu-arguments": 150000, | ||||||
|  |             "sndPair-memory-arguments": 32, | ||||||
|  |             "subtractInteger-cpu-arguments-intercept": 197209, | ||||||
|  |             "subtractInteger-cpu-arguments-slope": 0, | ||||||
|  |             "subtractInteger-memory-arguments-intercept": 1, | ||||||
|  |             "subtractInteger-memory-arguments-slope": 1, | ||||||
|  |             "tailList-cpu-arguments": 150000, | ||||||
|  |             "tailList-memory-arguments": 32, | ||||||
|  |             "trace-cpu-arguments": 150000, | ||||||
|  |             "trace-memory-arguments": 32, | ||||||
|  |             "unBData-cpu-arguments": 150000, | ||||||
|  |             "unBData-memory-arguments": 32, | ||||||
|  |             "unConstrData-cpu-arguments": 150000, | ||||||
|  |             "unConstrData-memory-arguments": 32, | ||||||
|  |             "unIData-cpu-arguments": 150000, | ||||||
|  |             "unIData-memory-arguments": 32, | ||||||
|  |             "unListData-cpu-arguments": 150000, | ||||||
|  |             "unListData-memory-arguments": 32, | ||||||
|  |             "unMapData-cpu-arguments": 150000, | ||||||
|  |             "unMapData-memory-arguments": 32, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-intercept": 3345831, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-slope": 1, | ||||||
|  |             "verifyEd25519Signature-memory-arguments": 1 | ||||||
|  |         }, | ||||||
|  |         "PlutusV2": { | ||||||
|  |             "addInteger-cpu-arguments-intercept": 205665, | ||||||
|  |             "addInteger-cpu-arguments-slope": 812, | ||||||
|  |             "addInteger-memory-arguments-intercept": 1, | ||||||
|  |             "addInteger-memory-arguments-slope": 1, | ||||||
|  |             "appendByteString-cpu-arguments-intercept": 1000, | ||||||
|  |             "appendByteString-cpu-arguments-slope": 571, | ||||||
|  |             "appendByteString-memory-arguments-intercept": 0, | ||||||
|  |             "appendByteString-memory-arguments-slope": 1, | ||||||
|  |             "appendString-cpu-arguments-intercept": 1000, | ||||||
|  |             "appendString-cpu-arguments-slope": 24177, | ||||||
|  |             "appendString-memory-arguments-intercept": 4, | ||||||
|  |             "appendString-memory-arguments-slope": 1, | ||||||
|  |             "bData-cpu-arguments": 1000, | ||||||
|  |             "bData-memory-arguments": 32, | ||||||
|  |             "blake2b_256-cpu-arguments-intercept": 117366, | ||||||
|  |             "blake2b_256-cpu-arguments-slope": 10475, | ||||||
|  |             "blake2b_256-memory-arguments": 4, | ||||||
|  |             "cekApplyCost-exBudgetCPU": 23000, | ||||||
|  |             "cekApplyCost-exBudgetMemory": 100, | ||||||
|  |             "cekBuiltinCost-exBudgetCPU": 23000, | ||||||
|  |             "cekBuiltinCost-exBudgetMemory": 100, | ||||||
|  |             "cekConstCost-exBudgetCPU": 23000, | ||||||
|  |             "cekConstCost-exBudgetMemory": 100, | ||||||
|  |             "cekDelayCost-exBudgetCPU": 23000, | ||||||
|  |             "cekDelayCost-exBudgetMemory": 100, | ||||||
|  |             "cekForceCost-exBudgetCPU": 23000, | ||||||
|  |             "cekForceCost-exBudgetMemory": 100, | ||||||
|  |             "cekLamCost-exBudgetCPU": 23000, | ||||||
|  |             "cekLamCost-exBudgetMemory": 100, | ||||||
|  |             "cekStartupCost-exBudgetCPU": 100, | ||||||
|  |             "cekStartupCost-exBudgetMemory": 100, | ||||||
|  |             "cekVarCost-exBudgetCPU": 23000, | ||||||
|  |             "cekVarCost-exBudgetMemory": 100, | ||||||
|  |             "chooseData-cpu-arguments": 19537, | ||||||
|  |             "chooseData-memory-arguments": 32, | ||||||
|  |             "chooseList-cpu-arguments": 175354, | ||||||
|  |             "chooseList-memory-arguments": 32, | ||||||
|  |             "chooseUnit-cpu-arguments": 46417, | ||||||
|  |             "chooseUnit-memory-arguments": 4, | ||||||
|  |             "consByteString-cpu-arguments-intercept": 221973, | ||||||
|  |             "consByteString-cpu-arguments-slope": 511, | ||||||
|  |             "consByteString-memory-arguments-intercept": 0, | ||||||
|  |             "consByteString-memory-arguments-slope": 1, | ||||||
|  |             "constrData-cpu-arguments": 89141, | ||||||
|  |             "constrData-memory-arguments": 32, | ||||||
|  |             "decodeUtf8-cpu-arguments-intercept": 497525, | ||||||
|  |             "decodeUtf8-cpu-arguments-slope": 14068, | ||||||
|  |             "decodeUtf8-memory-arguments-intercept": 4, | ||||||
|  |             "decodeUtf8-memory-arguments-slope": 2, | ||||||
|  |             "divideInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "divideInteger-memory-arguments-intercept": 0, | ||||||
|  |             "divideInteger-memory-arguments-minimum": 1, | ||||||
|  |             "divideInteger-memory-arguments-slope": 1, | ||||||
|  |             "encodeUtf8-cpu-arguments-intercept": 1000, | ||||||
|  |             "encodeUtf8-cpu-arguments-slope": 28662, | ||||||
|  |             "encodeUtf8-memory-arguments-intercept": 4, | ||||||
|  |             "encodeUtf8-memory-arguments-slope": 2, | ||||||
|  |             "equalsByteString-cpu-arguments-constant": 245000, | ||||||
|  |             "equalsByteString-cpu-arguments-intercept": 216773, | ||||||
|  |             "equalsByteString-cpu-arguments-slope": 62, | ||||||
|  |             "equalsByteString-memory-arguments": 1, | ||||||
|  |             "equalsData-cpu-arguments-intercept": 1060367, | ||||||
|  |             "equalsData-cpu-arguments-slope": 12586, | ||||||
|  |             "equalsData-memory-arguments": 1, | ||||||
|  |             "equalsInteger-cpu-arguments-intercept": 208512, | ||||||
|  |             "equalsInteger-cpu-arguments-slope": 421, | ||||||
|  |             "equalsInteger-memory-arguments": 1, | ||||||
|  |             "equalsString-cpu-arguments-constant": 187000, | ||||||
|  |             "equalsString-cpu-arguments-intercept": 1000, | ||||||
|  |             "equalsString-cpu-arguments-slope": 52998, | ||||||
|  |             "equalsString-memory-arguments": 1, | ||||||
|  |             "fstPair-cpu-arguments": 80436, | ||||||
|  |             "fstPair-memory-arguments": 32, | ||||||
|  |             "headList-cpu-arguments": 43249, | ||||||
|  |             "headList-memory-arguments": 32, | ||||||
|  |             "iData-cpu-arguments": 1000, | ||||||
|  |             "iData-memory-arguments": 32, | ||||||
|  |             "ifThenElse-cpu-arguments": 80556, | ||||||
|  |             "ifThenElse-memory-arguments": 1, | ||||||
|  |             "indexByteString-cpu-arguments": 57667, | ||||||
|  |             "indexByteString-memory-arguments": 4, | ||||||
|  |             "lengthOfByteString-cpu-arguments": 1000, | ||||||
|  |             "lengthOfByteString-memory-arguments": 10, | ||||||
|  |             "lessThanByteString-cpu-arguments-intercept": 197145, | ||||||
|  |             "lessThanByteString-cpu-arguments-slope": 156, | ||||||
|  |             "lessThanByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-intercept": 197145, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-slope": 156, | ||||||
|  |             "lessThanEqualsByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-intercept": 204924, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-slope": 473, | ||||||
|  |             "lessThanEqualsInteger-memory-arguments": 1, | ||||||
|  |             "lessThanInteger-cpu-arguments-intercept": 208896, | ||||||
|  |             "lessThanInteger-cpu-arguments-slope": 511, | ||||||
|  |             "lessThanInteger-memory-arguments": 1, | ||||||
|  |             "listData-cpu-arguments": 52467, | ||||||
|  |             "listData-memory-arguments": 32, | ||||||
|  |             "mapData-cpu-arguments": 64832, | ||||||
|  |             "mapData-memory-arguments": 32, | ||||||
|  |             "mkCons-cpu-arguments": 65493, | ||||||
|  |             "mkCons-memory-arguments": 32, | ||||||
|  |             "mkNilData-cpu-arguments": 22558, | ||||||
|  |             "mkNilData-memory-arguments": 32, | ||||||
|  |             "mkNilPairData-cpu-arguments": 16563, | ||||||
|  |             "mkNilPairData-memory-arguments": 32, | ||||||
|  |             "mkPairData-cpu-arguments": 76511, | ||||||
|  |             "mkPairData-memory-arguments": 32, | ||||||
|  |             "modInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "modInteger-memory-arguments-intercept": 0, | ||||||
|  |             "modInteger-memory-arguments-minimum": 1, | ||||||
|  |             "modInteger-memory-arguments-slope": 1, | ||||||
|  |             "multiplyInteger-cpu-arguments-intercept": 69522, | ||||||
|  |             "multiplyInteger-cpu-arguments-slope": 11687, | ||||||
|  |             "multiplyInteger-memory-arguments-intercept": 0, | ||||||
|  |             "multiplyInteger-memory-arguments-slope": 1, | ||||||
|  |             "nullList-cpu-arguments": 60091, | ||||||
|  |             "nullList-memory-arguments": 32, | ||||||
|  |             "quotientInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "quotientInteger-memory-arguments-intercept": 0, | ||||||
|  |             "quotientInteger-memory-arguments-minimum": 1, | ||||||
|  |             "quotientInteger-memory-arguments-slope": 1, | ||||||
|  |             "remainderInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "remainderInteger-memory-arguments-intercept": 0, | ||||||
|  |             "remainderInteger-memory-arguments-minimum": 1, | ||||||
|  |             "remainderInteger-memory-arguments-slope": 1, | ||||||
|  |             "serialiseData-cpu-arguments-intercept": 1159724, | ||||||
|  |             "serialiseData-cpu-arguments-slope": 392670, | ||||||
|  |             "serialiseData-memory-arguments-intercept": 0, | ||||||
|  |             "serialiseData-memory-arguments-slope": 2, | ||||||
|  |             "sha2_256-cpu-arguments-intercept": 806990, | ||||||
|  |             "sha2_256-cpu-arguments-slope": 30482, | ||||||
|  |             "sha2_256-memory-arguments": 4, | ||||||
|  |             "sha3_256-cpu-arguments-intercept": 1927926, | ||||||
|  |             "sha3_256-cpu-arguments-slope": 82523, | ||||||
|  |             "sha3_256-memory-arguments": 4, | ||||||
|  |             "sliceByteString-cpu-arguments-intercept": 265318, | ||||||
|  |             "sliceByteString-cpu-arguments-slope": 0, | ||||||
|  |             "sliceByteString-memory-arguments-intercept": 4, | ||||||
|  |             "sliceByteString-memory-arguments-slope": 0, | ||||||
|  |             "sndPair-cpu-arguments": 85931, | ||||||
|  |             "sndPair-memory-arguments": 32, | ||||||
|  |             "subtractInteger-cpu-arguments-intercept": 205665, | ||||||
|  |             "subtractInteger-cpu-arguments-slope": 812, | ||||||
|  |             "subtractInteger-memory-arguments-intercept": 1, | ||||||
|  |             "subtractInteger-memory-arguments-slope": 1, | ||||||
|  |             "tailList-cpu-arguments": 41182, | ||||||
|  |             "tailList-memory-arguments": 32, | ||||||
|  |             "trace-cpu-arguments": 212342, | ||||||
|  |             "trace-memory-arguments": 32, | ||||||
|  |             "unBData-cpu-arguments": 31220, | ||||||
|  |             "unBData-memory-arguments": 32, | ||||||
|  |             "unConstrData-cpu-arguments": 32696, | ||||||
|  |             "unConstrData-memory-arguments": 32, | ||||||
|  |             "unIData-cpu-arguments": 43357, | ||||||
|  |             "unIData-memory-arguments": 32, | ||||||
|  |             "unListData-cpu-arguments": 32247, | ||||||
|  |             "unListData-memory-arguments": 32, | ||||||
|  |             "unMapData-cpu-arguments": 38314, | ||||||
|  |             "unMapData-memory-arguments": 32, | ||||||
|  |             "verifyEcdsaSecp256k1Signature-cpu-arguments": 35892428, | ||||||
|  |             "verifyEcdsaSecp256k1Signature-memory-arguments": 10, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-intercept": 9462713, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-slope": 1021, | ||||||
|  |             "verifyEd25519Signature-memory-arguments": 10, | ||||||
|  |             "verifySchnorrSecp256k1Signature-cpu-arguments-intercept": 38887044, | ||||||
|  |             "verifySchnorrSecp256k1Signature-cpu-arguments-slope": 32947, | ||||||
|  |             "verifySchnorrSecp256k1Signature-memory-arguments": 10 | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     "executionPrices": { | ||||||
|  |         "prMem": 5.77e-2, | ||||||
|  |         "prSteps": 7.21e-5 | ||||||
|  |     }, | ||||||
|  |     "lovelacePerUTxOWord": 34482, | ||||||
|  |     "maxBlockExUnits": { | ||||||
|  |         "exUnitsMem": 80000000, | ||||||
|  |         "exUnitsSteps": 40000000000 | ||||||
|  |     }, | ||||||
|  |     "maxCollateralInputs": 3, | ||||||
|  |     "maxTxExUnits": { | ||||||
|  |         "exUnitsMem": 16000000, | ||||||
|  |         "exUnitsSteps": 10000000000 | ||||||
|  |     }, | ||||||
|  |     "maxValueSize": 5000 | ||||||
|  | } | ||||||
							
								
								
									
										36
									
								
								demo/devnet/genesis-byron.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										36
									
								
								demo/devnet/genesis-byron.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,36 @@ | |||||||
|  | { | ||||||
|  |     "protocolConsts": { | ||||||
|  |         "k": 2160, | ||||||
|  |         "protocolMagic": 42 | ||||||
|  |     }, | ||||||
|  |     "startTime": 1657186415, | ||||||
|  |     "blockVersionData": { | ||||||
|  |         "scriptVersion": 0, | ||||||
|  |         "slotDuration": "250", | ||||||
|  |         "maxBlockSize": "2000000", | ||||||
|  |         "maxHeaderSize": "2000000", | ||||||
|  |         "maxTxSize": "4096", | ||||||
|  |         "maxProposalSize": "700", | ||||||
|  |         "mpcThd": "20000000000000", | ||||||
|  |         "heavyDelThd": "300000000000", | ||||||
|  |         "updateVoteThd": "1000000000000", | ||||||
|  |         "updateProposalThd": "100000000000000", | ||||||
|  |         "updateImplicit": "10000", | ||||||
|  |         "softforkRule": { | ||||||
|  |             "initThd": "900000000000000", | ||||||
|  |             "minThd": "600000000000000", | ||||||
|  |             "thdDecrement": "50000000000000" | ||||||
|  |         }, | ||||||
|  |         "txFeePolicy": { | ||||||
|  |             "summand": "155381000000000", | ||||||
|  |             "multiplier": "43000000000" | ||||||
|  |         }, | ||||||
|  |         "unlockStakeEpoch": "18446744073709551615" | ||||||
|  |     }, | ||||||
|  |     "bootStakeholders": { | ||||||
|  |         "7a4519c93d7be4577dd85bd524c644e6b809e44eae0457b43128c1c7": 1 | ||||||
|  |     }, | ||||||
|  |     "heavyDelegation": {}, | ||||||
|  |     "nonAvvmBalances": {}, | ||||||
|  |     "avvmDistr": {} | ||||||
|  | } | ||||||
							
								
								
									
										65
									
								
								demo/devnet/genesis-shelley.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										65
									
								
								demo/devnet/genesis-shelley.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,65 @@ | |||||||
|  | { | ||||||
|  |     "epochLength": 432000, | ||||||
|  |     "activeSlotsCoeff": 1.0, | ||||||
|  |     "slotLength": 0.1, | ||||||
|  |     "securityParam": 2160, | ||||||
|  |     "genDelegs": {}, | ||||||
|  |     "initialFunds": { | ||||||
|  |         "00813c32c92aad21770ff8001de0918f598df8c06775f77f8e8839d2a0074a515f7f32bf31a4f41c7417a8136e8152bfb42f06d71b389a6896": 900000000000, | ||||||
|  |         "609783be7d3c54f11377966dfabc9284cd6c32fca1cd42ef0a4f1cc45b": 900000000000 | ||||||
|  |     }, | ||||||
|  |     "maxKESEvolutions": 60, | ||||||
|  |     "maxLovelaceSupply": 2000000000000, | ||||||
|  |     "networkId": "Testnet", | ||||||
|  |     "networkMagic": 42, | ||||||
|  |     "protocolParams": { | ||||||
|  |         "a0": 0.0, | ||||||
|  |         "decentralisationParam": 0, | ||||||
|  |         "eMax": 18, | ||||||
|  |         "extraEntropy": { | ||||||
|  |             "tag": "NeutralNonce" | ||||||
|  |         }, | ||||||
|  |         "keyDeposit": 0, | ||||||
|  |         "maxBlockBodySize": 65536, | ||||||
|  |         "maxBlockHeaderSize": 1100, | ||||||
|  |         "maxTxSize": 16384, | ||||||
|  |         "minFeeA": 44, | ||||||
|  |         "minFeeB": 155381, | ||||||
|  |         "minPoolCost": 0, | ||||||
|  |         "minUTxOValue": 0, | ||||||
|  |         "nOpt": 100, | ||||||
|  |         "poolDeposit": 0, | ||||||
|  |         "protocolVersion": { | ||||||
|  |             "major": 7, | ||||||
|  |             "minor": 0 | ||||||
|  |         }, | ||||||
|  |         "rho": 0.1, | ||||||
|  |         "tau": 0.1 | ||||||
|  |     }, | ||||||
|  |     "slotsPerKESPeriod": 129600, | ||||||
|  |     "staking": { | ||||||
|  |         "pools": { | ||||||
|  |             "8a219b698d3b6e034391ae84cee62f1d76b6fbc45ddfe4e31e0d4b60": { | ||||||
|  |                 "cost": 0, | ||||||
|  |                 "margin": 0.0, | ||||||
|  |                 "metadata": null, | ||||||
|  |                 "owners": [], | ||||||
|  |                 "pledge": 0, | ||||||
|  |                 "publicKey": "8a219b698d3b6e034391ae84cee62f1d76b6fbc45ddfe4e31e0d4b60", | ||||||
|  |                 "relays": [], | ||||||
|  |                 "rewardAccount": { | ||||||
|  |                     "credential": { | ||||||
|  |                         "key hash": "b6ffb20cf821f9286802235841d4348a2c2bafd4f73092b7de6655ea" | ||||||
|  |                     }, | ||||||
|  |                     "network": "Testnet" | ||||||
|  |                 }, | ||||||
|  |                 "vrf": "fec17ed60cbf2ec5be3f061fb4de0b6ef1f20947cfbfce5fb2783d12f3f69ff5" | ||||||
|  |             } | ||||||
|  |         }, | ||||||
|  |         "stake": { | ||||||
|  |             "074a515f7f32bf31a4f41c7417a8136e8152bfb42f06d71b389a6896": "8a219b698d3b6e034391ae84cee62f1d76b6fbc45ddfe4e31e0d4b60" | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     "systemStart": "2022-07-07T09:33:35Z", | ||||||
|  |     "updateQuorum": 2 | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								demo/devnet/kes.skey
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/devnet/kes.skey
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |     "type": "KesSigningKey_ed25519_kes_2^6", | ||||||
|  |     "description": "KES Signing Key", | ||||||
|  |     "cborHex": "590260a199f16b11da6c7f5c1e0f1eb0b9bbe278d3d8f35bfd50d0951c2ff94d0344cd57df5f64c9bac1dd60b4482f9c636168f40737d526625a2ec82f22ec0c72de0013f86ef743a7bba0286db6ddf3d85bf8e49ddbf14d9d3b7ee22f4857c77b740948f84f2e72f6bcf91f405e34ea50a2c53fa4876b43cfce2bcfe87c06a903de8bb33d968ca7930b67d0c23f5cb2d74e422d773ba80e388de384691000d6ba8a9b4dc7d3187f76048fbef9a52b72d80d835bb76eced7c0e0cdc5b58869b73c095dffa01db4ff51765afcead565395a5ed1cf74e5f2134d61076fece21aacd080bbbfaab94125401d7bbc74eafc7e7e3a2235f59dc03d6e332e53d558493a1e22213b92c77b1328ff1b83855da704fc366bf4415490602481d1939136eeaf252c65184912a779d9d94a90e32b72c1877ef60b6d79e707ce5a762acb4bed46436efe4fe62aae50b39068cc508a09427c92791cbcbea44318529cc68d297ca24e1b73b2394c385ec63fcd85ed56eec3de48860a1ec950aad4f91cbf741dbd7bf1d3c278875bd20e31ff5372339f6aa5280ad9b8bf3514889ac44600fe57ca0b535d6dc6b0b981e079595aad186ee0be9b07e837391ab165e4ca406601c876a86e246a3f53311e21199cccc0b080f28d18f4dc6987731e10e4ade00df7c6921c5ef3022b6f49a29ba307a2c8f4bd2ba42fcfa0aad68a2f0ad31fff69a99d3471f9036d3f5817a3edfeff7fc3c14e1151d767aaa043481cfd1a6ee55e8e5d7853ecdaf9da2bb36c716beae8d706bc648a790d4697e1d044a11a49f305ab8bc64a094bd81bda7395fe6f77dd5557c39919dd9bb9cf22a87fe47408ae3ec2247007d015a5" | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								demo/devnet/opcert.cert
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/devnet/opcert.cert
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |     "type": "NodeOperationalCertificate", | ||||||
|  |     "description": "", | ||||||
|  |     "cborHex": "828458204cd49bb05e9885142fe7af1481107995298771fd1a24e72b506a4d600ee2b3120000584089fc9e9f551b2ea873bf31643659d049152d5c8e8de86be4056370bccc5fa62dd12e3f152f1664e614763e46eaa7a17ed366b5cef19958773d1ab96941442e0b58205a3d778e76741a009e29d23093cfe046131808d34d7c864967b515e98dfc3583" | ||||||
|  | } | ||||||
							
								
								
									
										5
									
								
								demo/devnet/vrf.skey
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								demo/devnet/vrf.skey
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | { | ||||||
|  |     "type": "VrfSigningKey_PraosVRF", | ||||||
|  |     "description": "VRF Signing Key", | ||||||
|  |     "cborHex": "5840899795b70e9f34b737159fe21a6170568d6031e187f0cc84555c712b7c29b45cb882007593ef70f86e5c0948561a3b8e8851529a4f98975f2b24e768dda38ce2" | ||||||
|  | } | ||||||
							
								
								
									
										21
									
								
								demo/prepare-devnet.sh
									
									
									
									
									
										Executable file
									
								
							
							
						
						
									
										21
									
								
								demo/prepare-devnet.sh
									
									
									
									
									
										Executable file
									
								
							| @ -0,0 +1,21 @@ | |||||||
|  | #!/usr/bin/env bash | ||||||
|  | 
 | ||||||
|  | # Prepare a "devnet" directory holding credentials, a dummy topology and | ||||||
|  | # "up-to-date" genesis files. If the directory exists, it is wiped out. | ||||||
|  | set -e | ||||||
|  | 
 | ||||||
|  | BASEDIR=$(dirname $(realpath $0)) | ||||||
|  | TARGETDIR="${BASEDIR}/../devnet" | ||||||
|  | 
 | ||||||
|  | [ -d "$TARGETDIR" ] && { echo "Cleaning up directory $TARGETDIR" ; rm -r $TARGETDIR ; } | ||||||
|  | 
 | ||||||
|  | cp -af "$BASEDIR/devnet/" "$TARGETDIR" | ||||||
|  | cp -af "$BASEDIR/credentials" "$TARGETDIR" | ||||||
|  | cp -af "$BASEDIR/protocol-parameters.json" "$TARGETDIR" | ||||||
|  | echo '{"Producers": []}' > "$TARGETDIR/topology.json" | ||||||
|  | sed -i "s/\"startTime\": [0-9]*/\"startTime\": $(date +%s)/" "$TARGETDIR/genesis-byron.json" && \ | ||||||
|  | sed -i "s/\"systemStart\": \".*\"/\"systemStart\": \"$(date -u +%FT%TZ)\"/" "$TARGETDIR/genesis-shelley.json" | ||||||
|  | 
 | ||||||
|  | find $TARGETDIR -type f -exec chmod 0400 {} \; | ||||||
|  | mkdir "$TARGETDIR/ipc" | ||||||
|  | echo "Prepared devnet, you can start the cluster now" | ||||||
							
								
								
									
										386
									
								
								demo/protocol-parameters.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										386
									
								
								demo/protocol-parameters.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,386 @@ | |||||||
|  | { | ||||||
|  |     "collateralPercentage": 150, | ||||||
|  |     "costModels": { | ||||||
|  |         "PlutusScriptV1": { | ||||||
|  |             "addInteger-cpu-arguments-intercept": 197209, | ||||||
|  |             "addInteger-cpu-arguments-slope": 0, | ||||||
|  |             "addInteger-memory-arguments-intercept": 1, | ||||||
|  |             "addInteger-memory-arguments-slope": 1, | ||||||
|  |             "appendByteString-cpu-arguments-intercept": 396231, | ||||||
|  |             "appendByteString-cpu-arguments-slope": 621, | ||||||
|  |             "appendByteString-memory-arguments-intercept": 0, | ||||||
|  |             "appendByteString-memory-arguments-slope": 1, | ||||||
|  |             "appendString-cpu-arguments-intercept": 150000, | ||||||
|  |             "appendString-cpu-arguments-slope": 1000, | ||||||
|  |             "appendString-memory-arguments-intercept": 0, | ||||||
|  |             "appendString-memory-arguments-slope": 1, | ||||||
|  |             "bData-cpu-arguments": 150000, | ||||||
|  |             "bData-memory-arguments": 32, | ||||||
|  |             "blake2b_256-cpu-arguments-intercept": 2477736, | ||||||
|  |             "blake2b_256-cpu-arguments-slope": 29175, | ||||||
|  |             "blake2b_256-memory-arguments": 4, | ||||||
|  |             "cekApplyCost-exBudgetCPU": 29773, | ||||||
|  |             "cekApplyCost-exBudgetMemory": 100, | ||||||
|  |             "cekBuiltinCost-exBudgetCPU": 29773, | ||||||
|  |             "cekBuiltinCost-exBudgetMemory": 100, | ||||||
|  |             "cekConstCost-exBudgetCPU": 29773, | ||||||
|  |             "cekConstCost-exBudgetMemory": 100, | ||||||
|  |             "cekDelayCost-exBudgetCPU": 29773, | ||||||
|  |             "cekDelayCost-exBudgetMemory": 100, | ||||||
|  |             "cekForceCost-exBudgetCPU": 29773, | ||||||
|  |             "cekForceCost-exBudgetMemory": 100, | ||||||
|  |             "cekLamCost-exBudgetCPU": 29773, | ||||||
|  |             "cekLamCost-exBudgetMemory": 100, | ||||||
|  |             "cekStartupCost-exBudgetCPU": 100, | ||||||
|  |             "cekStartupCost-exBudgetMemory": 100, | ||||||
|  |             "cekVarCost-exBudgetCPU": 29773, | ||||||
|  |             "cekVarCost-exBudgetMemory": 100, | ||||||
|  |             "chooseData-cpu-arguments": 150000, | ||||||
|  |             "chooseData-memory-arguments": 32, | ||||||
|  |             "chooseList-cpu-arguments": 150000, | ||||||
|  |             "chooseList-memory-arguments": 32, | ||||||
|  |             "chooseUnit-cpu-arguments": 150000, | ||||||
|  |             "chooseUnit-memory-arguments": 32, | ||||||
|  |             "consByteString-cpu-arguments-intercept": 150000, | ||||||
|  |             "consByteString-cpu-arguments-slope": 1000, | ||||||
|  |             "consByteString-memory-arguments-intercept": 0, | ||||||
|  |             "consByteString-memory-arguments-slope": 1, | ||||||
|  |             "constrData-cpu-arguments": 150000, | ||||||
|  |             "constrData-memory-arguments": 32, | ||||||
|  |             "decodeUtf8-cpu-arguments-intercept": 150000, | ||||||
|  |             "decodeUtf8-cpu-arguments-slope": 1000, | ||||||
|  |             "decodeUtf8-memory-arguments-intercept": 0, | ||||||
|  |             "decodeUtf8-memory-arguments-slope": 8, | ||||||
|  |             "divideInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "divideInteger-memory-arguments-intercept": 0, | ||||||
|  |             "divideInteger-memory-arguments-minimum": 1, | ||||||
|  |             "divideInteger-memory-arguments-slope": 1, | ||||||
|  |             "encodeUtf8-cpu-arguments-intercept": 150000, | ||||||
|  |             "encodeUtf8-cpu-arguments-slope": 1000, | ||||||
|  |             "encodeUtf8-memory-arguments-intercept": 0, | ||||||
|  |             "encodeUtf8-memory-arguments-slope": 8, | ||||||
|  |             "equalsByteString-cpu-arguments-constant": 150000, | ||||||
|  |             "equalsByteString-cpu-arguments-intercept": 112536, | ||||||
|  |             "equalsByteString-cpu-arguments-slope": 247, | ||||||
|  |             "equalsByteString-memory-arguments": 1, | ||||||
|  |             "equalsData-cpu-arguments-intercept": 150000, | ||||||
|  |             "equalsData-cpu-arguments-slope": 10000, | ||||||
|  |             "equalsData-memory-arguments": 1, | ||||||
|  |             "equalsInteger-cpu-arguments-intercept": 136542, | ||||||
|  |             "equalsInteger-cpu-arguments-slope": 1326, | ||||||
|  |             "equalsInteger-memory-arguments": 1, | ||||||
|  |             "equalsString-cpu-arguments-constant": 1000, | ||||||
|  |             "equalsString-cpu-arguments-intercept": 150000, | ||||||
|  |             "equalsString-cpu-arguments-slope": 1000, | ||||||
|  |             "equalsString-memory-arguments": 1, | ||||||
|  |             "fstPair-cpu-arguments": 150000, | ||||||
|  |             "fstPair-memory-arguments": 32, | ||||||
|  |             "headList-cpu-arguments": 150000, | ||||||
|  |             "headList-memory-arguments": 32, | ||||||
|  |             "iData-cpu-arguments": 150000, | ||||||
|  |             "iData-memory-arguments": 32, | ||||||
|  |             "ifThenElse-cpu-arguments": 1, | ||||||
|  |             "ifThenElse-memory-arguments": 1, | ||||||
|  |             "indexByteString-cpu-arguments": 150000, | ||||||
|  |             "indexByteString-memory-arguments": 1, | ||||||
|  |             "lengthOfByteString-cpu-arguments": 150000, | ||||||
|  |             "lengthOfByteString-memory-arguments": 4, | ||||||
|  |             "lessThanByteString-cpu-arguments-intercept": 103599, | ||||||
|  |             "lessThanByteString-cpu-arguments-slope": 248, | ||||||
|  |             "lessThanByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-intercept": 103599, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-slope": 248, | ||||||
|  |             "lessThanEqualsByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-intercept": 145276, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-slope": 1366, | ||||||
|  |             "lessThanEqualsInteger-memory-arguments": 1, | ||||||
|  |             "lessThanInteger-cpu-arguments-intercept": 179690, | ||||||
|  |             "lessThanInteger-cpu-arguments-slope": 497, | ||||||
|  |             "lessThanInteger-memory-arguments": 1, | ||||||
|  |             "listData-cpu-arguments": 150000, | ||||||
|  |             "listData-memory-arguments": 32, | ||||||
|  |             "mapData-cpu-arguments": 150000, | ||||||
|  |             "mapData-memory-arguments": 32, | ||||||
|  |             "mkCons-cpu-arguments": 150000, | ||||||
|  |             "mkCons-memory-arguments": 32, | ||||||
|  |             "mkNilData-cpu-arguments": 150000, | ||||||
|  |             "mkNilData-memory-arguments": 32, | ||||||
|  |             "mkNilPairData-cpu-arguments": 150000, | ||||||
|  |             "mkNilPairData-memory-arguments": 32, | ||||||
|  |             "mkPairData-cpu-arguments": 150000, | ||||||
|  |             "mkPairData-memory-arguments": 32, | ||||||
|  |             "modInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "modInteger-memory-arguments-intercept": 0, | ||||||
|  |             "modInteger-memory-arguments-minimum": 1, | ||||||
|  |             "modInteger-memory-arguments-slope": 1, | ||||||
|  |             "multiplyInteger-cpu-arguments-intercept": 61516, | ||||||
|  |             "multiplyInteger-cpu-arguments-slope": 11218, | ||||||
|  |             "multiplyInteger-memory-arguments-intercept": 0, | ||||||
|  |             "multiplyInteger-memory-arguments-slope": 1, | ||||||
|  |             "nullList-cpu-arguments": 150000, | ||||||
|  |             "nullList-memory-arguments": 32, | ||||||
|  |             "quotientInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "quotientInteger-memory-arguments-intercept": 0, | ||||||
|  |             "quotientInteger-memory-arguments-minimum": 1, | ||||||
|  |             "quotientInteger-memory-arguments-slope": 1, | ||||||
|  |             "remainderInteger-cpu-arguments-constant": 148000, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-intercept": 425507, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-slope": 118, | ||||||
|  |             "remainderInteger-memory-arguments-intercept": 0, | ||||||
|  |             "remainderInteger-memory-arguments-minimum": 1, | ||||||
|  |             "remainderInteger-memory-arguments-slope": 1, | ||||||
|  |             "sha2_256-cpu-arguments-intercept": 2477736, | ||||||
|  |             "sha2_256-cpu-arguments-slope": 29175, | ||||||
|  |             "sha2_256-memory-arguments": 4, | ||||||
|  |             "sha3_256-cpu-arguments-intercept": 0, | ||||||
|  |             "sha3_256-cpu-arguments-slope": 82363, | ||||||
|  |             "sha3_256-memory-arguments": 4, | ||||||
|  |             "sliceByteString-cpu-arguments-intercept": 150000, | ||||||
|  |             "sliceByteString-cpu-arguments-slope": 5000, | ||||||
|  |             "sliceByteString-memory-arguments-intercept": 0, | ||||||
|  |             "sliceByteString-memory-arguments-slope": 1, | ||||||
|  |             "sndPair-cpu-arguments": 150000, | ||||||
|  |             "sndPair-memory-arguments": 32, | ||||||
|  |             "subtractInteger-cpu-arguments-intercept": 197209, | ||||||
|  |             "subtractInteger-cpu-arguments-slope": 0, | ||||||
|  |             "subtractInteger-memory-arguments-intercept": 1, | ||||||
|  |             "subtractInteger-memory-arguments-slope": 1, | ||||||
|  |             "tailList-cpu-arguments": 150000, | ||||||
|  |             "tailList-memory-arguments": 32, | ||||||
|  |             "trace-cpu-arguments": 150000, | ||||||
|  |             "trace-memory-arguments": 32, | ||||||
|  |             "unBData-cpu-arguments": 150000, | ||||||
|  |             "unBData-memory-arguments": 32, | ||||||
|  |             "unConstrData-cpu-arguments": 150000, | ||||||
|  |             "unConstrData-memory-arguments": 32, | ||||||
|  |             "unIData-cpu-arguments": 150000, | ||||||
|  |             "unIData-memory-arguments": 32, | ||||||
|  |             "unListData-cpu-arguments": 150000, | ||||||
|  |             "unListData-memory-arguments": 32, | ||||||
|  |             "unMapData-cpu-arguments": 150000, | ||||||
|  |             "unMapData-memory-arguments": 32, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-intercept": 3345831, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-slope": 1, | ||||||
|  |             "verifyEd25519Signature-memory-arguments": 1 | ||||||
|  |         }, | ||||||
|  |         "PlutusScriptV2": { | ||||||
|  |             "addInteger-cpu-arguments-intercept": 205665, | ||||||
|  |             "addInteger-cpu-arguments-slope": 812, | ||||||
|  |             "addInteger-memory-arguments-intercept": 1, | ||||||
|  |             "addInteger-memory-arguments-slope": 1, | ||||||
|  |             "appendByteString-cpu-arguments-intercept": 1000, | ||||||
|  |             "appendByteString-cpu-arguments-slope": 571, | ||||||
|  |             "appendByteString-memory-arguments-intercept": 0, | ||||||
|  |             "appendByteString-memory-arguments-slope": 1, | ||||||
|  |             "appendString-cpu-arguments-intercept": 1000, | ||||||
|  |             "appendString-cpu-arguments-slope": 24177, | ||||||
|  |             "appendString-memory-arguments-intercept": 4, | ||||||
|  |             "appendString-memory-arguments-slope": 1, | ||||||
|  |             "bData-cpu-arguments": 1000, | ||||||
|  |             "bData-memory-arguments": 32, | ||||||
|  |             "blake2b_256-cpu-arguments-intercept": 117366, | ||||||
|  |             "blake2b_256-cpu-arguments-slope": 10475, | ||||||
|  |             "blake2b_256-memory-arguments": 4, | ||||||
|  |             "cekApplyCost-exBudgetCPU": 23000, | ||||||
|  |             "cekApplyCost-exBudgetMemory": 100, | ||||||
|  |             "cekBuiltinCost-exBudgetCPU": 23000, | ||||||
|  |             "cekBuiltinCost-exBudgetMemory": 100, | ||||||
|  |             "cekConstCost-exBudgetCPU": 23000, | ||||||
|  |             "cekConstCost-exBudgetMemory": 100, | ||||||
|  |             "cekDelayCost-exBudgetCPU": 23000, | ||||||
|  |             "cekDelayCost-exBudgetMemory": 100, | ||||||
|  |             "cekForceCost-exBudgetCPU": 23000, | ||||||
|  |             "cekForceCost-exBudgetMemory": 100, | ||||||
|  |             "cekLamCost-exBudgetCPU": 23000, | ||||||
|  |             "cekLamCost-exBudgetMemory": 100, | ||||||
|  |             "cekStartupCost-exBudgetCPU": 100, | ||||||
|  |             "cekStartupCost-exBudgetMemory": 100, | ||||||
|  |             "cekVarCost-exBudgetCPU": 23000, | ||||||
|  |             "cekVarCost-exBudgetMemory": 100, | ||||||
|  |             "chooseData-cpu-arguments": 19537, | ||||||
|  |             "chooseData-memory-arguments": 32, | ||||||
|  |             "chooseList-cpu-arguments": 175354, | ||||||
|  |             "chooseList-memory-arguments": 32, | ||||||
|  |             "chooseUnit-cpu-arguments": 46417, | ||||||
|  |             "chooseUnit-memory-arguments": 4, | ||||||
|  |             "consByteString-cpu-arguments-intercept": 221973, | ||||||
|  |             "consByteString-cpu-arguments-slope": 511, | ||||||
|  |             "consByteString-memory-arguments-intercept": 0, | ||||||
|  |             "consByteString-memory-arguments-slope": 1, | ||||||
|  |             "constrData-cpu-arguments": 89141, | ||||||
|  |             "constrData-memory-arguments": 32, | ||||||
|  |             "decodeUtf8-cpu-arguments-intercept": 497525, | ||||||
|  |             "decodeUtf8-cpu-arguments-slope": 14068, | ||||||
|  |             "decodeUtf8-memory-arguments-intercept": 4, | ||||||
|  |             "decodeUtf8-memory-arguments-slope": 2, | ||||||
|  |             "divideInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "divideInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "divideInteger-memory-arguments-intercept": 0, | ||||||
|  |             "divideInteger-memory-arguments-minimum": 1, | ||||||
|  |             "divideInteger-memory-arguments-slope": 1, | ||||||
|  |             "encodeUtf8-cpu-arguments-intercept": 1000, | ||||||
|  |             "encodeUtf8-cpu-arguments-slope": 28662, | ||||||
|  |             "encodeUtf8-memory-arguments-intercept": 4, | ||||||
|  |             "encodeUtf8-memory-arguments-slope": 2, | ||||||
|  |             "equalsByteString-cpu-arguments-constant": 245000, | ||||||
|  |             "equalsByteString-cpu-arguments-intercept": 216773, | ||||||
|  |             "equalsByteString-cpu-arguments-slope": 62, | ||||||
|  |             "equalsByteString-memory-arguments": 1, | ||||||
|  |             "equalsData-cpu-arguments-intercept": 1060367, | ||||||
|  |             "equalsData-cpu-arguments-slope": 12586, | ||||||
|  |             "equalsData-memory-arguments": 1, | ||||||
|  |             "equalsInteger-cpu-arguments-intercept": 208512, | ||||||
|  |             "equalsInteger-cpu-arguments-slope": 421, | ||||||
|  |             "equalsInteger-memory-arguments": 1, | ||||||
|  |             "equalsString-cpu-arguments-constant": 187000, | ||||||
|  |             "equalsString-cpu-arguments-intercept": 1000, | ||||||
|  |             "equalsString-cpu-arguments-slope": 52998, | ||||||
|  |             "equalsString-memory-arguments": 1, | ||||||
|  |             "fstPair-cpu-arguments": 80436, | ||||||
|  |             "fstPair-memory-arguments": 32, | ||||||
|  |             "headList-cpu-arguments": 43249, | ||||||
|  |             "headList-memory-arguments": 32, | ||||||
|  |             "iData-cpu-arguments": 1000, | ||||||
|  |             "iData-memory-arguments": 32, | ||||||
|  |             "ifThenElse-cpu-arguments": 80556, | ||||||
|  |             "ifThenElse-memory-arguments": 1, | ||||||
|  |             "indexByteString-cpu-arguments": 57667, | ||||||
|  |             "indexByteString-memory-arguments": 4, | ||||||
|  |             "lengthOfByteString-cpu-arguments": 1000, | ||||||
|  |             "lengthOfByteString-memory-arguments": 10, | ||||||
|  |             "lessThanByteString-cpu-arguments-intercept": 197145, | ||||||
|  |             "lessThanByteString-cpu-arguments-slope": 156, | ||||||
|  |             "lessThanByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-intercept": 197145, | ||||||
|  |             "lessThanEqualsByteString-cpu-arguments-slope": 156, | ||||||
|  |             "lessThanEqualsByteString-memory-arguments": 1, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-intercept": 204924, | ||||||
|  |             "lessThanEqualsInteger-cpu-arguments-slope": 473, | ||||||
|  |             "lessThanEqualsInteger-memory-arguments": 1, | ||||||
|  |             "lessThanInteger-cpu-arguments-intercept": 208896, | ||||||
|  |             "lessThanInteger-cpu-arguments-slope": 511, | ||||||
|  |             "lessThanInteger-memory-arguments": 1, | ||||||
|  |             "listData-cpu-arguments": 52467, | ||||||
|  |             "listData-memory-arguments": 32, | ||||||
|  |             "mapData-cpu-arguments": 64832, | ||||||
|  |             "mapData-memory-arguments": 32, | ||||||
|  |             "mkCons-cpu-arguments": 65493, | ||||||
|  |             "mkCons-memory-arguments": 32, | ||||||
|  |             "mkNilData-cpu-arguments": 22558, | ||||||
|  |             "mkNilData-memory-arguments": 32, | ||||||
|  |             "mkNilPairData-cpu-arguments": 16563, | ||||||
|  |             "mkNilPairData-memory-arguments": 32, | ||||||
|  |             "mkPairData-cpu-arguments": 76511, | ||||||
|  |             "mkPairData-memory-arguments": 32, | ||||||
|  |             "modInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "modInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "modInteger-memory-arguments-intercept": 0, | ||||||
|  |             "modInteger-memory-arguments-minimum": 1, | ||||||
|  |             "modInteger-memory-arguments-slope": 1, | ||||||
|  |             "multiplyInteger-cpu-arguments-intercept": 69522, | ||||||
|  |             "multiplyInteger-cpu-arguments-slope": 11687, | ||||||
|  |             "multiplyInteger-memory-arguments-intercept": 0, | ||||||
|  |             "multiplyInteger-memory-arguments-slope": 1, | ||||||
|  |             "nullList-cpu-arguments": 60091, | ||||||
|  |             "nullList-memory-arguments": 32, | ||||||
|  |             "quotientInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "quotientInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "quotientInteger-memory-arguments-intercept": 0, | ||||||
|  |             "quotientInteger-memory-arguments-minimum": 1, | ||||||
|  |             "quotientInteger-memory-arguments-slope": 1, | ||||||
|  |             "remainderInteger-cpu-arguments-constant": 196500, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-intercept": 453240, | ||||||
|  |             "remainderInteger-cpu-arguments-model-arguments-slope": 220, | ||||||
|  |             "remainderInteger-memory-arguments-intercept": 0, | ||||||
|  |             "remainderInteger-memory-arguments-minimum": 1, | ||||||
|  |             "remainderInteger-memory-arguments-slope": 1, | ||||||
|  |             "serialiseData-cpu-arguments-intercept": 1159724, | ||||||
|  |             "serialiseData-cpu-arguments-slope": 392670, | ||||||
|  |             "serialiseData-memory-arguments-intercept": 0, | ||||||
|  |             "serialiseData-memory-arguments-slope": 2, | ||||||
|  |             "sha2_256-cpu-arguments-intercept": 806990, | ||||||
|  |             "sha2_256-cpu-arguments-slope": 30482, | ||||||
|  |             "sha2_256-memory-arguments": 4, | ||||||
|  |             "sha3_256-cpu-arguments-intercept": 1927926, | ||||||
|  |             "sha3_256-cpu-arguments-slope": 82523, | ||||||
|  |             "sha3_256-memory-arguments": 4, | ||||||
|  |             "sliceByteString-cpu-arguments-intercept": 265318, | ||||||
|  |             "sliceByteString-cpu-arguments-slope": 0, | ||||||
|  |             "sliceByteString-memory-arguments-intercept": 4, | ||||||
|  |             "sliceByteString-memory-arguments-slope": 0, | ||||||
|  |             "sndPair-cpu-arguments": 85931, | ||||||
|  |             "sndPair-memory-arguments": 32, | ||||||
|  |             "subtractInteger-cpu-arguments-intercept": 205665, | ||||||
|  |             "subtractInteger-cpu-arguments-slope": 812, | ||||||
|  |             "subtractInteger-memory-arguments-intercept": 1, | ||||||
|  |             "subtractInteger-memory-arguments-slope": 1, | ||||||
|  |             "tailList-cpu-arguments": 41182, | ||||||
|  |             "tailList-memory-arguments": 32, | ||||||
|  |             "trace-cpu-arguments": 212342, | ||||||
|  |             "trace-memory-arguments": 32, | ||||||
|  |             "unBData-cpu-arguments": 31220, | ||||||
|  |             "unBData-memory-arguments": 32, | ||||||
|  |             "unConstrData-cpu-arguments": 32696, | ||||||
|  |             "unConstrData-memory-arguments": 32, | ||||||
|  |             "unIData-cpu-arguments": 43357, | ||||||
|  |             "unIData-memory-arguments": 32, | ||||||
|  |             "unListData-cpu-arguments": 32247, | ||||||
|  |             "unListData-memory-arguments": 32, | ||||||
|  |             "unMapData-cpu-arguments": 38314, | ||||||
|  |             "unMapData-memory-arguments": 32, | ||||||
|  |             "verifyEcdsaSecp256k1Signature-cpu-arguments": 35892428, | ||||||
|  |             "verifyEcdsaSecp256k1Signature-memory-arguments": 10, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-intercept": 9462713, | ||||||
|  |             "verifyEd25519Signature-cpu-arguments-slope": 1021, | ||||||
|  |             "verifyEd25519Signature-memory-arguments": 10, | ||||||
|  |             "verifySchnorrSecp256k1Signature-cpu-arguments-intercept": 38887044, | ||||||
|  |             "verifySchnorrSecp256k1Signature-cpu-arguments-slope": 32947, | ||||||
|  |             "verifySchnorrSecp256k1Signature-memory-arguments": 10 | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     "decentralization": null, | ||||||
|  |     "executionUnitPrices": { | ||||||
|  |         "priceMemory": 5.77e-2, | ||||||
|  |         "priceSteps": 7.21e-5 | ||||||
|  |     }, | ||||||
|  |     "extraPraosEntropy": null, | ||||||
|  |     "maxBlockBodySize": 65536, | ||||||
|  |     "maxBlockExecutionUnits": { | ||||||
|  |         "memory": 80000000, | ||||||
|  |         "steps": 40000000000 | ||||||
|  |     }, | ||||||
|  |     "maxBlockHeaderSize": 1100, | ||||||
|  |     "maxCollateralInputs": 3, | ||||||
|  |     "maxTxExecutionUnits": { | ||||||
|  |         "memory": 16000000, | ||||||
|  |         "steps": 10000000000 | ||||||
|  |     }, | ||||||
|  |     "maxTxSize": 16384, | ||||||
|  |     "maxValueSize": 5000, | ||||||
|  |     "minPoolCost": 0, | ||||||
|  |     "minUTxOValue": null, | ||||||
|  |     "monetaryExpansion": 1.78650067e-3, | ||||||
|  |     "poolPledgeInfluence": 0.1, | ||||||
|  |     "poolRetireMaxEpoch": 18, | ||||||
|  |     "protocolVersion": { | ||||||
|  |         "major": 7, | ||||||
|  |         "minor": 0 | ||||||
|  |     }, | ||||||
|  |     "stakeAddressDeposit": 400000, | ||||||
|  |     "stakePoolDeposit": 500000000, | ||||||
|  |     "stakePoolTargetNum": 50, | ||||||
|  |     "treasuryCut": 0.1, | ||||||
|  |     "txFeeFixed": 0, | ||||||
|  |     "txFeePerByte": 0, | ||||||
|  |     "utxoCostPerWord": 34488, | ||||||
|  |     "utxoCostPerByte": 4310 | ||||||
|  | } | ||||||
							
								
								
									
										2
									
								
								dep/aeson-gadt-th/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/aeson-gadt-th/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										7
									
								
								dep/aeson-gadt-th/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								dep/aeson-gadt-th/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | { | ||||||
|  |   "owner": "obsidiansystems", | ||||||
|  |   "repo": "aeson-gadt-th", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "71a315a4873c2875ad737a3320320849bdcf8a2a", | ||||||
|  |   "sha256": "1sjl0a0ig0xfssl4bglakk1b7sj8wqz8hwbg7k2fk10qi7z5hb50" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/aeson-gadt-th/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/aeson-gadt-th/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/aeson/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/aeson/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										8
									
								
								dep/aeson/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								dep/aeson/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |   "owner": "haskell", | ||||||
|  |   "repo": "aeson", | ||||||
|  |   "branch": "aeson-1.5", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "78e838df44288ac7d7ac2cd77863d2c026d86dbb", | ||||||
|  |   "sha256": "181v1nz05jq992wgmz6ns7iwqx0c5w003hv5ki1kvc9zlg5dh3vf" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/aeson/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/aeson/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/constraints-extras/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/constraints-extras/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										8
									
								
								dep/constraints-extras/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								dep/constraints-extras/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |   "owner": "obsidiansystems", | ||||||
|  |   "repo": "constraints-extras", | ||||||
|  |   "branch": "release/0.3.2.0", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "42835fd9e1b4b3c4a72cd1237c04789f01c92dd0", | ||||||
|  |   "sha256": "0z7yfxxi4jywzhlkphs8ss3hd7fll8c90bbl6nr2bj63c87jx6sw" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/constraints-extras/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/constraints-extras/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/entropy/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/entropy/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										8
									
								
								dep/entropy/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								dep/entropy/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |   "owner": "obsidiansystems", | ||||||
|  |   "repo": "entropy", | ||||||
|  |   "branch": "aa/nonwindow-contexts", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "ddda007c44390d0ec8cad58810710864dee60ddd", | ||||||
|  |   "sha256": "00b7cqdqbjf953r5gmwmkn2mn36zdxqwx90xih925cni94vi4vw9" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/entropy/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/entropy/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/hydra-poc/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/hydra-poc/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										7
									
								
								dep/hydra-poc/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								dep/hydra-poc/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | { | ||||||
|  |   "owner": "input-output-hk", | ||||||
|  |   "repo": "hydra-poc", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "b0ebec965f27553a1c99e5457441219dcb9306d6", | ||||||
|  |   "sha256": "0i9sflwpy87hhrbb6c0wbjs1mj2nsln69yah6vnacb6l0j47ik4b" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/hydra-poc/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/hydra-poc/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/logging-effect/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/logging-effect/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										7
									
								
								dep/logging-effect/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								dep/logging-effect/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | { | ||||||
|  |   "owner": "ocharles", | ||||||
|  |   "repo": "logging-effect", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "efc856b62aec15f0087be0edb7ba621ed49b0bb1", | ||||||
|  |   "sha256": "13nr3zw39415js1af2vd322vhzwqb2rkrv7b5lkd97h211d7zbj9" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/logging-effect/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/logging-effect/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/nix-thunk/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/nix-thunk/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										7
									
								
								dep/nix-thunk/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										7
									
								
								dep/nix-thunk/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,7 @@ | |||||||
|  | { | ||||||
|  |   "owner": "obsidiansystems", | ||||||
|  |   "repo": "nix-thunk", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "bd0de53129ca4ac5ce313a3e021edf3638a3a22c", | ||||||
|  |   "sha256": "0cn74ylcfr9v2w94jpga9v18jn6zafb9k996afszn59iqlcfi74q" | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								dep/nix-thunk/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								dep/nix-thunk/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import (builtins.fetchTarball { | ||||||
|  |   url = "https://github.com/NixOS/nixpkgs/archive/3aad50c30c826430b0270fcf8264c8c41b005403.tar.gz"; | ||||||
|  |   sha256 = "0xwqsf08sywd23x0xvw4c4ghq0l28w2ki22h0bdn766i16z9q2gr"; | ||||||
|  | }) {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/reflex-gadt-api/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/reflex-gadt-api/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										8
									
								
								dep/reflex-gadt-api/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								dep/reflex-gadt-api/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |   "owner": "reflex-frp", | ||||||
|  |   "repo": "reflex-gadt-api", | ||||||
|  |   "branch": "develop", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "69aca72b57a74fbf0d06a1b2ac1d247ce9aa207f", | ||||||
|  |   "sha256": "116c6lna8pkas95lclm4halagc4f6rxv5hmf7rnl10lxqi9m48qw" | ||||||
|  | } | ||||||
							
								
								
									
										9
									
								
								dep/reflex-gadt-api/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										9
									
								
								dep/reflex-gadt-api/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,9 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import <nixpkgs> {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										2
									
								
								dep/reflex-platform/default.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										2
									
								
								dep/reflex-platform/default.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,2 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | import (import ./thunk.nix) | ||||||
							
								
								
									
										8
									
								
								dep/reflex-platform/github.json
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										8
									
								
								dep/reflex-platform/github.json
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,8 @@ | |||||||
|  | { | ||||||
|  |   "owner": "reflex-frp", | ||||||
|  |   "repo": "reflex-platform", | ||||||
|  |   "branch": "release/0.9.2.0", | ||||||
|  |   "private": false, | ||||||
|  |   "rev": "123a6f487ca954fd983f6d4cd6b2a69d4c463d10", | ||||||
|  |   "sha256": "16q1rq0rwi6l28fv46q8m0hvb9rxrzf574j865vaz04xy8d5p1ya" | ||||||
|  | } | ||||||
							
								
								
									
										12
									
								
								dep/reflex-platform/thunk.nix
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								dep/reflex-platform/thunk.nix
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,12 @@ | |||||||
|  | # DO NOT HAND-EDIT THIS FILE | ||||||
|  | let fetch = { private ? false, fetchSubmodules ? false, owner, repo, rev, sha256, ... }: | ||||||
|  |   if !fetchSubmodules && !private then builtins.fetchTarball { | ||||||
|  |     url = "https://github.com/${owner}/${repo}/archive/${rev}.tar.gz"; inherit sha256; | ||||||
|  |   } else (import (builtins.fetchTarball { | ||||||
|  |   url = "https://github.com/NixOS/nixpkgs/archive/3aad50c30c826430b0270fcf8264c8c41b005403.tar.gz"; | ||||||
|  |   sha256 = "0xwqsf08sywd23x0xvw4c4ghq0l28w2ki22h0bdn766i16z9q2gr"; | ||||||
|  | }) {}).fetchFromGitHub { | ||||||
|  |     inherit owner repo rev sha256 fetchSubmodules private; | ||||||
|  |   }; | ||||||
|  |   json = builtins.fromJSON (builtins.readFile ./github.json); | ||||||
|  | in fetch json | ||||||
							
								
								
									
										4
									
								
								hie.yaml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										4
									
								
								hie.yaml
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,4 @@ | |||||||
|  | cradle: | ||||||
|  |   cabal: | ||||||
|  |     - path: "src" | ||||||
|  |       component: "exe:hydra-head-demo" | ||||||
							
								
								
									
										55
									
								
								hydra-head-demo.cabal
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										55
									
								
								hydra-head-demo.cabal
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,55 @@ | |||||||
|  | name: hydra-head-demo | ||||||
|  | version: 0.1 | ||||||
|  | cabal-version: >= 1.8 | ||||||
|  | build-type: Simple | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | executable hydra-head-demo | ||||||
|  |   main-is: Main.hs | ||||||
|  |   hs-source-dirs: src | ||||||
|  |   ghc-options: -Wall -Wredundant-constraints -Wincomplete-uni-patterns -Wincomplete-record-updates -O -threaded -fno-show-valid-hole-fits | ||||||
|  |   if impl(ghcjs) | ||||||
|  |     buildable: False | ||||||
|  |   other-modules: Hydra.ClientInput | ||||||
|  |                , Hydra.Devnet | ||||||
|  |                , Hydra.ServerOutput | ||||||
|  |                , Hydra.Types | ||||||
|  |                , Paths | ||||||
|  |   build-depends: base | ||||||
|  |                , process | ||||||
|  |                , async | ||||||
|  |                , which | ||||||
|  |                -- , monad-logger | ||||||
|  |                -- , logging-effect | ||||||
|  |                , prettyprinter | ||||||
|  |                , string-interpolate | ||||||
|  |                , containers | ||||||
|  |                , text | ||||||
|  |                , witherable | ||||||
|  |                -- , io-streams | ||||||
|  |                , some | ||||||
|  |                , directory | ||||||
|  |                , aeson | ||||||
|  |                , resource-pool | ||||||
|  |                , bytestring | ||||||
|  |                , uuid | ||||||
|  |                , time | ||||||
|  |                , reflex | ||||||
|  |                , reflex-dom | ||||||
|  |                , mtl | ||||||
|  |                , jsaddle | ||||||
|  | 
 | ||||||
|  |   default-extensions: | ||||||
|  |     OverloadedStrings | ||||||
|  |     LambdaCase | ||||||
|  |     GADTs | ||||||
|  |     ScopedTypeVariables | ||||||
|  |     OverloadedStrings | ||||||
|  |     FlexibleContexts | ||||||
|  |     QuasiQuotes | ||||||
|  |     DeriveGeneric | ||||||
|  |     RecursiveDo | ||||||
|  |     RankNTypes | ||||||
|  |     PartialTypeSignatures | ||||||
|  |     TypeFamilies | ||||||
|  |     TypeApplications | ||||||
							
								
								
									
										31
									
								
								src/Hydra/ClientInput.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										31
									
								
								src/Hydra/ClientInput.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,31 @@ | |||||||
|  | -- |  | ||||||
|  | 
 | ||||||
|  | module Hydra.ClientInput where | ||||||
|  | 
 | ||||||
|  | import GHC.Generics | ||||||
|  | import Data.Aeson | ||||||
|  | 
 | ||||||
|  | import Data.Map (Map) | ||||||
|  | import qualified Data.Text as T | ||||||
|  | import Data.Time (UTCTime) | ||||||
|  | import Data.Text (Text) | ||||||
|  | import Numeric.Natural (Natural) | ||||||
|  | import Hydra.Types | ||||||
|  | 
 | ||||||
|  | -- Copied and adapted from hydra-poc:hydra-node/src/hydra/ServerOutput.hs | ||||||
|  | -- Anything that took effort was replaced by Value | ||||||
|  | 
 | ||||||
|  | data ClientInput | ||||||
|  |   = Init {contestationPeriod :: ContestationPeriod} | ||||||
|  |   | Abort | ||||||
|  |   | Commit {utxo :: WholeUTXO} | ||||||
|  |   | NewTx {transaction :: T.Text} | ||||||
|  |   | GetUTxO | ||||||
|  |   | Close | ||||||
|  |   | Contest | ||||||
|  |   | Fanout | ||||||
|  |   deriving (Eq, Generic, Show) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | instance ToJSON ClientInput | ||||||
|  | instance FromJSON ClientInput | ||||||
							
								
								
									
										377
									
								
								src/Hydra/Devnet.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										377
									
								
								src/Hydra/Devnet.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,377 @@ | |||||||
|  | {-# LANGUAGE TemplateHaskell #-} | ||||||
|  | {-# LANGUAGE TupleSections #-} | ||||||
|  | 
 | ||||||
|  | -- |  | ||||||
|  | 
 | ||||||
|  | module Hydra.Devnet | ||||||
|  |   ( HydraScriptTxId | ||||||
|  |   , HydraKeyInfo(..) | ||||||
|  |   , SigningKey | ||||||
|  |   , KeyPair(..) | ||||||
|  |   , getCardanoAddress | ||||||
|  |   , seedAddressFromFaucetAndWait | ||||||
|  |   , publishReferenceScripts | ||||||
|  |   , queryAddressUTXOs | ||||||
|  |   , buildSignedHydraTx | ||||||
|  |   , generateKeys | ||||||
|  |   , cardanoNodePath | ||||||
|  |   , hydraNodePath | ||||||
|  |   , prepareDevnet | ||||||
|  |   , devnetMagic | ||||||
|  |   , minTxLovelace | ||||||
|  |   ) | ||||||
|  | 
 | ||||||
|  | where | ||||||
|  | 
 | ||||||
|  | import System.Which | ||||||
|  | import System.Directory | ||||||
|  | import System.Process | ||||||
|  | 
 | ||||||
|  | import Control.Monad | ||||||
|  | import Control.Monad.IO.Class | ||||||
|  | import Data.Aeson | ||||||
|  | 
 | ||||||
|  | import Control.Concurrent | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | import Data.Map (Map) | ||||||
|  | 
 | ||||||
|  | import Data.Bool | ||||||
|  | import qualified Data.Text as T | ||||||
|  | import qualified Data.Map as Map | ||||||
|  | import Data.String.Interpolate (i) | ||||||
|  | import Paths | ||||||
|  | 
 | ||||||
|  | import qualified Data.ByteString.Lazy.Char8 as BS | ||||||
|  | 
 | ||||||
|  | import Hydra.Types | ||||||
|  | import qualified Data.UUID.V4 as UUIDV4 | ||||||
|  | import qualified Data.UUID as UUID | ||||||
|  | import Data.UUID (UUID) | ||||||
|  | import Data.Maybe (fromMaybe) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | devnetMagic :: Int | ||||||
|  | devnetMagic = 42 | ||||||
|  | 
 | ||||||
|  | prepareDevnet :: IO () | ||||||
|  | prepareDevnet = do | ||||||
|  |   output <- readCreateProcess (shell "[ -d devnet ] || ./demo/prepare-devnet.sh") "" | ||||||
|  |   putStrLn output | ||||||
|  | 
 | ||||||
|  | cardanoNodePath :: FilePath | ||||||
|  | cardanoNodePath = $(staticWhich "cardano-node") | ||||||
|  | 
 | ||||||
|  | cardanoCliPath :: FilePath | ||||||
|  | cardanoCliPath = $(staticWhich "cardano-cli") | ||||||
|  | 
 | ||||||
|  | hydraNodePath :: FilePath | ||||||
|  | hydraNodePath = $(staticWhich "hydra-node") | ||||||
|  | 
 | ||||||
|  | jqPath :: FilePath | ||||||
|  | jqPath = $(staticWhich "jq") | ||||||
|  | 
 | ||||||
|  | type TxId = T.Text | ||||||
|  | 
 | ||||||
|  | type HydraScriptTxId = T.Text | ||||||
|  | 
 | ||||||
|  | type DraftTx = FilePath | ||||||
|  | type SignedTx = FilePath | ||||||
|  | 
 | ||||||
|  | devnetNetworkId :: Int | ||||||
|  | devnetNetworkId = 42 | ||||||
|  | 
 | ||||||
|  | generateKeys :: (MonadIO m) => m HydraKeyInfo | ||||||
|  | generateKeys = do | ||||||
|  |   basePath <- liftIO getTempPath' | ||||||
|  |   HydraKeyInfo <$> generateCardanoKeys basePath <*> generateHydraKeys basePath | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | type SigningKey = String | ||||||
|  | type VerificationKey = String | ||||||
|  | 
 | ||||||
|  | data KeyPair = KeyPair | ||||||
|  |   { _signingKey :: SigningKey | ||||||
|  |   , _verificationKey :: VerificationKey | ||||||
|  |   } | ||||||
|  |   deriving (Show,Read) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | data HydraKeyInfo = HydraKeyInfo | ||||||
|  |   { _cardanoKeys :: KeyPair | ||||||
|  |   , _hydraKeys :: KeyPair | ||||||
|  |   } | ||||||
|  |   deriving (Show,Read) | ||||||
|  | 
 | ||||||
|  | -- | Generate Cardano keys. Calling with an e.g. "my/keys/alice" | ||||||
|  | -- argument results in "my/keys/alice.cardano.{vk,sk}" keys being | ||||||
|  | -- written. | ||||||
|  | generateCardanoKeys :: (MonadIO m) => String -> m KeyPair | ||||||
|  | generateCardanoKeys path = do | ||||||
|  |   output <- liftIO $ | ||||||
|  |     readCreateProcess | ||||||
|  |     (proc cardanoCliPath [ "address" | ||||||
|  |                          , "key-gen" | ||||||
|  |                          , "--verification-key-file" | ||||||
|  |                          , [i|#{path}.cardano.vk|] | ||||||
|  |                          , "--signing-key-file" | ||||||
|  |                          , [i|#{path}.cardano.sk|] | ||||||
|  |                          ]) | ||||||
|  |     "" | ||||||
|  |   liftIO $ putStrLn output | ||||||
|  |   pure $ KeyPair [i|#{path}.cardano.sk|] [i|#{path}.cardano.vk|] | ||||||
|  | 
 | ||||||
|  | -- | Generate Hydra keys. Calling with an e.g. "my/keys/alice" | ||||||
|  | -- argument results in "my/keys/alice.hydra.{vk,sk}" keys being | ||||||
|  | -- written. | ||||||
|  | generateHydraKeys :: (MonadIO m) => String -> m KeyPair | ||||||
|  | generateHydraKeys path = do | ||||||
|  |   output <- liftIO $ | ||||||
|  |     readCreateProcess | ||||||
|  |     (proc hydraToolsPath [ "gen-hydra-key" | ||||||
|  |                          , "--output-file" | ||||||
|  |                          , [i|#{path}.hydra|] | ||||||
|  |                          ]) | ||||||
|  |     "" | ||||||
|  |   liftIO $ putStrLn output | ||||||
|  |   pure $ KeyPair [i|#{path}.hydra.sk|] [i|#{path}.hydra.vk|] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | publishReferenceScripts :: (MonadIO m) => m HydraScriptTxId | ||||||
|  | publishReferenceScripts = do | ||||||
|  |   liftIO . putStrLn $ "Publishing reference scripts ('νInitial' & 'νCommit')..." | ||||||
|  |   fmap (T.strip . T.pack) $ liftIO $ readCreateProcess cp "" | ||||||
|  |   where | ||||||
|  |     cp = proc hydraNodePath [ "publish-scripts" | ||||||
|  |                               , "--network-id" | ||||||
|  |                               , show devnetNetworkId | ||||||
|  |                               , "--node-socket" | ||||||
|  |                               , "devnet/node.socket" | ||||||
|  |                               , "--cardano-signing-key" | ||||||
|  |                               , "devnet/credentials/faucet.sk" | ||||||
|  |                               ] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | waitForTxIn :: (MonadIO m) => TxIn -> m () | ||||||
|  | waitForTxIn txin = do | ||||||
|  |   liftIO . putStrLn $ "Waiting for utxo " <> show txin <> ".." | ||||||
|  |   liftIO waitFn | ||||||
|  |   where | ||||||
|  |     waitFn = do | ||||||
|  |       exists <- txInExists txin | ||||||
|  |       threadDelay 10000 | ||||||
|  |       unless exists waitFn | ||||||
|  | 
 | ||||||
|  | txInExists :: TxIn -> IO Bool | ||||||
|  | txInExists txin = do | ||||||
|  |   result <- fmap (T.strip . T.pack) $ readCreateProcess cp "" >>= readProcess jqPath (pure $ ".\"" <> asStr <> "\"") | ||||||
|  |   pure $ case result of | ||||||
|  |     "null" -> False | ||||||
|  |     _ -> True | ||||||
|  |   where | ||||||
|  |     asStr = T.unpack txin | ||||||
|  |     cp = (proc cardanoCliPath [ "query" | ||||||
|  |                               , "utxo" | ||||||
|  |                               , "--tx-in" | ||||||
|  |                               , asStr | ||||||
|  |                               , "--out-file" | ||||||
|  |                               , "/dev/stdout" | ||||||
|  |                               , "--testnet-magic" | ||||||
|  |                               , "42" | ||||||
|  |                               ]) { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  | 
 | ||||||
|  | txInput :: Int -> TxId -> TxIn | ||||||
|  | txInput index txid = txid <> "#" <> (T.pack . show) index | ||||||
|  | 
 | ||||||
|  | -- TODO: use this in checks? | ||||||
|  | minTxLovelace :: Int | ||||||
|  | minTxLovelace = 857690 | ||||||
|  | 
 | ||||||
|  | queryAddressUTXOs :: MonadIO m => Address -> m WholeUTXO | ||||||
|  | queryAddressUTXOs addr = liftIO $ do | ||||||
|  |   let queryProc = | ||||||
|  |         (proc cardanoCliPath [ "query" | ||||||
|  |                              , "utxo" | ||||||
|  |                              , "--address" | ||||||
|  |                              , addr | ||||||
|  |                              , "--testnet-magic" | ||||||
|  |                              , "42" | ||||||
|  |                              , "--out-file" | ||||||
|  |                              , "/dev/stdout" | ||||||
|  |                              ]) | ||||||
|  |         { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  |   str <- readCreateProcess queryProc "" | ||||||
|  |   pure $ fromMaybe mempty $ decode $ BS.pack str | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | getTempPath' :: IO FilePath | ||||||
|  | getTempPath' = snd <$> getTempPath | ||||||
|  | 
 | ||||||
|  | getTempPath :: IO (UUID, FilePath) | ||||||
|  | getTempPath = do | ||||||
|  |   createDirectoryIfMissing True "tmp" | ||||||
|  |   uid <- UUIDV4.nextRandom | ||||||
|  |   pure . (uid,) . ("tmp/" <>) . UUID.toString $ uid | ||||||
|  | 
 | ||||||
|  | -- TODO(skylar): Check lovelace vs the full amount! | ||||||
|  | buildSignedHydraTx :: SigningKey -> Address -> Address -> Map TxIn Lovelace -> Lovelace -> IO String | ||||||
|  | buildSignedHydraTx signingKey fromAddr toAddr txInAmounts amount = do | ||||||
|  |   let fullAmount = sum txInAmounts | ||||||
|  |   txBodyPath <- snd <$> getTempPath | ||||||
|  |   void $ readCreateProcess (proc cardanoCliPath | ||||||
|  |                        ([ "transaction" | ||||||
|  |                         , "build-raw" | ||||||
|  |                         , "--babbage-era" | ||||||
|  |                         ] | ||||||
|  |                         <> (concatMap (\txin -> ["--tx-in", T.unpack txin]) . Map.keys $ txInAmounts) | ||||||
|  |                         <> | ||||||
|  |                         [ "--tx-out" | ||||||
|  |                         , [i|#{toAddr}+#{amount}|] | ||||||
|  |                         , "--tx-out" | ||||||
|  |                         , [i|#{fromAddr}+#{fullAmount - amount}|] | ||||||
|  |                         , "--fee" | ||||||
|  |                         , "0" | ||||||
|  |                         , "--out-file" | ||||||
|  |                         , txBodyPath | ||||||
|  |                         ])) | ||||||
|  |     "" | ||||||
|  |   readCreateProcess | ||||||
|  |     (proc cardanoCliPath | ||||||
|  |       [ "transaction" | ||||||
|  |       , "sign" | ||||||
|  |       , "--tx-body-file" | ||||||
|  |       , txBodyPath | ||||||
|  |       , "--signing-key-file" | ||||||
|  |       , signingKey | ||||||
|  |       , "--out-file" | ||||||
|  |       , "/dev/stdout" | ||||||
|  |       ]) | ||||||
|  |     "" | ||||||
|  | --    { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  | 
 | ||||||
|  | -- | Convenience for getting faucet Output for seeding | ||||||
|  | getFirstTxIn :: Address -> IO TxIn | ||||||
|  | getFirstTxIn addr = | ||||||
|  |   readCreateProcess cp "" >>= readProcess jqPath ["-r", "keys[0]"] >>= \a -> pure $ T.strip $ T.pack a | ||||||
|  |   where | ||||||
|  |     cp = (proc cardanoCliPath [ "query" | ||||||
|  |                               , "utxo" | ||||||
|  |                               , "--address" | ||||||
|  |                               , addr | ||||||
|  |                               , "--testnet-magic" | ||||||
|  |                               , "42" | ||||||
|  |                               , "--out-file" | ||||||
|  |                               , "/dev/stdout" | ||||||
|  |                               ]) { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | getCardanoAddress :: VerificationKey -> IO Address | ||||||
|  | getCardanoAddress keyPath = | ||||||
|  |   readCreateProcess cp "" | ||||||
|  |   where | ||||||
|  |     cp = (proc cardanoCliPath [ "address" | ||||||
|  |                               , "build" | ||||||
|  |                               , "--payment-verification-key-file" | ||||||
|  |                               , keyPath | ||||||
|  |                               , "--testnet-magic" | ||||||
|  |                               , "42" | ||||||
|  |                               ]) { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | getFaucetAddress :: IO Address | ||||||
|  | getFaucetAddress = readCreateProcess cp "" | ||||||
|  |   where | ||||||
|  |     cp = (proc cardanoCliPath [ "address" | ||||||
|  |                               , "build" | ||||||
|  |                               , "--payment-verification-key-file" | ||||||
|  |                               , "devnet/credentials/faucet.vk" | ||||||
|  |                               , "--testnet-magic" | ||||||
|  |                               , "42" | ||||||
|  |                               ]) { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  | 
 | ||||||
|  | seedAddressFromFaucetAndWait :: (MonadIO m) => Address -> Lovelace -> Bool -> m TxIn | ||||||
|  | seedAddressFromFaucetAndWait addr amount isFuel = do | ||||||
|  |   txin <- liftIO $ seedAddressFromFaucet addr amount isFuel | ||||||
|  |   waitForTxIn txin | ||||||
|  |   pure txin | ||||||
|  | 
 | ||||||
|  | -- | Send an amount in lovelace to the named actor | ||||||
|  | seedAddressFromFaucet :: Address -> Lovelace -> Bool -> IO TxIn | ||||||
|  | seedAddressFromFaucet addr amount isFuel = do | ||||||
|  |   draftTx <- buildSeedTxForAddress addr amount isFuel | ||||||
|  |   signedTx <- signSeedTx' draftTx | ||||||
|  |   txin <- txInput 0 <$> seedTxIdFromSignedTx signedTx | ||||||
|  |   submitTx signedTx | ||||||
|  |   pure txin | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | buildSeedTxForAddress :: Address -> Lovelace -> Bool -> IO DraftTx | ||||||
|  | buildSeedTxForAddress addr amount isFuel = do | ||||||
|  |   filename <- getTempPath' | ||||||
|  |   -- when (amount < minTxLovelace) $ error $ "Minmum required UTxO: Lovelace " <> show minTxLovelace | ||||||
|  |   let cp faucet hash = (proc cardanoCliPath $ filter (/= "") | ||||||
|  |                               [ "transaction" | ||||||
|  |                               , "build" | ||||||
|  |                               , "--babbage-era" | ||||||
|  |                               , "--cardano-mode" | ||||||
|  |                               , "--change-address" | ||||||
|  |                               , faucet | ||||||
|  |                               , "--tx-in" | ||||||
|  |                               , hash | ||||||
|  |                               , "--tx-out" | ||||||
|  |                               , addr <> "+" <> show amount | ||||||
|  |                               ] | ||||||
|  |                               <> bool [] [ "--tx-out-datum-hash", T.unpack fuelMarkerDatumHash ] isFuel | ||||||
|  |                               <> | ||||||
|  |                               [ "--out-file" | ||||||
|  |                               , filename | ||||||
|  |                               , "--testnet-magic" | ||||||
|  |                               , "42" | ||||||
|  |                               ]) | ||||||
|  |                             { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  |   faucet <- getFaucetAddress | ||||||
|  |   hash <- getFirstTxIn faucet | ||||||
|  |   _ <- readCreateProcess (cp faucet (T.unpack hash)) "" | ||||||
|  |   pure filename | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | signSeedTx' :: DraftTx -> IO SignedTx | ||||||
|  | signSeedTx' draftFile = do | ||||||
|  |   outFile <- getTempPath' | ||||||
|  |   let cp = (proc cardanoCliPath [ "transaction" | ||||||
|  |                                 , "sign" | ||||||
|  |                                 , "--tx-body-file" | ||||||
|  |                                 , draftFile | ||||||
|  |                                 , "--signing-key-file" | ||||||
|  |                                 , "devnet/credentials/faucet.sk" | ||||||
|  |                                 , "--out-file" | ||||||
|  |                                 , outFile | ||||||
|  |                                 , "--testnet-magic" | ||||||
|  |                                 , "42" | ||||||
|  |                                 ]) | ||||||
|  |            { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  |   _ <- readCreateProcess cp "" | ||||||
|  |   pure outFile | ||||||
|  | 
 | ||||||
|  | seedTxIdFromSignedTx :: SignedTx -> IO TxId | ||||||
|  | seedTxIdFromSignedTx filename = | ||||||
|  |   T.strip . T.pack <$> readCreateProcess cp "" | ||||||
|  |   where | ||||||
|  |     cp = (proc cardanoCliPath [ "transaction" | ||||||
|  |                               , "txid" | ||||||
|  |                               , "--tx-file" | ||||||
|  |                               , filename | ||||||
|  |                               ]) { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
|  | 
 | ||||||
|  | submitTx :: SignedTx -> IO () | ||||||
|  | submitTx signedFile = do | ||||||
|  |   _ <- readCreateProcess cp "" | ||||||
|  |   pure () | ||||||
|  |   where | ||||||
|  |     cp = (proc cardanoCliPath [ "transaction" | ||||||
|  |                               , "submit" | ||||||
|  |                               , "--tx-file" | ||||||
|  |                               , signedFile | ||||||
|  |                               , "--testnet-magic" | ||||||
|  |                               , "42" | ||||||
|  |                               ]) { env = Just [("CARDANO_NODE_SOCKET_PATH", "devnet/node.socket")] } | ||||||
							
								
								
									
										1
									
								
								src/Hydra/NodeTypes.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								src/Hydra/NodeTypes.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | 
 | ||||||
							
								
								
									
										57
									
								
								src/Hydra/ServerOutput.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										57
									
								
								src/Hydra/ServerOutput.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,57 @@ | |||||||
|  | -- |  | ||||||
|  | 
 | ||||||
|  | module Hydra.ServerOutput where | ||||||
|  | 
 | ||||||
|  | import GHC.Generics | ||||||
|  | import Data.Aeson | ||||||
|  | 
 | ||||||
|  | import Data.Map (Map) | ||||||
|  | import qualified Data.Text as T | ||||||
|  | import Data.Time (UTCTime) | ||||||
|  | import Data.Text (Text) | ||||||
|  | import Numeric.Natural (Natural) | ||||||
|  | import Hydra.Types | ||||||
|  | import Hydra.ClientInput | ||||||
|  | 
 | ||||||
|  | -- Copied and adapted from hydra-poc:hydra-node/src/hydra/ServerOutput.hs | ||||||
|  | -- Anything that took effort was replaced by Value | ||||||
|  | 
 | ||||||
|  | data ServerOutput tx | ||||||
|  |   = PeerConnected {peer :: Host} | ||||||
|  |   | PeerDisconnected {peer :: Host} | ||||||
|  |   | ReadyToCommit {parties :: Value} | ||||||
|  |   | Committed {party :: Party, utxo :: WholeUTXO} | ||||||
|  |   | HeadIsOpen {utxo :: WholeUTXO} | ||||||
|  |   | HeadIsClosed | ||||||
|  |       { snapshotNumber :: SnapshotNumber | ||||||
|  |       , -- | Nominal deadline until which contest can be submitted and after | ||||||
|  |         -- which fanout is possible. NOTE: Use this only for informational | ||||||
|  |         -- purpose and wait for 'ReadyToFanout' instead before sending 'Fanout' | ||||||
|  |         -- as the ledger of our cardano-node might not have progressed | ||||||
|  |         -- sufficiently in time yet and we do not re-submit transactions (yet). | ||||||
|  |         contestationDeadline :: UTCTime | ||||||
|  |       } | ||||||
|  |   | HeadIsContested {snapshotNumber :: SnapshotNumber} | ||||||
|  |   | ReadyToFanout | ||||||
|  |   | HeadIsAborted {utxo :: WholeUTXO} | ||||||
|  |   | HeadIsFinalized {utxo :: WholeUTXO} | ||||||
|  |   | CommandFailed {clientInput :: ClientInput} | ||||||
|  |   | TxSeen {transaction :: tx} | ||||||
|  |   | TxValid {transaction :: tx} | ||||||
|  |   | TxInvalid {utxo :: WholeUTXO, transaction :: tx, validationError :: ValidationError} | ||||||
|  |   | SnapshotConfirmed | ||||||
|  |       { snapshot :: Snapshot tx | ||||||
|  |       , signatures :: MultiSignature (Snapshot tx) | ||||||
|  |       } | ||||||
|  |   | GetUTxOResponse {utxo :: WholeUTXO} | ||||||
|  |   | InvalidInput {reason :: String, input :: Text} | ||||||
|  |   | -- | A friendly welcome message which tells a client something about the | ||||||
|  |     -- node. Currently used for knowing what signing key the server uses (it | ||||||
|  |     -- only knows one). | ||||||
|  |     Greetings {me :: Party} | ||||||
|  |   | PostTxOnChainFailed {postChainTx :: PostChainTx tx, postTxError :: PostTxError tx} | ||||||
|  |   | RolledBack | ||||||
|  |   deriving (Eq, Show, Generic) | ||||||
|  | 
 | ||||||
|  | instance (ToJSON tx) => ToJSON (ServerOutput tx) | ||||||
|  | instance (FromJSON tx) => FromJSON (ServerOutput tx) | ||||||
							
								
								
									
										63
									
								
								src/Hydra/Types.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										63
									
								
								src/Hydra/Types.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,63 @@ | |||||||
|  | -- |  | ||||||
|  | 
 | ||||||
|  | module Hydra.Types where | ||||||
|  | 
 | ||||||
|  | import GHC.Generics | ||||||
|  | import Data.Aeson | ||||||
|  | 
 | ||||||
|  | import Data.Map (Map) | ||||||
|  | import qualified Data.Text as T | ||||||
|  | import Data.Text (Text) | ||||||
|  | import Numeric.Natural (Natural) | ||||||
|  | 
 | ||||||
|  | -- | Cardano address | ||||||
|  | type Address = String | ||||||
|  | 
 | ||||||
|  | type Lovelace = Int | ||||||
|  | 
 | ||||||
|  | type TxIn = T.Text | ||||||
|  | type WholeUTXO = Map TxIn TxInInfo | ||||||
|  | 
 | ||||||
|  | data TxInInfo = TxInInfo | ||||||
|  |   { address :: Address | ||||||
|  |   , datumhash :: Maybe T.Text | ||||||
|  |   , value :: Map T.Text Int | ||||||
|  |   } | ||||||
|  |   deriving (Eq, Show, Generic) | ||||||
|  | 
 | ||||||
|  | instance FromJSON TxInInfo | ||||||
|  | instance ToJSON TxInInfo | ||||||
|  | 
 | ||||||
|  | fuelMarkerDatumHash :: T.Text | ||||||
|  | fuelMarkerDatumHash = "a654fb60d21c1fed48db2c320aa6df9737ec0204c0ba53b9b94a09fb40e757f3" | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | -- REVIEW(SN): This is also used in hydra-tui | ||||||
|  | 
 | ||||||
|  | -- Below various types copied/adapted from hydra-poc code | ||||||
|  | 
 | ||||||
|  | data Host = Host | ||||||
|  |   { hostname :: Text | ||||||
|  |   , port :: PortNumber | ||||||
|  |   } | ||||||
|  |   deriving (Ord, Generic, Eq, Show) | ||||||
|  | 
 | ||||||
|  | instance ToJSON Host | ||||||
|  | instance FromJSON Host | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | newtype Party = Party {vkey :: T.Text} | ||||||
|  |   deriving (Eq, Show, Read, Generic) | ||||||
|  | 
 | ||||||
|  | instance ToJSON Party | ||||||
|  | instance FromJSON Party | ||||||
|  | 
 | ||||||
|  | type UTxOType tx = Value | ||||||
|  | type Snapshot tx = Value | ||||||
|  | type MultiSignature x = Value | ||||||
|  | type PostChainTx tx = Value  | ||||||
|  | type PostTxError tx = Value | ||||||
|  | type ValidationError = Value | ||||||
|  | type SnapshotNumber = Natural | ||||||
|  | type PortNumber = Natural | ||||||
|  | type ContestationPeriod = Natural | ||||||
							
								
								
									
										650
									
								
								src/Main.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										650
									
								
								src/Main.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,650 @@ | |||||||
|  | {-# LANGUAGE TemplateHaskell #-} | ||||||
|  | {-# LANGUAGE PartialTypeSignatures #-} | ||||||
|  | {-# OPTIONS_GHC -Wno-unused-do-bind #-} | ||||||
|  | {-# LANGUAGE TupleSections #-} | ||||||
|  | {-# LANGUAGE ConstraintKinds #-} | ||||||
|  | {-# OPTIONS_GHC -Wno-unrecognised-pragmas #-} | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | module Main | ||||||
|  | 
 | ||||||
|  | (main) | ||||||
|  | 
 | ||||||
|  | where | ||||||
|  | 
 | ||||||
|  | import Prelude hiding (filter) | ||||||
|  | 
 | ||||||
|  | import Hydra.Devnet | ||||||
|  | 
 | ||||||
|  | import Control.Monad | ||||||
|  | 
 | ||||||
|  | import System.Directory | ||||||
|  | import Control.Monad.IO.Class (MonadIO, liftIO) | ||||||
|  | 
 | ||||||
|  | import qualified Data.Map as Map | ||||||
|  | import Data.Map (Map, (!)) | ||||||
|  | import Data.Witherable | ||||||
|  | import Data.String.Interpolate ( i, iii, __i ) | ||||||
|  | import qualified Data.Text as T | ||||||
|  | 
 | ||||||
|  | import Control.Concurrent | ||||||
|  | import System.Process | ||||||
|  | 
 | ||||||
|  | import Data.Aeson as Aeson | ||||||
|  |     ( decode, (.:), withObject, Value ) | ||||||
|  | 
 | ||||||
|  | import Data.Aeson.Text (encodeToTextBuilder) | ||||||
|  | import Data.Text.Lazy (toStrict) | ||||||
|  | import Data.Text.Lazy.Builder (toLazyText) | ||||||
|  | 
 | ||||||
|  | import qualified Data.Map.Merge.Lazy as Map | ||||||
|  | import qualified Hydra.Types as HT | ||||||
|  | import Data.Maybe (fromJust, fromMaybe) | ||||||
|  | import Data.Aeson.Types (parseMaybe) | ||||||
|  | import System.IO (IOMode(WriteMode), openFile) | ||||||
|  | import Data.IORef (readIORef, writeIORef, IORef, newIORef) | ||||||
|  | import Hydra.Types | ||||||
|  | import Data.Text (Text) | ||||||
|  | import qualified Data.ByteString.Lazy.Char8 as ByteString.Char8 | ||||||
|  | import Data.Time (UTCTime, diffUTCTime) | ||||||
|  | 
 | ||||||
|  | import Reflex | ||||||
|  | import Reflex.Dom | ||||||
|  | import Control.Monad.Fix | ||||||
|  | import Hydra.ClientInput | ||||||
|  | import Hydra.ServerOutput | ||||||
|  | import Data.Bool (bool) | ||||||
|  | import Text.Read (readMaybe) | ||||||
|  | import Data.Traversable (for) | ||||||
|  | import Data.Semigroup (First(getFirst, First)) | ||||||
|  | import Data.Aeson | ||||||
|  | 
 | ||||||
|  | import Control.Monad.Trans (lift) | ||||||
|  | import Language.Javascript.JSaddle.Types ( MonadJSM ) | ||||||
|  | 
 | ||||||
|  | standupDemoHydraNetwork :: (MonadIO m) | ||||||
|  |   => HydraScriptTxId | ||||||
|  |   -> Map Text HydraKeyInfo | ||||||
|  |   -> m (Map Text (ProcessHandle, HydraNodeInfo)) | ||||||
|  | standupDemoHydraNetwork hstxid actors = do | ||||||
|  |   liftIO $ createDirectoryIfMissing True "demo-logs" | ||||||
|  |   liftIO $ sequence . flip Map.mapWithKey nodes $ \name node'' -> do | ||||||
|  |     logHndl <- openFile [iii|demo-logs/hydra-node-#{name}.log|] WriteMode | ||||||
|  |     errHndl <- openFile [iii|demo-logs/phydra-node-#{name}.error.log|] WriteMode | ||||||
|  |     let cp = (mkHydraNodeCP sharedInfo node'' (filter ((/= _nodeId node'') . _nodeId) (Map.elems nodes))) | ||||||
|  |              { std_out = UseHandle logHndl | ||||||
|  |              , std_err = UseHandle errHndl | ||||||
|  |              } | ||||||
|  |     (_,_,_,handle) <- createProcess cp | ||||||
|  |     pure (handle, node'') | ||||||
|  |   where | ||||||
|  |     portNum p n = p * 1000 + n | ||||||
|  |     node' (n, (name, keys)) = | ||||||
|  |       ( name | ||||||
|  |       , HydraNodeInfo n (portNum 5 n) (portNum 9 n) (portNum 6 n) keys | ||||||
|  |       ) | ||||||
|  |     nodes = Map.fromList . fmap node' $ zip [1 ..] (Map.toList actors) | ||||||
|  |     sharedInfo = HydraSharedInfo | ||||||
|  |       { _hydraScriptsTxId = T.unpack hstxid | ||||||
|  |       , _ledgerGenesis = "devnet/genesis-shelley.json" | ||||||
|  |       , _ledgerProtocolParameters = "devnet/protocol-parameters.json" | ||||||
|  |       , _networkId = show devnetMagic | ||||||
|  |       , _nodeSocket = "devnet/node.socket" | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  | -- | Takes the node participant and the list of peers | ||||||
|  | mkHydraNodeCP :: HydraSharedInfo -> HydraNodeInfo -> [HydraNodeInfo] -> CreateProcess | ||||||
|  | mkHydraNodeCP sharedInfo node peers = | ||||||
|  |   (proc hydraNodePath $ sharedArgs sharedInfo <> nodeArgs node <> concatMap peerArgs peers) | ||||||
|  |   { std_out = Inherit | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | data HydraSharedInfo = HydraSharedInfo | ||||||
|  |   { _hydraScriptsTxId :: String | ||||||
|  |   , _ledgerGenesis :: FilePath | ||||||
|  |   , _ledgerProtocolParameters :: FilePath | ||||||
|  |   , _networkId :: String | ||||||
|  |   , _nodeSocket :: FilePath | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | data HydraNodeInfo = HydraNodeInfo | ||||||
|  |   { _nodeId :: Int | ||||||
|  |   , _port :: Int | ||||||
|  |   , _apiPort :: Int | ||||||
|  |   , _monitoringPort :: Int | ||||||
|  |   , _keys :: HydraKeyInfo | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  | sharedArgs :: HydraSharedInfo -> [String] | ||||||
|  | sharedArgs (HydraSharedInfo hydraScriptsTxId ledgerGenesis protocolParams networkId nodeSocket) = | ||||||
|  |   [ "--ledger-genesis" | ||||||
|  |   , ledgerGenesis | ||||||
|  |   , "--ledger-protocol-parameters" | ||||||
|  |   , protocolParams | ||||||
|  |   , "--network-id" | ||||||
|  |   , networkId | ||||||
|  |   , "--node-socket" | ||||||
|  |   , nodeSocket | ||||||
|  |   , "--hydra-scripts-tx-id" | ||||||
|  |   , hydraScriptsTxId | ||||||
|  |   ] | ||||||
|  | 
 | ||||||
|  | nodeArgs :: HydraNodeInfo -> [String] | ||||||
|  | nodeArgs (HydraNodeInfo nodeId port' apiPort monitoringPort | ||||||
|  |            (HydraKeyInfo | ||||||
|  |             (KeyPair cskPath _cvkPath) | ||||||
|  |             (KeyPair hskPath _hvkPath))) = | ||||||
|  |   [ "--node-id" | ||||||
|  |   , show nodeId | ||||||
|  |   , "--port" | ||||||
|  |   , show port' | ||||||
|  |   , "--api-port" | ||||||
|  |   , show apiPort | ||||||
|  |   , "--monitoring-port" | ||||||
|  |   , show monitoringPort | ||||||
|  |   , "--hydra-signing-key" | ||||||
|  |   , hskPath | ||||||
|  |   , "--cardano-signing-key" | ||||||
|  |   , cskPath | ||||||
|  |   ] | ||||||
|  | 
 | ||||||
|  | peerArgs :: HydraNodeInfo -> [String] | ||||||
|  | peerArgs ni = | ||||||
|  |   [ "--peer" | ||||||
|  |   , [i|127.0.0.1:#{_port ni}|] | ||||||
|  |   , "--hydra-verification-key" | ||||||
|  |   , _verificationKey . _hydraKeys . _keys $ ni | ||||||
|  |   , "--cardano-verification-key" | ||||||
|  |   , _verificationKey . _cardanoKeys . _keys $ ni | ||||||
|  |   ] | ||||||
|  | 
 | ||||||
|  | cardanoNodeCreateProcess :: CreateProcess | ||||||
|  | cardanoNodeCreateProcess = | ||||||
|  |   (proc cardanoNodePath | ||||||
|  |    [ "run" | ||||||
|  |    , "--config" | ||||||
|  |    , "devnet/cardano-node.json" | ||||||
|  |    , "--topology" | ||||||
|  |    , "devnet/topology.json" | ||||||
|  |    , "--database-path" | ||||||
|  |    , "devnet/db" | ||||||
|  |    , "--socket-path" | ||||||
|  |    , "devnet/node.socket" | ||||||
|  |    , "--shelley-operational-certificate" | ||||||
|  |    , "devnet/opcert.cert" | ||||||
|  |    , "--shelley-kes-key" | ||||||
|  |    , "devnet/kes.skey" | ||||||
|  |    , "--shelley-vrf-key" | ||||||
|  |    , "devnet/vrf.skey" | ||||||
|  |    ]) { std_out = CreatePipe | ||||||
|  |       } | ||||||
|  | 
 | ||||||
|  | runHydraDemo :: (MonadIO m) | ||||||
|  |   => HydraDemo | ||||||
|  |   -> m (Map Text ( ProcessHandle | ||||||
|  |                    , Address -- Cardano address | ||||||
|  |                    , HydraNodeInfo | ||||||
|  |                    )) | ||||||
|  | runHydraDemo nodes = do | ||||||
|  |   keysAddresses <- forM nodes $ \(actorSeed, fuelSeed) -> do | ||||||
|  |     keys@(HydraKeyInfo (KeyPair _ vk) _) <- generateKeys | ||||||
|  |     addr <- liftIO $ getCardanoAddress vk | ||||||
|  |     void $ seedAddressFromFaucetAndWait addr actorSeed False | ||||||
|  |     void $ seedAddressFromFaucetAndWait addr fuelSeed True | ||||||
|  |     pure (keys, addr) | ||||||
|  |   liftIO . putStrLn $ "Publishing reference scripts" | ||||||
|  |   hstxid <- publishReferenceScripts | ||||||
|  |   handles <- standupDemoHydraNetwork hstxid (fmap fst keysAddresses) | ||||||
|  |   liftIO . putStrLn $ [i|"Hydra Network Running for nodes #{Map.keys nodes}|] | ||||||
|  |   pure $ Map.merge Map.dropMissing Map.dropMissing (Map.zipWithMatched (\_ addr (handle, nodeInfo) -> (handle, addr, nodeInfo))) (fmap snd keysAddresses) handles | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | type State = Map Text ( ProcessHandle | ||||||
|  |                              , Address -- Cardano address | ||||||
|  |                              , HydraNodeInfo | ||||||
|  |                              ) | ||||||
|  | 
 | ||||||
|  | headElement :: forall t m. ( TriggerEvent t m, DomBuilder t m) =>m () | ||||||
|  | headElement = do | ||||||
|  |     el "title" $ text "Hydra Head Demo" | ||||||
|  |     elAttr "script" ("src"=:"https://cdn.tailwindcss.com") blank | ||||||
|  | 
 | ||||||
|  | main :: IO () | ||||||
|  | main =  liftIO $ do | ||||||
|  |  prepareDevnet | ||||||
|  |  withCreateProcess cardanoNodeCreateProcess $ \_ _stdout _ _handle -> do | ||||||
|  |    putStrLn "Devnet is running" | ||||||
|  |    threadDelay $ seconds 3 | ||||||
|  |    mainWidgetWithHead headElement app | ||||||
|  | 
 | ||||||
|  | makeTx :: () => IORef State -> Text | ||||||
|  |                   -> Map TxIn TxInInfo -> Lovelace -> Text -> IO Text | ||||||
|  | makeTx hydraProcessHandlesRef fromName utxos lovelace toName = do | ||||||
|  |   print (fromName, utxos, toName) | ||||||
|  |   let lovelaceUtxos = mapMaybe (Map.lookup "lovelace" . HT.value) utxos | ||||||
|  |   actors <- readIORef hydraProcessHandlesRef | ||||||
|  |   jsonStr <- | ||||||
|  |     buildSignedHydraTx | ||||||
|  |       (_signingKey . _cardanoKeys . _keys . (\(_, _, hn) -> hn) $ actors ! fromName) | ||||||
|  |       ((\(_, addr, _) -> addr) $ actors ! fromName) | ||||||
|  |       ((\(_, addr, _) -> addr) $ actors ! toName) | ||||||
|  |       lovelaceUtxos | ||||||
|  |       lovelace | ||||||
|  |   let jsonTx :: Aeson.Value = fromMaybe (error "Failed to parse TX") . Aeson.decode . ByteString.Char8.pack $ jsonStr | ||||||
|  |   pure . fromJust . parseMaybe (withObject "signed tx" (.: "cborHex")) $ jsonTx | ||||||
|  | 
 | ||||||
|  | startDemo :: MonadIO m => IORef State -> HydraDemo -> m RunningNodes | ||||||
|  | startDemo hydraProcessHandlesRef demo = do | ||||||
|  |   liftIO (mapM (terminateProcess . (\(hndl, _, _) -> hndl)) =<< readIORef hydraProcessHandlesRef) | ||||||
|  |   nodeInfos <- runHydraDemo demo | ||||||
|  |   liftIO . writeIORef hydraProcessHandlesRef $ nodeInfos | ||||||
|  |   actorList :: RunningNodes <- forM nodeInfos $ \(_, addr, nInfo) -> do | ||||||
|  |     pure | ||||||
|  |       ( addr, | ||||||
|  |         [iii|ws://localhost:#{_apiPort nInfo}|] | ||||||
|  |       ) | ||||||
|  |   pure actorList | ||||||
|  | 
 | ||||||
|  | -- | Friendly name for a Hydra node. | ||||||
|  | type DemoNodeName = Text | ||||||
|  | 
 | ||||||
|  | -- | WebSocket URL | ||||||
|  | type ApiUrl = Text | ||||||
|  | 
 | ||||||
|  | type RunningNodes = Map DemoNodeName ( Address -- Cardano address | ||||||
|  |                                      , ApiUrl | ||||||
|  |                                      ) | ||||||
|  | 
 | ||||||
|  | type HydraDemo =  Map | ||||||
|  |                   DemoNodeName | ||||||
|  |                   ( Lovelace -- Seed for actor | ||||||
|  |                   , Lovelace -- Seed for fuel | ||||||
|  |                   ) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | seconds :: Int -> Int | ||||||
|  | seconds = (* 1000000) | ||||||
|  | 
 | ||||||
|  | alicebobcarolDemo :: HydraDemo | ||||||
|  | alicebobcarolDemo = Map.fromList [("Alice", (1000000000, 100000000)), ("Bob", (500000000, 100000000)), ("Carol", (250000000, 100000000))] | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | filterOutFuel :: WholeUTXO -> WholeUTXO | ||||||
|  | filterOutFuel = Map.filter (not . isFuel) | ||||||
|  | 
 | ||||||
|  | isFuel :: TxInInfo -> Bool | ||||||
|  | isFuel txinfo = datumhash txinfo == Just fuelMarkerDatumHash | ||||||
|  | 
 | ||||||
|  | -- | Tracks the state of the head based on Hydra Node responses | ||||||
|  | data HeadState | ||||||
|  |   = Idle | ||||||
|  |   | Initializing | ||||||
|  |   | Open | ||||||
|  |   | Closed UTCTime | ||||||
|  |   | StateReadyToFanout | ||||||
|  |   deriving (Eq, Show) | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | buttonClass :: (PostBuild t m, DomBuilder t m) => Dynamic t T.Text -> m b -> m (Event t ()) | ||||||
|  | buttonClass cls content = do | ||||||
|  |   (buttonEl, _) <- elDynClass' "button" cls content | ||||||
|  |   pure $ domEvent Click buttonEl | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | utxoPicker :: forall t m. (DomBuilder t m, MonadFix m, MonadHold t m, PostBuild t m) => Bool -> WholeUTXO -> m (Dynamic t (Maybe WholeUTXO)) | ||||||
|  | utxoPicker pickable wholeUtxo = mdo | ||||||
|  |   elClass "div" "font-semibold text-lg mb-2" $ text "UTxOs" | ||||||
|  | 
 | ||||||
|  |   currentUtxo <- holdDyn Nothing selectedUtxo | ||||||
|  |   selectedUtxo <- fmap (leftmost . Map.elems) $ elClass "div" "flex flex-row flex-wrap gap-2" $ flip Map.traverseWithKey wholeUtxo $ \k v -> mdo | ||||||
|  |       let amiSelected = maybe False ((k ==) . fst) <$> currentUtxo | ||||||
|  |       let cls = ("text-white font-bold text-xl px-4 py-2 rounded-md flex flex-row cursor-pointer mr-2 " <>) | ||||||
|  |             . bool | ||||||
|  |               "bg-gray-500 hover:bg-gray-400 active:bg-gray-300" | ||||||
|  |               "bg-blue-500 hover:bg-blue-400 active:bg-blue-300" | ||||||
|  |             <$> amiSelected | ||||||
|  |       (buttonEl, _) <- elDynClass' "button" cls $ do | ||||||
|  |         elClass "div" "text-sm text-gray-300 font-semibold flex justify-between" $ do | ||||||
|  |           elClass "div" "flex flex-col" $ do | ||||||
|  |             elClass "div" "w-full flex flex-row justify-between" $ do | ||||||
|  |               elClass "div" "text-gray-400 mr-4" $ text "lovelace" | ||||||
|  |               when (isFuel v) $ elClass "div" "px-2 py-0 flex items-center justify-center leading-node bg-green-500 text-xs text-white font-semibold text-sm rounded-full flex" $ | ||||||
|  |                 el "div" $ text "FUEL" | ||||||
|  |             elClass "div" "text-lg text-left font-semibold" $ text $ maybe "" (T.pack . show) (Map.lookup "lovelace" $ HT.value v) | ||||||
|  | 
 | ||||||
|  |       pure $ bool Nothing (Just (k, v)) . (pickable &&) . not <$> current amiSelected <@ domEvent Click buttonEl | ||||||
|  |   pure $ fmap (uncurry Map.singleton) <$> currentUtxo | ||||||
|  | 
 | ||||||
|  | demoSettings :: (DomBuilder t m, PostBuild t m, MonadHold t m, MonadFix m) => HydraDemo -> m (Dynamic t HydraDemo) | ||||||
|  | demoSettings setngs = elClass "div" "flex flex-col pl-4 pr-4" $ do | ||||||
|  |   let initSize = Map.size setngs | ||||||
|  |   let initialList = Map.fromList (zip [1 .. ] (Map.toList setngs)) | ||||||
|  |   rec | ||||||
|  |     nextIdentityNumber <- fmap (1 + initSize +) <$> count newNode | ||||||
|  |     let updates = | ||||||
|  |           ((\n -> Map.singleton n (Just ([i|Node #{n}|], (100000000, 100000000))))  <$> current nextIdentityNumber <@ newNode) | ||||||
|  |           <> deleteEs | ||||||
|  |     (((), deleteEs), demoDyn) <- runDynamicWriterT $ runEventWriterT $ void $ elClass "div" "flex-col space-y-2" $ do | ||||||
|  |       elClass "p" "text-white text-2xl my-4" $ text "Configure a Hydra Head by specifying the node names and their initial funds in Lovelace." | ||||||
|  |       listHoldWithKey initialList updates $ \k (name, (actorSeed, _hydraSeed)) -> elClass "div" "flex flex-col" $ do | ||||||
|  |         name' <- elClass "div" "flex flex-row space-x-2" $ do | ||||||
|  |           name' <- fmap _inputElement_value . inputElement $ | ||||||
|  |             def & inputElementConfig_initialValue .~ name | ||||||
|  |             & initialAttributes .~ ("class" =: "text-white bg-gray-800 text-2xl font-bold focus:outline-none p-2" <> "type" =: "text") | ||||||
|  |           amount' <- fmap _inputElement_value . inputElement $ | ||||||
|  |             def & inputElementConfig_initialValue .~ (T.pack . show $ actorSeed) | ||||||
|  |             & initialAttributes .~ ("class" =: "text-white bg-gray-800 text-2xl font-bold focus:outline-none p-2" <> "type" =: "number") | ||||||
|  |           deleteE <- buttonClass "bg-gray-400 hover:bg-gray-300 active:bg-gray-200 text-white font-bold text-xl px-4 py-2 rounded-md" $ | ||||||
|  |                      text "×" | ||||||
|  |           let actorSeed' = (\n a -> (n,) <$> readMaybe (T.unpack a)) <$> name' <*> amount' | ||||||
|  |           tellDyn (maybe mempty (Map.singleton k. (\(actor,sd) -> (actor, (sd, 100000000)))) <$> actorSeed') | ||||||
|  |           tellEvent (Map.singleton k Nothing <$ deleteE) | ||||||
|  |           pure name' | ||||||
|  |         let hasDuplicateName = (\n ns -> (> 1) . Map.size . Map.filter (\(n',_) -> n == n') $ ns) <$> name' <*> demoDyn | ||||||
|  |         let duplicateNameMsg = elClass "div" "text-red-400 m-2" $ text "Duplicate name" | ||||||
|  |         dyn_ (bool blank duplicateNameMsg <$> fromUniqDynamic (uniqDynamic hasDuplicateName)) | ||||||
|  |         pure () | ||||||
|  |     newNode <- buttonClass "bg-gray-400 hover:bg-gray-300 active:bg-gray-200 text-white font-bold text-xl my-4 px-4 py-2 rounded-md w-32" $ | ||||||
|  |              text "Add node" | ||||||
|  |     let demoDyn' = Map.fromList . Map.elems <$> demoDyn | ||||||
|  |   pure demoDyn' | ||||||
|  | 
 | ||||||
|  | startStopDemoControls :: | ||||||
|  |   ( DomBuilder t m, | ||||||
|  |     MonadFix m, | ||||||
|  |     PostBuild t m, | ||||||
|  |     MonadHold t m, MonadIO (Performable m), PerformEvent t m) => | ||||||
|  |   IORef State -> | ||||||
|  |   m (Event t RunningNodes) | ||||||
|  | startStopDemoControls hydraProcessHandlesRef = mdo | ||||||
|  |   headRunning <- toggle False headStartedOrStoppedE | ||||||
|  |   ((), demoConfig) <- runDynamicWriterT $ dyn_ (bool (tellDyn =<< demoSettings alicebobcarolDemo) blank <$> headRunning) | ||||||
|  |   startStopHeadE <- buttonClass ((\running -> | ||||||
|  |                                     let color :: Text = bool "green" "red" running | ||||||
|  |                                     in [__i|bg-#{color}-500 hover:bg-#{color}-400 active:bg-#{color}-300 | ||||||
|  |                                            text-white font-bold text-xl m-4 px-4 py-2 rounded-md|] | ||||||
|  |                                       :: Text) | ||||||
|  |                                   <$> headRunning) | ||||||
|  |                     $ dynText (bool "Start head" "Stop head" <$> headRunning) | ||||||
|  |   let startStopWithConfE = current demoConfig <@ startStopHeadE | ||||||
|  |   headStartedOrStoppedE <- performEvent $ | ||||||
|  |     -- Start with mempty to stop the demo: | ||||||
|  |     (\running conf -> startDemo hydraProcessHandlesRef $ bool conf mempty running) | ||||||
|  |     <$> current headRunning | ||||||
|  |     <@> startStopWithConfE | ||||||
|  |   let headStartingDom conf = | ||||||
|  |         if Map.null conf | ||||||
|  |         then blank | ||||||
|  |         else elClass "div" "text-white text-2xl m-4" $ text "Head starting..." | ||||||
|  |   void $ runWithReplace blank $ leftmost [ headStartingDom <$> startStopWithConfE | ||||||
|  |                                          , blank <$ headStartedOrStoppedE | ||||||
|  |                                          ] | ||||||
|  |   pure headStartedOrStoppedE | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | app :: | ||||||
|  |   forall t m. | ||||||
|  |   ( PostBuild t m, | ||||||
|  |     DomBuilder t m, | ||||||
|  |     MonadFix m, | ||||||
|  |     MonadJSM m, MonadJSM (Performable m), | ||||||
|  |     MonadHold t m, PerformEvent t m, TriggerEvent t m) => | ||||||
|  |   m () | ||||||
|  | app = do | ||||||
|  |  hydraProcessHandlesRef :: IORef State <- liftIO (newIORef mempty) | ||||||
|  |  elClass "div" "w-screen h-screen bg-gray-900 overflow-y-scroll overflow-x-hidden" $ do | ||||||
|  |   elClass "div" "p-4 m-4 text-white text-5xl font-bold" $ text "Hydra Proof Of Concept Demo" | ||||||
|  |   mdo | ||||||
|  |     headStartedE <- startStopDemoControls hydraProcessHandlesRef | ||||||
|  |     void $ runWithReplace blank $ ffor headStartedE $ \actors -> mdo | ||||||
|  |       let actorNames = ffor (Map.toList actors) $ \(name, (_,_)) -> name | ||||||
|  |       headState <- holdDyn Idle newState | ||||||
|  |       let headStateDom = elClass "div" "text-lg" . text . ("Head State: " <>) | ||||||
|  |       unless (null actors) $ elClass "div" "ml-4 mt-8 mr-4 mb-2 w-full font-black text-green-500" $ dyn_ $ ffor headState $ \case | ||||||
|  |         Idle -> do | ||||||
|  |           headStateDom "Idle" | ||||||
|  |           elClass "div" "text-green-700 text-sm" $ text "Waiting for participant to init..." | ||||||
|  |         Initializing -> do | ||||||
|  |           headStateDom "Initializing" | ||||||
|  |           elClass "div" "text-green-700 text-sm" $ text $ "Waiting for commits from: " <> T.intercalate ", " actorNames | ||||||
|  |         Open -> headStateDom "Open" | ||||||
|  |         Closed _ -> headStateDom "Closed/Contestation period" | ||||||
|  |         StateReadyToFanout -> headStateDom "Ready to fanout" | ||||||
|  | 
 | ||||||
|  |       newState <- elClass "div" "ml-4 mr-4 overflow-hidden rounded-lg hover:drop-shadow-xl transition-all drop-shadow bg-gray-800" $ mdo | ||||||
|  |         rec | ||||||
|  |           currentTab <- holdDyn (head actorNames) changeTab | ||||||
|  | 
 | ||||||
|  |           changeTab <- fmap leftmost $ elClass "div" "w-full flex flex-row justify-start" $ for actorNames $ \name -> do | ||||||
|  |             let | ||||||
|  |               isSelected = (== name) <$> currentTab | ||||||
|  |               mkClasses selected = | ||||||
|  |                 T.intercalate " " [ "leading-none p-4 font-bold text-2xl text-gray-100 flex items-center justify-center" | ||||||
|  |                                   , bool "bg-gray-800 text-gray-300 pointer-cursor" "bg-gray-700 text-gray-100" selected | ||||||
|  |                                   ] | ||||||
|  |             (buttonEl, _) <- elDynClass' "button" (mkClasses <$> isSelected) $ text name | ||||||
|  |             pure $ name <$ domEvent Click buttonEl | ||||||
|  |         fmap (fmap getFirst . snd) . runEventWriterT $ forM (Map.toList actors) $ \(name, (actorAddress, wsUrl)) -> mdo | ||||||
|  |               let wsCfg = (WebSocketConfig @t @ClientInput) action never True [] | ||||||
|  |               ws <- jsonWebSocket wsUrl wsCfg | ||||||
|  |               let isSelected = (== name) <$> currentTab | ||||||
|  |               let mkClasses selected = | ||||||
|  |                      T.intercalate " " [ "p-2 bg-gray-700 text-white flex flex-col items-left" | ||||||
|  |                                        , bool "hidden" "" selected | ||||||
|  |                                        ] | ||||||
|  |               (_, action) <- elDynClass "div" (mkClasses <$> isSelected) $ runEventWriterT $ runWithReplace (elClass "div" "text-white" $ text "Connecting to node...") . ffor (_webSocket_open ws) $ \() -> do | ||||||
|  |                 let | ||||||
|  |                   webSocketMessage :: Event t (ServerOutput Aeson.Value) = | ||||||
|  |                     fromMaybe (error "Parsing message from Hydra node failed") <$> _webSocket_recv ws | ||||||
|  |                   processLog = \case | ||||||
|  |                     ReadyToCommit {} -> Just Initializing | ||||||
|  |                     HeadIsOpen {} -> Just Open | ||||||
|  |                     HeadIsClosed _ fanoutTime -> Just (Closed fanoutTime) | ||||||
|  |                     ReadyToFanout {} -> Just StateReadyToFanout | ||||||
|  |                     HeadIsAborted {} -> Just Idle | ||||||
|  |                     HeadIsFinalized {} -> Just Idle | ||||||
|  |                     _ -> Nothing | ||||||
|  |                 let stateChange = fmapMaybe processLog webSocketMessage | ||||||
|  |                 let | ||||||
|  | 
 | ||||||
|  |                 myVKeyB :: Behavior t (Maybe T.Text) <- | ||||||
|  |                    hold Nothing | ||||||
|  |                      . fmap Just | ||||||
|  |                      . mapMaybe (\case | ||||||
|  |                                     Greetings (Party vkey') -> Just vkey' | ||||||
|  |                                     _ -> Nothing) | ||||||
|  |                      $ webSocketMessage | ||||||
|  |                 headStateE <- mdo | ||||||
|  |                   void $ dyn $ ffor headState $ \case | ||||||
|  |                     Idle -> idleScreen name | ||||||
|  |                     Initializing -> initializingScreen actorAddress myVKeyB webSocketMessage | ||||||
|  |                     Open -> openScreen hydraProcessHandlesRef name actorNames actorAddress webSocketMessage | ||||||
|  |                     Closed fanoutTime -> closedScreen fanoutTime | ||||||
|  |                     StateReadyToFanout -> | ||||||
|  |                       tellAction | ||||||
|  |                         . (Fanout <$) | ||||||
|  |                         <=< buttonClass "bg-green-400 hover:bg-green-400 active:bg-green-200 text-white font-bold text-xl my-2 px-4 py-2 rounded-md w-32" $ text "Do fanout" | ||||||
|  |                   elClass "div" "mt-4" $ do | ||||||
|  |                     elClass "div" "mb-1 font-semibold text-sm" $ text "Hydra Node Log" | ||||||
|  |                     elClass "div" "p-2 bg-gray-800 rounded-md drop-shadow" $ | ||||||
|  |                       el "ul" $ do | ||||||
|  |                       comms <- foldDyn (++) [] $ | ||||||
|  |                         ((:[]) . ("Rcv: " <>) . toStrict . toLazyText . encodeToTextBuilder . toJSON <$> webSocketMessage) | ||||||
|  |                         <> | ||||||
|  |                         fmap (fmap (("Snd: " <>) . toStrict . toLazyText . encodeToTextBuilder . toJSON)) action | ||||||
|  |                       dyn_ $ mapM (el "li" . text) <$> comms | ||||||
|  |                   pure stateChange | ||||||
|  |                 lift $ tellEvent (First <$> headStateE) | ||||||
|  |               pure () | ||||||
|  |       pure () | ||||||
|  | 
 | ||||||
|  | filterUtxos :: Address -> WholeUTXO -> WholeUTXO | ||||||
|  | filterUtxos addr = Map.filter ((== addr) . HT.address) | ||||||
|  | 
 | ||||||
|  | tellAction :: (EventWriter t [a] m, Reflex t) => Event t a -> m () | ||||||
|  | tellAction = tellEvent . fmap (:[]) | ||||||
|  | 
 | ||||||
|  | idleScreen :: (EventWriter t [ClientInput] m, DomBuilder t m) => Text -> m () | ||||||
|  | idleScreen name = | ||||||
|  |   elClass "div" "p-2 flex flex-row" $ do | ||||||
|  |     (buttonEl, _) <- elClass' "button" "bg-blue-500 hover:bg-blue-400 active:bg-blue-300 text-white font-bold text-xl px-4 py-2 rounded-md" $ text $ "Initialize head as " <> name | ||||||
|  |     tellAction $ Init 10 <$ domEvent Click buttonEl | ||||||
|  | 
 | ||||||
|  | initializingScreen :: | ||||||
|  |   ( EventWriter t [ClientInput] m, | ||||||
|  |     DomBuilder t m, | ||||||
|  |     MonadFix m, | ||||||
|  |     MonadHold t m, | ||||||
|  |     PostBuild t m, MonadIO m) => | ||||||
|  |   Address -> | ||||||
|  |   Behavior t (Maybe Text) -> | ||||||
|  |   Event t (ServerOutput tx) -> | ||||||
|  |   m () | ||||||
|  | initializingScreen actorAddress myVKeyB webSocketMessage = do | ||||||
|  |   elClass "div" "p-2 flex flex-col" $ do | ||||||
|  |     -- TODO: did not use performEvent here | ||||||
|  |     newUTXOs <- liftIO $ queryAddressUTXOs actorAddress -- fmapMaybe eitherToMaybe <$> (undefined . (DemoApi_GetActorUTXO actorAddress <$) =<< getPostBuild) | ||||||
|  |     let commitSelection doCommit = do | ||||||
|  |           (_, currentSet) <- | ||||||
|  |             runDynamicWriterT $ (tellDyn <=< utxoPicker True) newUTXOs | ||||||
|  | --              runWithReplace (elClass "div" "p-4 bg-gray-800 rounded mb-2" $ text $ "Getting " <> name <> "'s UTXOs...") $ | ||||||
|  | --                (tellDyn <=< utxoPicker True) <$> newUTXOs | ||||||
|  |           tellAction $ fmap (Commit . fromMaybe mempty) $ current currentSet <@ doCommit | ||||||
|  |     let hasCommitted = | ||||||
|  |           attachWithMaybe | ||||||
|  |             ( \mvkey -> \case | ||||||
|  |                 Committed (Party vk) _ -> guard (Just vk == mvkey) | ||||||
|  |                 _ -> Nothing | ||||||
|  |             ) | ||||||
|  |             myVKeyB | ||||||
|  |             webSocketMessage | ||||||
|  | 
 | ||||||
|  |     mdo | ||||||
|  |       void . runWithReplace (commitSelection doCommit) . ffor hasCommitted $ \() -> | ||||||
|  |         elClass "div" "text-xl py-4" $ text "Committed, waiting for the others." | ||||||
|  |       doCommit <- elClass "div" "flex flex-row mt-4" $ do | ||||||
|  |         -- Until the head is committed starting the head can be aborted: | ||||||
|  |         tellAction | ||||||
|  |           . (Hydra.ClientInput.Abort <$) | ||||||
|  |           <=< buttonClass "bg-gray-400 hover:bg-gray-300 active:bg-gray-200 text-white font-bold text-xl px-4 py-2 rounded-md mr-2" | ||||||
|  |           $ text "Abort" | ||||||
|  |         isDisabled <- holdDyn False (True <$ hasCommitted) | ||||||
|  |         let cls = | ||||||
|  |               (bool "bg-blue-500 hover:bg-blue-400 active:bg-blue-300" "bg-gray-500 hover:bg-gray-500 active:bg-gray-500 cursor-not-allowed " <$> isDisabled) | ||||||
|  |                 <> " text-white font-bold text-xl px-4 py-2 rounded-md" | ||||||
|  |         buttonClass cls $ text "Commit" | ||||||
|  |       pure () | ||||||
|  |     pure () | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | openScreen :: | ||||||
|  |   ( EventWriter t [ClientInput] m, | ||||||
|  |     DomBuilder t m, | ||||||
|  |     MonadFix m, | ||||||
|  |     MonadHold t m, | ||||||
|  |     PostBuild t m, | ||||||
|  |     MonadIO (Performable m), PerformEvent t m) => | ||||||
|  |   IORef State -> | ||||||
|  |   Text -> | ||||||
|  |   [Text] -> | ||||||
|  |   Address -> | ||||||
|  |   Event t (ServerOutput tx) -> | ||||||
|  |   m () | ||||||
|  | openScreen hydraProcessHandlesRef name actorNames actorAddress webSocketMessage = do | ||||||
|  |   -- Get your UTxOs on page load and when we observe a transaction | ||||||
|  |   tellAction . (GetUTxO <$) | ||||||
|  |     . ( ( void $ | ||||||
|  |             filter | ||||||
|  |               ( \case | ||||||
|  |                   TxSeen {} -> True | ||||||
|  |                   _ -> False | ||||||
|  |               ) | ||||||
|  |               webSocketMessage | ||||||
|  |         ) | ||||||
|  |           <> | ||||||
|  |       ) | ||||||
|  |     =<< getPostBuild | ||||||
|  |   let updatedUTXOs = | ||||||
|  |         fmap (filterUtxos actorAddress) | ||||||
|  |           . mapMaybe | ||||||
|  |             ( \case | ||||||
|  |                 GetUTxOResponse utxoz -> Just utxoz | ||||||
|  |                 _ -> Nothing | ||||||
|  |             ) | ||||||
|  |           $ webSocketMessage | ||||||
|  |   currentUTXOs <- holdDyn mempty updatedUTXOs | ||||||
|  |   let ifUTXOs yes no = dyn_ (bool yes no <$> fmap Map.null currentUTXOs) | ||||||
|  |       ifUTXOsDyn yes no = dyn (bool yes no <$> fmap Map.null currentUTXOs) | ||||||
|  |   mdo | ||||||
|  |     (_, currentSet) <- | ||||||
|  |       runDynamicWriterT | ||||||
|  |         . runWithReplace (elClass "div" "text-white text-2xl" $ text "Getting your UTxOs") | ||||||
|  |         $ fmap (tellDyn <=< (pure . pure . filterOutFuel)) updatedUTXOs | ||||||
|  |     _ <- elClass "div" "mb-4 ml-2" $ dyn_ $ utxoPicker False <$> currentSet | ||||||
|  |     elClass "div" "text-xl mb-8 ml-2" $ ifUTXOs (text "Send Ada to a participant:") (text "No UTXOs for this participant") | ||||||
|  |     flip ifUTXOs blank $ do | ||||||
|  |       (recipientDyn, lovelaceDyn) <- elClass "div" "flex ml-2 mb-2" $ | ||||||
|  |         elClass "div" "w-auto flex flex-row rounded bg-gray-800 mb-2 overflow-hidden" $ do | ||||||
|  |           ie <- elClass "div" "flex flex-col p-2" $ do | ||||||
|  |             elClass "div" "text-gray-600 text-sm font-semibold" $ text "LOVELACES" | ||||||
|  |             inputElement $ | ||||||
|  |               def | ||||||
|  |                 & initialAttributes .~ ("class" =: "bg-gray-800 text-2xl font-bold focus:outline-none p-2" <> "type" =: "number") | ||||||
|  |                 & inputElementConfig_initialValue .~ "1000000" | ||||||
|  |           recipient <- fmap Reflex.Dom.value $ | ||||||
|  |             elClass "div" "flex flex-col p-2" $ do | ||||||
|  |               elClass "div" "text-gray-600 text-sm font-semibold uppercase" $ text "To" | ||||||
|  |               -- FIXME: unsafe head, will crash with <= 1 actors | ||||||
|  |               dropdown | ||||||
|  |                 (head $ filter (/= name) actorNames) | ||||||
|  |                 (pure (Map.filter (/= name) $ Map.fromList (fmap (\n -> (n, n)) actorNames))) | ||||||
|  |                 $ def & dropdownConfig_attributes .~ pure ("class" =: "bg-gray-800 hover:bg-gray-700 active:bg-gray-900 text-gray-100 font-semibold text-xl px-4 py-2 rounded-md m-2") | ||||||
|  |           pure (recipient, readMaybe . T.unpack <$> _inputElement_value ie) | ||||||
|  |       elClass "div" "flex" $ do | ||||||
|  |         signedTxE <- | ||||||
|  |           performEvent . fmap liftIO $ | ||||||
|  |             makeTx hydraProcessHandlesRef name | ||||||
|  |               <$> current currentSet | ||||||
|  |               -- NOTE/TODO(skylar): This is just to default to the minimum | ||||||
|  |               <*> current (fromMaybe 1000000 <$> lovelaceDyn) | ||||||
|  |               <*> current recipientDyn | ||||||
|  |               <@ doSend | ||||||
|  |         tellAction | ||||||
|  |           . fmap NewTx | ||||||
|  |           $ signedTxE | ||||||
|  | 
 | ||||||
|  |     doSend <- elClass "div" "flex flex-row ml-2" $ do | ||||||
|  |       sendButtonClick <- flip ifUTXOsDyn (pure never) $ do | ||||||
|  |         buttonClass "bg-green-500 hover:bg-green-400 active:bg-green-200 text-white font-bold text-xl mr-2 px-4 py-2 rounded-md" $ text "Send" | ||||||
|  |       tellAction | ||||||
|  |         . (Close <$) | ||||||
|  |         <=< buttonClass "bg-red-500 hover:bg-red-400 active:bg-red-200 text-white font-bold text-xl px-4 py-2 rounded-md" | ||||||
|  |         $ text "Close Head" | ||||||
|  |       switchHold never sendButtonClick | ||||||
|  |     pure () | ||||||
|  |   pure () | ||||||
|  | 
 | ||||||
|  | closedScreen :: | ||||||
|  |   ( MonadFix m, | ||||||
|  |     MonadIO m, | ||||||
|  |     MonadIO (Performable m), | ||||||
|  |     DomBuilder t m, | ||||||
|  |     PostBuild t m, | ||||||
|  |     TriggerEvent t m, | ||||||
|  |     PerformEvent t m, | ||||||
|  |     MonadHold t m | ||||||
|  |   ) => | ||||||
|  |   UTCTime -> | ||||||
|  |   m () | ||||||
|  | closedScreen fanoutTime = do | ||||||
|  |   countDownDyn <- clockLossy 1 fanoutTime | ||||||
|  |   elClass "div" "text-white text-2xl my-4 ml-2" $ do | ||||||
|  |     text "Fanout time left: " | ||||||
|  |     dyn_ | ||||||
|  |       ( text . T.pack . show @Integer | ||||||
|  |           . ceiling | ||||||
|  |           . diffUTCTime fanoutTime | ||||||
|  |           . _tickInfo_lastUTC | ||||||
|  |           <$> countDownDyn | ||||||
|  |       ) | ||||||
|  |     text " seconds" | ||||||
							
								
								
									
										14
									
								
								src/Paths.hs
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										14
									
								
								src/Paths.hs
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,14 @@ | |||||||
|  | {-# LANGUAGE TemplateHaskell #-} | ||||||
|  | module Paths where | ||||||
|  | 
 | ||||||
|  | import System.Which (staticWhich) | ||||||
|  | 
 | ||||||
|  | hydraToolsPath :: FilePath | ||||||
|  | hydraToolsPath = $(staticWhich "hydra-tools") | ||||||
|  | 
 | ||||||
|  | realpathPath :: FilePath | ||||||
|  | realpathPath = $(staticWhich "realpath") | ||||||
|  | 
 | ||||||
|  | dirnamePath :: FilePath | ||||||
|  | dirnamePath = $(staticWhich "dirname") | ||||||
|  | 
 | ||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user