2023-04-12 11:53:55 +00:00
|
|
|
|
using KubernetesWorkflow;
|
|
|
|
|
|
|
|
|
|
namespace DistTestCore.Codex
|
|
|
|
|
{
|
|
|
|
|
public class CodexAccess
|
|
|
|
|
{
|
2023-06-01 07:35:18 +00:00
|
|
|
|
private readonly TestLifecycle lifecycle;
|
2023-04-30 08:08:32 +00:00
|
|
|
|
|
2023-06-01 07:35:18 +00:00
|
|
|
|
public CodexAccess(TestLifecycle lifecycle, RunningContainer runningContainer)
|
2023-04-12 11:53:55 +00:00
|
|
|
|
{
|
2023-06-01 07:35:18 +00:00
|
|
|
|
this.lifecycle = lifecycle;
|
2023-04-13 07:33:10 +00:00
|
|
|
|
Container = runningContainer;
|
2023-04-12 11:53:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-13 07:33:10 +00:00
|
|
|
|
public RunningContainer Container { get; }
|
|
|
|
|
|
2023-04-12 11:53:55 +00:00
|
|
|
|
public CodexDebugResponse GetDebugInfo()
|
|
|
|
|
{
|
2023-05-11 10:44:53 +00:00
|
|
|
|
return Http(TimeSpan.FromSeconds(2)).HttpGetJson<CodexDebugResponse>("debug/info");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public CodexDebugPeerResponse GetDebugPeer(string peerId)
|
|
|
|
|
{
|
2023-05-18 08:42:04 +00:00
|
|
|
|
return GetDebugPeer(peerId, TimeSpan.FromSeconds(2));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public CodexDebugPeerResponse GetDebugPeer(string peerId, TimeSpan timeout)
|
|
|
|
|
{
|
2023-05-31 11:15:41 +00:00
|
|
|
|
var http = Http(timeout);
|
|
|
|
|
var str = http.HttpGetString($"debug/peer/{peerId}");
|
|
|
|
|
|
|
|
|
|
if (str.ToLowerInvariant() == "unable to find peer!")
|
|
|
|
|
{
|
|
|
|
|
return new CodexDebugPeerResponse
|
|
|
|
|
{
|
|
|
|
|
IsPeerFound = false
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
var result = http.TryJsonDeserialize<CodexDebugPeerResponse>(str);
|
|
|
|
|
result.IsPeerFound = true;
|
|
|
|
|
return result;
|
2023-04-12 11:53:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-13 07:33:10 +00:00
|
|
|
|
public string UploadFile(FileStream fileStream)
|
|
|
|
|
{
|
|
|
|
|
return Http().HttpPostStream("upload", fileStream);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Stream DownloadFile(string contentId)
|
|
|
|
|
{
|
|
|
|
|
return Http().HttpGetStream("download/" + contentId);
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-18 13:33:12 +00:00
|
|
|
|
public CodexSalesAvailabilityResponse SalesAvailability(CodexSalesAvailabilityRequest request)
|
|
|
|
|
{
|
|
|
|
|
return Http().HttpPostJson<CodexSalesAvailabilityRequest, CodexSalesAvailabilityResponse>("sales/availability", request);
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-24 14:07:32 +00:00
|
|
|
|
public string RequestStorage(CodexSalesRequestStorageRequest request, string contentId)
|
2023-04-18 13:33:12 +00:00
|
|
|
|
{
|
2023-04-24 14:07:32 +00:00
|
|
|
|
return Http().HttpPostJson($"storage/request/{contentId}", request);
|
2023-04-18 13:33:12 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-05-11 10:44:53 +00:00
|
|
|
|
public string ConnectToPeer(string peerId, string peerMultiAddress)
|
|
|
|
|
{
|
|
|
|
|
return Http().HttpGetString($"connect/{peerId}?addrs={peerMultiAddress}");
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-04 09:34:43 +00:00
|
|
|
|
public void EnsureOnline()
|
|
|
|
|
{
|
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
var debugInfo = GetDebugInfo();
|
|
|
|
|
if (debugInfo == null || string.IsNullOrEmpty(debugInfo.id)) throw new InvalidOperationException("Unable to get debug-info from codex node at startup.");
|
|
|
|
|
|
|
|
|
|
var nodePeerId = debugInfo.id;
|
|
|
|
|
var nodeName = Container.Name;
|
2023-06-01 07:35:18 +00:00
|
|
|
|
lifecycle.Log.AddStringReplace(nodePeerId, nodeName);
|
|
|
|
|
lifecycle.Log.AddStringReplace(debugInfo.table.localNode.nodeId, nodeName);
|
2023-05-04 09:34:43 +00:00
|
|
|
|
}
|
|
|
|
|
catch (Exception e)
|
|
|
|
|
{
|
2023-06-01 07:35:18 +00:00
|
|
|
|
lifecycle.Log.Error($"Failed to start codex node: {e}. Test infra failure.");
|
2023-05-04 09:34:43 +00:00
|
|
|
|
throw new InvalidOperationException($"Failed to start codex node. Test infra failure.", e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-05-11 10:44:53 +00:00
|
|
|
|
private Http Http(TimeSpan? timeoutOverride = null)
|
2023-04-12 11:53:55 +00:00
|
|
|
|
{
|
2023-06-01 07:35:18 +00:00
|
|
|
|
var address = lifecycle.Configuration.GetAddress(Container);
|
|
|
|
|
return new Http(lifecycle.Log, lifecycle.TimeSet, address, baseUrl: "/api/codex/v1", timeoutOverride);
|
2023-04-13 07:33:10 +00:00
|
|
|
|
}
|
2023-04-12 11:53:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class CodexDebugResponse
|
|
|
|
|
{
|
|
|
|
|
public string id { get; set; } = string.Empty;
|
|
|
|
|
public string[] addrs { get; set; } = new string[0];
|
|
|
|
|
public string repo { get; set; } = string.Empty;
|
|
|
|
|
public string spr { get; set; } = string.Empty;
|
2023-04-26 12:40:54 +00:00
|
|
|
|
public EnginePeerResponse[] enginePeers { get; set; } = Array.Empty<EnginePeerResponse>();
|
|
|
|
|
public SwitchPeerResponse[] switchPeers { get; set; } = Array.Empty<SwitchPeerResponse>();
|
2023-04-12 11:53:55 +00:00
|
|
|
|
public CodexDebugVersionResponse codex { get; set; } = new();
|
2023-05-10 06:53:57 +00:00
|
|
|
|
public CodexDebugTableResponse table { get; set; } = new();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class CodexDebugTableResponse
|
|
|
|
|
{
|
|
|
|
|
public CodexDebugTableNodeResponse localNode { get; set; } = new();
|
|
|
|
|
public CodexDebugTableNodeResponse[] nodes { get; set; } = Array.Empty<CodexDebugTableNodeResponse>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class CodexDebugTableNodeResponse
|
|
|
|
|
{
|
|
|
|
|
public string nodeId { get; set; } = string.Empty;
|
2023-05-18 08:42:04 +00:00
|
|
|
|
public string peerId { get; set; } = string.Empty;
|
2023-05-10 06:53:57 +00:00
|
|
|
|
public string record { get; set; } = string.Empty;
|
2023-05-29 06:35:46 +00:00
|
|
|
|
public string address { get; set; } = string.Empty;
|
2023-05-10 06:53:57 +00:00
|
|
|
|
public bool seen { get; set; }
|
2023-04-12 11:53:55 +00:00
|
|
|
|
}
|
|
|
|
|
|
2023-04-26 12:40:54 +00:00
|
|
|
|
public class EnginePeerResponse
|
|
|
|
|
{
|
|
|
|
|
public string peerId { get; set; } = string.Empty;
|
|
|
|
|
public EnginePeerContextResponse context { get; set; } = new();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class EnginePeerContextResponse
|
|
|
|
|
{
|
|
|
|
|
public int blocks { get; set; } = 0;
|
|
|
|
|
public int peerWants { get; set; } = 0;
|
|
|
|
|
public int exchanged { get; set; } = 0;
|
|
|
|
|
public string lastExchange { get; set; } = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class SwitchPeerResponse
|
|
|
|
|
{
|
|
|
|
|
public string peerId { get; set; } = string.Empty;
|
|
|
|
|
public string key { get; set; } = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-12 11:53:55 +00:00
|
|
|
|
public class CodexDebugVersionResponse
|
|
|
|
|
{
|
|
|
|
|
public string version { get; set; } = string.Empty;
|
|
|
|
|
public string revision { get; set; } = string.Empty;
|
|
|
|
|
}
|
2023-04-18 13:33:12 +00:00
|
|
|
|
|
2023-05-11 10:44:53 +00:00
|
|
|
|
public class CodexDebugPeerResponse
|
|
|
|
|
{
|
2023-05-31 11:15:41 +00:00
|
|
|
|
public bool IsPeerFound { get; set; }
|
|
|
|
|
|
2023-05-11 10:44:53 +00:00
|
|
|
|
public string peerId { get; set; } = string.Empty;
|
|
|
|
|
public long seqNo { get; set; }
|
|
|
|
|
public CodexDebugPeerAddressResponse[] addresses { get; set; } = Array.Empty<CodexDebugPeerAddressResponse>();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class CodexDebugPeerAddressResponse
|
|
|
|
|
{
|
|
|
|
|
public string address { get; set; } = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
2023-04-18 13:33:12 +00:00
|
|
|
|
public class CodexSalesAvailabilityRequest
|
|
|
|
|
{
|
|
|
|
|
public string size { get; set; } = string.Empty;
|
|
|
|
|
public string duration { get; set; } = string.Empty;
|
|
|
|
|
public string minPrice { get; set; } = string.Empty;
|
|
|
|
|
public string maxCollateral { get; set; } = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class CodexSalesAvailabilityResponse
|
|
|
|
|
{
|
|
|
|
|
public string id { get; set; } = string.Empty;
|
|
|
|
|
public string size { get; set; } = string.Empty;
|
|
|
|
|
public string duration { get; set; } = string.Empty;
|
|
|
|
|
public string minPrice { get; set; } = string.Empty;
|
|
|
|
|
public string maxCollateral { get; set; } = string.Empty;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class CodexSalesRequestStorageRequest
|
|
|
|
|
{
|
|
|
|
|
public string duration { get; set; } = string.Empty;
|
|
|
|
|
public string proofProbability { get; set; } = string.Empty;
|
|
|
|
|
public string reward { get; set; } = string.Empty;
|
|
|
|
|
public string collateral { get; set; } = string.Empty;
|
|
|
|
|
public string? expiry { get; set; }
|
|
|
|
|
public uint? nodes { get; set; }
|
|
|
|
|
public uint? tolerance { get; set;}
|
|
|
|
|
}
|
2023-04-12 11:53:55 +00:00
|
|
|
|
}
|