using CodexContractsPlugin.Marketplace; using Core; using GethPlugin; using KubernetesWorkflow; using KubernetesWorkflow.Types; using Logging; using Newtonsoft.Json; using Utils; namespace CodexContractsPlugin { public class CodexContractsStarter { private readonly IPluginTools tools; public CodexContractsStarter(IPluginTools tools) { this.tools = tools; } public CodexContractsDeployment Deploy(CoreInterface ci, IGethNode gethNode) { Log("Starting Codex SmartContracts container..."); var workflow = tools.CreateWorkflow(); var startupConfig = CreateStartupConfig(gethNode); startupConfig.NameOverride = "codex-contracts"; var containers = workflow.Start(1, new CodexContractsContainerRecipe(), startupConfig).WaitForOnline(); if (containers.Containers.Length != 1) throw new InvalidOperationException("Expected 1 Codex contracts container to be created. Test infra failure."); var container = containers.Containers[0]; Log("Container started."); try { var result = DeployContract(container, workflow, gethNode); workflow.Stop(containers, waitTillStopped: false); Log("Container stopped."); return result; } catch (Exception ex) { Log("Failed to deploy contract: " + ex); Log("Downloading Codex SmartContracts container log..."); ci.DownloadLog(container); throw; } } public ICodexContracts Wrap(IGethNode gethNode, CodexContractsDeployment deployment) { return new CodexContractsAccess(tools.GetLog(), gethNode, deployment); } private CodexContractsDeployment DeployContract(RunningContainer container, IStartupWorkflow workflow, IGethNode gethNode) { Log("Deploying SmartContract..."); WaitUntil(() => { var logHandler = new ContractsReadyLogHandler(tools.GetLog()); workflow.DownloadContainerLog(container, logHandler, 100); return logHandler.Found; }, nameof(DeployContract)); Log("Contracts deployed. Extracting addresses..."); var extractor = new ContractsContainerInfoExtractor(tools.GetLog(), workflow, container); var marketplaceAddress = extractor.ExtractMarketplaceAddress(); var (abi, bytecode) = extractor.ExtractMarketplaceAbiAndByteCode(); EnsureCompatbility(abi, bytecode); var interaction = new ContractInteractions(tools.GetLog(), gethNode); var tokenAddress = interaction.GetTokenAddress(marketplaceAddress); Log("Extract completed. Checking sync..."); Time.WaitUntil(() => interaction.IsSynced(marketplaceAddress, abi), nameof(DeployContract)); Log("Synced. Codex SmartContracts deployed. Getting configuration..."); var config = GetMarketplaceConfiguration(marketplaceAddress, gethNode); Log("Got config: " + JsonConvert.SerializeObject(config)); return new CodexContractsDeployment(config, marketplaceAddress, abi, tokenAddress); } private MarketplaceConfig GetMarketplaceConfiguration(string marketplaceAddress, IGethNode gethNode) { var func = new ConfigurationFunctionBase(); var response = gethNode.Call(marketplaceAddress, func); return response.ReturnValue1; } private void EnsureCompatbility(string abi, string bytecode) { var expectedByteCode = MarketplaceDeploymentBase.BYTECODE.ToLowerInvariant(); if (bytecode != expectedByteCode) { Log("Deployed contract is incompatible with current build of CodexContracts plugin. Running self-updater..."); var selfUpdater = new SelfUpdater(); selfUpdater.Update(abi, bytecode); } } private void Log(string msg) { tools.GetLog().Log(msg); } private void WaitUntil(Func predicate, string msg) { Time.WaitUntil(predicate, TimeSpan.FromMinutes(5), TimeSpan.FromSeconds(2), msg); } private StartupConfig CreateStartupConfig(IGethNode gethNode) { var startupConfig = new StartupConfig(); var contractsConfig = new CodexContractsContainerConfig(gethNode); startupConfig.Add(contractsConfig); return startupConfig; } } public class ContractsReadyLogHandler : LogHandler { // Log should contain 'Compiled 15 Solidity files successfully' at some point. private const string RequiredCompiledString = "Solidity files successfully"; // When script is done, it prints the ready-string. private const string ReadyString = "Done! Sleeping indefinitely..."; private readonly ILog log; public ContractsReadyLogHandler(ILog log) { this.log = log; log.Debug($"Looking for '{RequiredCompiledString}' and '{ReadyString}' in container logs..."); } public bool SeenCompileString { get; private set; } public bool Found { get; private set; } protected override void ProcessLine(string line) { log.Debug(line); if (line.Contains(RequiredCompiledString)) SeenCompileString = true; if (line.Contains(ReadyString)) { if (!SeenCompileString) throw new Exception("CodexContracts deployment failed. " + "Solidity files not compiled before process exited."); Found = true; } } } }