BLAM3.SDK 3.1.5.12

There is a newer version of this package available.
See the version list below for details.
dotnet add package BLAM3.SDK --version 3.1.5.12                
NuGet\Install-Package BLAM3.SDK -Version 3.1.5.12                
This command is intended to be used within the Package Manager Console in Visual Studio, as it uses the NuGet module's version of Install-Package.
<PackageReference Include="BLAM3.SDK" Version="3.1.5.12" />                
For projects that support PackageReference, copy this XML node into the project file to reference the package.
paket add BLAM3.SDK --version 3.1.5.12                
#r "nuget: BLAM3.SDK, 3.1.5.12"                
#r directive can be used in F# Interactive and Polyglot Notebooks. Copy this into the interactive tool or source code of the script to reference the package.
// Install BLAM3.SDK as a Cake Addin
#addin nuget:?package=BLAM3.SDK&version=3.1.5.12

// Install BLAM3.SDK as a Cake Tool
#tool nuget:?package=BLAM3.SDK&version=3.1.5.12                

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.

Blue Lucy SDK Logo

NuGet Packages

Package Latest Version
BLAM3.SDK NuGet

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 Ids.

// 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 Compatible and additional computed target framework versions.
.NET net7.0 is compatible.  net7.0-android was computed.  net7.0-ios was computed.  net7.0-maccatalyst was computed.  net7.0-macos was computed.  net7.0-tvos was computed.  net7.0-windows was computed.  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.  net9.0 was computed.  net9.0-android was computed.  net9.0-browser was computed.  net9.0-ios was computed.  net9.0-maccatalyst was computed.  net9.0-macos was computed.  net9.0-tvos was computed.  net9.0-windows was computed. 
Compatible target framework(s)
Included target framework(s) (in package)
Learn more about Target Frameworks and .NET Standard.

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 375 12/4/2024
3.1.5.12 2,089 5/16/2024
3.1.5.11 291 5/7/2024
3.1.5.8 472 4/3/2024
3.1.4.9 512 2/6/2024
3.1.4.8 128 2/5/2024
3.1.3.19 181 2/5/2024
3.1.3.16 1,746 10/4/2023
3.1.0.25 331 8/14/2023
3.1.0.24 334 7/25/2023
3.1.0.23 210 7/17/2023
3.1.0.19 433 6/20/2023
3.1.0.18 217 6/8/2023
3.0.8.16 338 3/15/2023
3.0.8.11 358 2/7/2023
3.0.8.10 364 2/1/2023
3.0.7.16 465 11/29/2022
3.0.7.13 1,077 11/9/2022
3.0.6.62 489 10/4/2022
3.0.6.60 794 7/25/2022
3.0.6.59 505 7/15/2022
3.0.6.57 584 7/7/2022
3.0.6.49 590 6/15/2022
3.0.6.44 1,985 5/18/2022
3.0.6.42 550 5/11/2022
3.0.6.34 745 4/19/2022
3.0.6.33 598 4/7/2022
3.0.6.23 577 3/15/2022
3.0.6.21 549 3/7/2022
3.0.6.20 536 2/28/2022
3.0.6.14 568 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 445 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 416 7/21/2021
3.0.5.21 478 7/16/2021
3.0.5.20 488 7/16/2021
3.0.5.19 481 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 452 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 478 3/8/2021
3.0.4.11 473 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 520 2/24/2021
3.0.4.5 517 2/17/2021
3.0.4.4 516 2/11/2021
3.0.4.3 509 2/11/2021
3.0.4.2 508 2/9/2021
3.0.4.1 489 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 535 12/23/2020
3.0.2.8 538 12/22/2020
3.0.2.7 578 12/10/2020
3.0.2.6 564 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 589 11/18/2020
3.0.2 583 11/17/2020
3.0.1.49 575 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 583 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 562 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 629 9/3/2020
3.0.1.32 591 9/1/2020
3.0.1.31 581 8/26/2020
3.0.1.30 589 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 549 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 634 7/29/2020
3.0.1.20 586 7/29/2020
3.0.1.19 674 7/16/2020
3.0.1.18 595 7/15/2020
3.0.1.17 606 7/13/2020
3.0.1.16 659 7/10/2020
3.0.1.15 562 7/10/2020
3.0.1.14 629 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 588 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 684 6/14/2020
3.0.0.1 660 6/12/2020
3.0.0 653 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 639 5/5/2020
0.4.26 696 5/3/2020
0.4.25 634 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 632 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 677 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