BLAM3.SDK
3.1.7.25
dotnet add package BLAM3.SDK --version 3.1.7.25
NuGet\Install-Package BLAM3.SDK -Version 3.1.7.25
<PackageReference Include="BLAM3.SDK" Version="3.1.7.25" />
paket add BLAM3.SDK --version 3.1.7.25
#r "nuget: BLAM3.SDK, 3.1.7.25"
// Install BLAM3.SDK as a Cake Addin #addin nuget:?package=BLAM3.SDK&version=3.1.7.25 // Install BLAM3.SDK as a Cake Tool #tool nuget:?package=BLAM3.SDK&version=3.1.7.25
Blue Lucy SDK
Blue Lucy is a media services integration platform that unifies production and supply chain operations for the media industry. The Blue Lucy .Net SDK provides the coding framework for developers to create their own BLidgets and Plugins for running within the Blue Lucy Workflow Runner. It can also be used to rapidly delevop bespoke integrations directly with the Blue Lucy API.
NuGet Packages
Package | Latest Version |
---|---|
BLAM3.SDK |
Documentation
This README provides a quick overview of how to make use of the Blue Lucy .Net SDK. You can check out some sample code on our public GitHub repository for more in depth examples.
Quick Start
[!IMPORTANT] To use the Blue Lucy .Net SDK, you will need to have an active application account on a Blue Lucy system. Please contact support, or your system adminstrator for assistance in setting up an account on your system.
To get started, first add the BLAM3.SDK package to your project by running the following command:
dotnet add package BLAM3.SDK
You can now create a BlamClient
using your application credentials and begin searching your media:
// Create a new BlamClient instance using application credentials
var blamClient = new BlamClient(new BlamClientConfig
{
BlamApiEndpoint = "https://<your>.bluelucy.com/api",
Username = "<ApplicationUsername>",
Password = "<ApplicationPassword>"
});
// Search for the first 100 video assets
var results = await blamClient.Search(new SearchQueryInputModel
{
Query = "type=video",
Limit = 100
});
// Select first asset from search results
var asset = results.Items.First();
Update asset metadata
// Create a new metadata snapshot updating just the specified values
var snapshot = await blamClient.CreateMetadataSnapshot(new AssetMetadataSnapshotInputModel
{
AssetId = asset.Id,
Values = new Dictionary<string, object>
{
{ "show_name", "Cats on Air" },
{ "vtr", 899223 },
{ "language", "Dutch" },
{ "tx_date", "2033-01-01T15:00:00+01:00" },
{ "for_broadcast", true }
},
Merge = true
});
Updating the metadata on an asset creates a new snapshot version. Creating a snapshot with Merge = true
will use the current metadata as the snapshot base and replace just the specified values.
Add an asset to a folder
// Search for an existing folder
var folders = await blamClient.Search(new FolderQueryInputModel
{
Query = "name=Cats on Air",
Limit = 1,
Depth = FolderStructureDepth.Contents
});
// Select first folder from search results
var folder = folders.Items.First();
// Attach the asset to the folder
await blamClient.AttachAssetsToFolder(new FolderAssetListInputModel
{
FolderId = folder.Id,
AssetIds = [asset.Id]
});
Assets can be added or removed in bulk by providing a collection of asset Id
s.
Creating a link to the proxy
// Create a temporary link to the proxy file
var link = await blamClient.CreateStreamToken(new StreamTokenRequestInputModel
{
Asset = new AssetSelectorInputModel { AssetId = asset.Id, Selector = "Current Version;Browse" },
MaxAllowedClaims = MaxAllowedClaims.Unlimited,
ShortUrl = true
});
// Use the url for streaming
var url = link.Url;
Links can be single or multi-use and a sharable with external users or processes (e.g. transcoding into different formats).
Triggering a workflow
// Fetch the list of available workflows
var blueprints = await blamClient.GetWorkflowBlueprints();
var blueprint = blueprints.First(b => b.Name == "Deliver Asset to S3");
// Trigger the selected workflow blueprint
var run = await blamClient.TriggerWorkflow(blueprint.Id, new WorkflowRunInputModel
{
Name = "Delivery for <Customer>",
Type = BlidgetDataType.Assets,
Data = new AssetsIO { AssetIds = [asset.Id] }
});
// Check the current state of the workflow run
var runState = await blamClient.GetWorkflowRun(run.Ids.First());
while (runState.Status != WorkflowRunStatus.Complete)
{
// ...
}
[!IMPORTANT] Workflows can be highly customised per platform. You should contact your workflow administrator to understand what workflows you have access to on your platform. Access to some workflows can be restricted based on your account group access permissions.
BLidgets
BLidgets are short lived micro-processes executed as steps within a Blue Lucy workflow. A BLidget represents a "unit of work" within a workflow and its scope limited to a single task during execution (e.g. Transcode video, Copy file, Import metadata, Run job etc.)
The Blue Lucy SDK simplifies BLidget creation by providing an inheritable base class which implements the BLidget control flow and handles the passing of messages between the BLidget process and the Workflow runner.
BLidget control flow is broken down into 3 phases; Init
, Run
and Complete
:
// ExampleBlidget implements base class `Blidget` with I/O of type `AssetsIO`
internal class ExampleBlidget : Blidget<AssetsIO, AssetsIO>
{
private BlamClient _client;
private ExampleState _runState;
// INIT PHASE - runs on every blidget run
protected override void Init(AssetsIO inputData)
{
_client = new BlamClient(); // Initialise BlamClient (inherits configuration from the Workflow runner)
_runState = GetRunState<ExampleState>(); // Initialise or retrieve the current saved run state
}
// INIT PAHSE - runs on every blidget run (asynchronous variant of `void Init(...)`)
// protected override Task InitAsync(AssetsIO inputData)
// {
// // Asynchronous init operations
// }
// RUN PHASE - runs on the first BLidget execution only, skipped after waking up from `Waiting` or `Idle`
protected override async Task<BlidgetOutput<AssetsIO>> RunAsync(AssetsIO inputData)
{
// Fetch BLidget configuration value
var nameTemplate = GetArgument<string>("name_template");
foreach (var assetId in inputData.AssetIds)
{
// Fetch input resources
var asset = await _client.GetAssetById(assetId);
// Use interpolator to resolve runtime values
var name = await Interpolate(nameTemplate, _client, new TemplateData { Asset = asset });
// Do work ...
// Add data to the run state
var job = new Job { Uuid = Guid.NewGuid() };
_runState.Jobs.Add(job);
// Log Debug message
LogDebug("Added job with ID: '{id}'", job.Uuid);
}
// End process and save current run state; wake again once time has elapsed or receiving a call back
return Waiting(_runState, pollingIntervalSeconds: PollingInterval.OneHour, message: "Waking again in one hour.");
// or return Idle(_runState, _runState.Jobs.Select(j => new Callback("jobId", $"{j.Uuid}")), message: "Waking on callback.");
// or return Complete(inputData);
}
// COMPLETE PHASE - called each time the BLidget runs after waking up from `Waiting` or `Idle`
protected override async Task<BlidgetOutput<AssetsIO>> CompleteAsync(AssetsIO inputData)
{
// Loop through jobs and update run state
foreach (var job in _runState.Jobs)
{
// Do work ...
// Update progress state (used to report progress)
UpdateProgress(50, "Processed 1 of 2 job(s).");
}
// Complete process and pass on output data
return Complete(inputData, message: $"Processed {_runState.Jobs.Count} job(s).");
// or return Waiting(_runState, pollingIntervalSeconds: PollingInterval.OneHour, message: "Waking again in one hour.");
// or return Idle(_runState, _runState.Jobs.Select(j => new Callback("jobId", $"{j.Uuid}")), message: "Waking on callback.");
}
}
internal class ExampleState
{
public List<Job> Jobs { get; set; } = [];
}
internal class Job
{
public Guid Uuid { get; set; }
}
More detailed examples can be found on the samples GitHub repository.
Plugins
Plugins are long running processes which are commonly used for watching for and reacting to internal or external events (e.g. files moving into a folder, a timer elapsing, receiving a message). They can also be used as extensions to the API by implementing additional end points or even act as web server for custom UIs.
Plugins implement the .Net IHostedService
and provide two methods for control flow; OnStart
and OnStop
. Additional services or middleware can also be added using .Net's Dependency Injection
allowing developers to create fully featured services within a single Plugin.
There are two possible builder patterns depending on whether a web server is required:
Service Pattern
// ExamplePlugin implements base class `PluginInitialiser`
internal class ExamplePlugin : PluginInitialiser
{
private async Task CreateHostedService()
{
var builder = new HostBuilder()
.ConfigureAppConfiguration((hostContext, configuration) =>
{
})
.ConfigureServices((hostContext, services) =>
{ // Configure serivces using dependency injection
services.AddSingleton<BlamClient>();
services.AddHostedService<ExamplePluginService>();
});
var host = builder
.Build();
// Runs hosted service(s)
await host.RunAsync();
}
public override async Task InitAsync()
{ // Plugin entry point - fetch configuration values here by calling `GetArgumnet`
await CreateHostedService();
}
}
// Inject `BlamClient` singleton using dependency injection
internal class ExamplePluginService(BlamClient client) : PluginService
{
private ExampleState _runState;
private WorkflowBlueprintViewModel _blueprint;
// ONSTART PHASE - called when the hosted service first starts on `host.RunAsync()`
protected override async Task OnStartAsync(CancellationToken cancellationToken)
{
_runState = GetRunState<ExampleState>(); // Initialise or retrieve the current saved run state
var blueprintId = GetArgument<int>("blueprint_id"); // Fetch BLidget configuration value
_blueprint = await client.GetWorkflowBlueprint(blueprintId); // Fetch argument resource
ProcessWork(cancellationToken); // Begin work
}
// ONSTOP PHASE - called when the host application is stopped e.g. SIGTERM received
protected override Task OnStopAsync(CancellationToken cancellationToken)
{
// Clean up steps ...
return Task.CompletedTask;
}
private void ProcessWork(CancellationToken cancellationToken) =>
Task.Run(async () => await ProcessWorkAsync(cancellationToken), cancellationToken);
private async Task ProcessWorkAsync(CancellationToken cancellationToken)
{
while (!cancellationToken.IsCancellationRequested)
{
// Do work ...
// Save run state during plugin execution
Save(_runState);
// Log Debug
LogDebug("Saved run state.");
}
}
}
internal class ExampleState
{
public List<TrackedItem> Items { get; set; }
}
internal class TrackedItem
{
public Guid Uuid { get; set; }
}
Web Server Pattern
// ExamplePlugin implements base class `PluginInitialiser`
internal class ExamplePlugin : PluginInitialiser
{
private async Task CreateWebApp(string baseRoute, int port)
{
baseRoute = string.IsNullOrWhiteSpace(baseRoute) ? null : $"/{baseRoute}";
var builder = WebApplication.CreateBuilder();
// Configure serivces using dependency injection
builder.Services.AddSingleton<ExamplePluginService>();
builder.Services.AddHostedService(p => p.GetService<ExamplePluginService>());
builder.Logging.ClearProviders();
var app = builder.Build();
// Define API endpoints using .Net minimal API
app.MapGet($"{baseRoute}/hello", () =>
{
return "hello, world";
});
// Start web server and listen on incoming port
await app.RunAsync($"http://*:{port}");
}
public override async Task InitAsync()
{ // Plugin entry point - fetch configuration values here by calling `GetArgumnet`
var baseRoute = GetArgument<string>("base_route");
var port = GetArgument<int>("port");
await CreateWebApp(baseRoute.Trim('/'), port);
}
}
Next steps
To learn more about Blue Lucy, visit the Blue Lucy website.
Samples
- Samples: Samples in this repository serve as a reference for developing BLigdets and Plugins
Product | Versions Compatible and additional computed target framework versions. |
---|---|
.NET | net8.0 is compatible. net8.0-android was computed. net8.0-browser was computed. net8.0-ios was computed. net8.0-maccatalyst was computed. net8.0-macos was computed. net8.0-tvos was computed. net8.0-windows was computed. |
-
net8.0
- Microsoft.AspNetCore.SignalR.Client (>= 8.0.8)
- Microsoft.AspNetCore.SignalR.Protocols.NewtonsoftJson (>= 8.0.8)
- Microsoft.Extensions.Diagnostics (>= 8.0.0)
- Microsoft.Extensions.Hosting.Abstractions (>= 8.0.0)
- Microsoft.Extensions.Http (>= 8.0.0)
- Microsoft.Extensions.Logging.Abstractions (>= 8.0.1)
- Microsoft.IdentityModel.JsonWebTokens (>= 8.0.2)
- Newtonsoft.Json (>= 13.0.3)
- OpenTelemetry.Exporter.OpenTelemetryProtocol (>= 1.9.0)
- OpenTelemetry.Extensions.Hosting (>= 1.9.0)
- OpenTelemetry.Instrumentation.AspNetCore (>= 1.9.0)
- OpenTelemetry.Instrumentation.Http (>= 1.9.0)
- Polly (>= 8.4.1)
- System.Reactive (>= 6.0.1)
NuGet packages
This package is not used by any NuGet packages.
GitHub repositories
This package is not used by any popular GitHub repositories.
Version | Downloads | Last updated |
---|---|---|
3.1.7.25 | 250 | 12/4/2024 |
3.1.5.12 | 2,088 | 5/16/2024 |
3.1.5.11 | 290 | 5/7/2024 |
3.1.5.8 | 471 | 4/3/2024 |
3.1.4.9 | 511 | 2/6/2024 |
3.1.4.8 | 127 | 2/5/2024 |
3.1.3.19 | 180 | 2/5/2024 |
3.1.3.16 | 1,745 | 10/4/2023 |
3.1.0.25 | 330 | 8/14/2023 |
3.1.0.24 | 333 | 7/25/2023 |
3.1.0.23 | 209 | 7/17/2023 |
3.1.0.19 | 432 | 6/20/2023 |
3.1.0.18 | 216 | 6/8/2023 |
3.0.8.16 | 338 | 3/15/2023 |
3.0.8.11 | 357 | 2/7/2023 |
3.0.8.10 | 363 | 2/1/2023 |
3.0.7.16 | 464 | 11/29/2022 |
3.0.7.13 | 1,076 | 11/9/2022 |
3.0.6.62 | 486 | 10/4/2022 |
3.0.6.60 | 793 | 7/25/2022 |
3.0.6.59 | 504 | 7/15/2022 |
3.0.6.57 | 583 | 7/7/2022 |
3.0.6.49 | 589 | 6/15/2022 |
3.0.6.44 | 1,984 | 5/18/2022 |
3.0.6.42 | 549 | 5/11/2022 |
3.0.6.34 | 744 | 4/19/2022 |
3.0.6.33 | 597 | 4/7/2022 |
3.0.6.23 | 576 | 3/15/2022 |
3.0.6.21 | 548 | 3/7/2022 |
3.0.6.20 | 535 | 2/28/2022 |
3.0.6.14 | 567 | 1/16/2022 |
3.0.6.10 | 1,824 | 12/23/2021 |
3.0.6.8 | 389 | 12/15/2021 |
3.0.6.6 | 390 | 12/10/2021 |
3.0.6.5 | 556 | 12/9/2021 |
3.0.6.4 | 388 | 12/2/2021 |
3.0.6.2 | 3,082 | 11/25/2021 |
3.0.6 | 425 | 11/17/2021 |
3.0.5.46 | 434 | 11/10/2021 |
3.0.5.45 | 421 | 11/8/2021 |
3.0.5.42 | 414 | 10/26/2021 |
3.0.5.41 | 452 | 10/19/2021 |
3.0.5.36 | 500 | 9/15/2021 |
3.0.5.35 | 486 | 9/9/2021 |
3.0.5.34 | 466 | 8/31/2021 |
3.0.5.33 | 453 | 8/26/2021 |
3.0.5.32 | 427 | 8/20/2021 |
3.0.5.31 | 431 | 8/20/2021 |
3.0.5.30 | 446 | 8/19/2021 |
3.0.5.29 | 441 | 8/18/2021 |
3.0.5.28 | 437 | 8/18/2021 |
3.0.5.27 | 429 | 8/18/2021 |
3.0.5.26 | 443 | 8/16/2021 |
3.0.5.25 | 429 | 8/13/2021 |
3.0.5.24 | 445 | 8/12/2021 |
3.0.5.23 | 583 | 7/23/2021 |
3.0.5.22 | 414 | 7/21/2021 |
3.0.5.21 | 478 | 7/16/2021 |
3.0.5.20 | 488 | 7/16/2021 |
3.0.5.19 | 477 | 7/6/2021 |
3.0.5.18 | 555 | 6/24/2021 |
3.0.5.17 | 558 | 6/12/2021 |
3.0.5.16 | 494 | 6/3/2021 |
3.0.5.15 | 491 | 6/1/2021 |
3.0.5.14 | 473 | 5/20/2021 |
3.0.5.13 | 467 | 5/13/2021 |
3.0.5.12 | 450 | 5/10/2021 |
3.0.5.10 | 488 | 4/27/2021 |
3.0.5.8 | 488 | 4/21/2021 |
3.0.5.7 | 466 | 4/20/2021 |
3.0.5.5 | 490 | 4/9/2021 |
3.0.5.4 | 479 | 4/8/2021 |
3.0.5.3 | 437 | 4/1/2021 |
3.0.5.2 | 448 | 3/31/2021 |
3.0.5.1 | 450 | 3/31/2021 |
3.0.5 | 487 | 3/26/2021 |
3.0.4.14 | 509 | 3/24/2021 |
3.0.4.13 | 483 | 3/10/2021 |
3.0.4.12 | 476 | 3/8/2021 |
3.0.4.11 | 471 | 3/5/2021 |
3.0.4.10 | 456 | 3/4/2021 |
3.0.4.9 | 471 | 3/3/2021 |
3.0.4.8 | 494 | 3/2/2021 |
3.0.4.7 | 532 | 2/26/2021 |
3.0.4.6 | 518 | 2/24/2021 |
3.0.4.5 | 517 | 2/17/2021 |
3.0.4.4 | 514 | 2/11/2021 |
3.0.4.3 | 507 | 2/11/2021 |
3.0.4.2 | 508 | 2/9/2021 |
3.0.4.1 | 487 | 2/3/2021 |
3.0.4 | 515 | 2/2/2021 |
3.0.3.9 | 561 | 1/26/2021 |
3.0.3.8 | 506 | 1/26/2021 |
3.0.3.7 | 514 | 1/20/2021 |
3.0.3.6 | 506 | 1/18/2021 |
3.0.3.5 | 562 | 1/14/2021 |
3.0.3.4 | 525 | 1/12/2021 |
3.0.3.3 | 501 | 1/11/2021 |
3.0.3.2 | 504 | 1/10/2021 |
3.0.3 | 530 | 1/5/2021 |
3.0.2.9 | 533 | 12/23/2020 |
3.0.2.8 | 538 | 12/22/2020 |
3.0.2.7 | 578 | 12/10/2020 |
3.0.2.6 | 563 | 12/9/2020 |
3.0.2.5 | 541 | 12/8/2020 |
3.0.2.4 | 601 | 11/20/2020 |
3.0.2.3 | 558 | 11/19/2020 |
3.0.2.2 | 572 | 11/19/2020 |
3.0.2.1 | 587 | 11/18/2020 |
3.0.2 | 581 | 11/17/2020 |
3.0.1.49 | 573 | 11/11/2020 |
3.0.1.48 | 558 | 11/9/2020 |
3.0.1.47 | 568 | 10/27/2020 |
3.0.1.46 | 731 | 10/13/2020 |
3.0.1.45 | 581 | 10/12/2020 |
3.0.1.44 | 553 | 10/12/2020 |
3.0.1.43 | 588 | 10/8/2020 |
3.0.1.42 | 549 | 10/6/2020 |
3.0.1.41 | 563 | 10/5/2020 |
3.0.1.40 | 560 | 10/2/2020 |
3.0.1.39 | 543 | 10/1/2020 |
3.0.1.38 | 578 | 9/30/2020 |
3.0.1.37 | 559 | 9/30/2020 |
3.0.1.35 | 619 | 9/28/2020 |
3.0.1.34 | 604 | 9/11/2020 |
3.0.1.33 | 627 | 9/3/2020 |
3.0.1.32 | 589 | 9/1/2020 |
3.0.1.31 | 581 | 8/26/2020 |
3.0.1.30 | 587 | 8/25/2020 |
3.0.1.29 | 591 | 8/21/2020 |
3.0.1.28 | 571 | 8/18/2020 |
3.0.1.27 | 580 | 8/18/2020 |
3.0.1.26 | 565 | 8/17/2020 |
3.0.1.25 | 547 | 8/14/2020 |
3.0.1.24 | 558 | 8/10/2020 |
3.0.1.23 | 628 | 8/6/2020 |
3.0.1.22 | 661 | 7/29/2020 |
3.0.1.21 | 632 | 7/29/2020 |
3.0.1.20 | 585 | 7/29/2020 |
3.0.1.19 | 674 | 7/16/2020 |
3.0.1.18 | 595 | 7/15/2020 |
3.0.1.17 | 604 | 7/13/2020 |
3.0.1.16 | 659 | 7/10/2020 |
3.0.1.15 | 562 | 7/10/2020 |
3.0.1.14 | 627 | 7/8/2020 |
3.0.1.13 | 566 | 7/7/2020 |
3.0.1.12 | 667 | 7/2/2020 |
3.0.1.11 | 569 | 7/2/2020 |
3.0.1.10 | 586 | 7/2/2020 |
3.0.1.9 | 548 | 7/1/2020 |
3.0.1.8 | 541 | 7/1/2020 |
3.0.1.7 | 595 | 6/30/2020 |
3.0.1.6 | 575 | 6/30/2020 |
3.0.1.5 | 580 | 6/23/2020 |
3.0.1.4 | 638 | 6/22/2020 |
3.0.1.3 | 646 | 6/15/2020 |
3.0.1.2 | 682 | 6/14/2020 |
3.0.0.1 | 660 | 6/12/2020 |
3.0.0 | 651 | 6/10/2020 |
0.4.34 | 587 | 6/5/2020 |
0.4.33 | 685 | 6/4/2020 |
0.4.32 | 633 | 6/3/2020 |
0.4.31 | 631 | 6/2/2020 |
0.4.30 | 691 | 5/29/2020 |
0.4.29 | 621 | 5/20/2020 |
0.4.28 | 609 | 5/19/2020 |
0.4.27 | 637 | 5/5/2020 |
0.4.26 | 696 | 5/3/2020 |
0.4.25 | 632 | 5/1/2020 |
0.4.24 | 623 | 4/27/2020 |
0.4.23 | 640 | 4/23/2020 |
0.4.22 | 585 | 4/20/2020 |
0.4.21 | 579 | 4/20/2020 |
0.4.20 | 630 | 4/13/2020 |
0.4.19 | 768 | 4/3/2020 |
0.4.18 | 638 | 3/30/2020 |
0.4.16 | 759 | 3/28/2020 |
0.4.15 | 638 | 3/26/2020 |
0.4.14 | 703 | 3/25/2020 |
0.4.13 | 617 | 3/20/2020 |
0.4.12 | 599 | 3/19/2020 |
0.4.11 | 633 | 3/17/2020 |
0.4.10 | 626 | 3/10/2020 |
0.4.8 | 604 | 3/6/2020 |
0.4.7 | 595 | 3/6/2020 |
0.4.6 | 598 | 3/6/2020 |
0.4.5 | 641 | 3/5/2020 |
0.4.4 | 651 | 3/4/2020 |
0.4.3 | 628 | 2/27/2020 |
0.4.2 | 578 | 2/26/2020 |
0.4.1 | 633 | 2/18/2020 |
0.4.0 | 748 | 2/17/2020 |
0.0.3.24 | 660 | 2/17/2020 |
0.0.3.23 | 599 | 2/13/2020 |
0.0.3.22 | 650 | 2/10/2020 |
0.0.3.21 | 734 | 2/8/2020 |
0.0.3.20 | 748 | 1/30/2020 |
0.0.3.18 | 649 | 1/27/2020 |
0.0.3.17 | 669 | 1/17/2020 |
0.0.3.16 | 620 | 1/17/2020 |
0.0.3.15 | 607 | 1/17/2020 |
0.0.3.14 | 759 | 1/9/2020 |
0.0.3.13 | 679 | 1/8/2020 |
0.0.3.12 | 675 | 1/7/2020 |
0.0.3.11 | 659 | 1/6/2020 |
0.0.3.10 | 744 | 1/3/2020 |
0.0.3.9 | 719 | 1/2/2020 |
0.0.3.8 | 732 | 12/31/2019 |
0.0.3.7 | 683 | 12/20/2019 |
0.0.3.6 | 872 | 12/18/2019 |
0.0.3.5 | 675 | 12/17/2019 |
0.0.3.4 | 714 | 12/16/2019 |
0.0.3.3 | 734 | 12/2/2019 |
0.0.3.2 | 646 | 11/20/2019 |
0.0.3.1 | 696 | 11/18/2019 |
0.0.3 | 665 | 11/7/2019 |
0.0.2.9 | 695 | 11/1/2019 |
0.0.2.8 | 676 | 10/30/2019 |
0.0.2.7 | 787 | 10/26/2019 |
0.0.2.6 | 794 | 10/24/2019 |
0.0.2.5 | 675 | 10/22/2019 |
0.0.2.4 | 1,442 | 10/11/2019 |
0.0.2.3 | 678 | 10/11/2019 |
0.0.2.2 | 701 | 10/1/2019 |
0.0.2.1 | 689 | 9/26/2019 |
0.0.2 | 714 | 9/26/2019 |