This plugin makes it easy to manage assets when using iconik storage options with different price/performance/functionality. For example, you can store master copies of full resolution assets in Backblaze B2 Cloud Storage, working on the iconik proxy files in iconik and Premiere Pro. When editing is complete, you can use the plugin to copy the masters to a LucidLink Filespace for full-resolution corrections and rendering. Once the final renders are approved, you can again use the plugin to remove the copies from LucidLink, safe in the knowledge that the masters are safely archived in B2.
Here's how it works:
A videographer saves full resolution video files to a Backblaze B2 Cloud Storage bucket, either directly or via iconik Storage Gateway. The director can review the footage in the iconik browser UI, set in and out points, and work with the iconik proxy files in Premiere Pro.
When it's time to work with the full resolution files, a user right-clicks the collections and/or assets and selects 'Add to LucidLink'. iconik sends a custom action notification to the plugin, which uses the iconik Files API to request that all of the selected collections and assets are copied to the LucidLink storage. iconik relays the request to an instance of iconik Storage Gateway, which retrieves the files from B2 and saves them to its local LucidLink Filespace.
The production team can now access the full resolution assets via LucidLink, performing final corrections and rendering output in Premiere Pro at full resolution. If the workflow includes creating files in LucidLink, you can configure the iconik Storage Gateway to automatically upload them to Backblaze B2.
Once the rendered deliverable is approved, a user can right-click the collections and/or assets in iconik and select 'Remove from LucidLink'. iconik sends a custom action message to the plugin, which ensures that all assets in LucidLink are present in Backblaze B2 before deleting the files from LucidLink.
The plugin is implemented in Python and may be deployed as a standalone Flask app or as a Google Cloud Function.
You must create an iconik Storage with Storage Purpose Files and Storage Type Backblaze B2 with access to your Backblaze B2 bucket. Enable scan for the storage and set it to auto scan with an appropriate interval. Make a note of the storage ID.
You must deploy both iconik Storage Gateway (ISG) and the LucidLink client to a machine with access to iconik, LucidLink and B2. Either Vultr or Equinix Metal are ideal for this purpose, as they both enjoy zero cost egress from B2. If you deploy ISG elsewhere, you will be paying $10/TB for data downloaded from B2.
Configure the LucidLink client to access the desired FileSpace. Configure ISG to use the LucidLink directory as Files Storage. Make a note of the storage ID.
Create an iconik Application Token for the plugin and make a note of the token id and value.
Create a random string to use as a secret shared by iconik and the plugin. For example, with openssl
:
openssl rand -hex 32
Keep a note of the shared secret.
Clone this project:
git clone https://github.com/backblaze-b2-samples/b2-iconik-plugin.git
Change to the plugin directory and install the required Python modules:
cd b2-iconik-plugin
pip install -r requirements.txt
Create a file in the plugin directory named .env
containing your iconik token id, the shared secret you created and, optionally, the iconik format name, if it is not ORIGINAL
.
ICONIK_ID='<required: your iconik application token id>'
BZ_SHARED_SECRET='<required: your shared secret>'
FORMAT_NAMES='<optional: defaults to ORIGINAL,PPRO_PROXY>'
Open b2-iconik-plugin.service
and edit the User
,
WorkingDirectory
and ExecStart
entries to match your system configuration.
Deploy the plugin as a systemd service:
sudo cp b2-iconik-plugin.service /etc/systemd/system
sudo systemctl daemon-reload
sudo systemctl start b2-iconik-plugin
sudo systemctl status b2-iconik-plugin
You should see output similar to this:
● b2-iconik-plugin.service - Backblaze B2 iconik Plugin
Loaded: loaded (/etc/systemd/system/b2-iconik-plugin.service; disabled; vendor preset: enabled)
Active: active (running) since Tue 2022-08-16 19:06:36 UTC; 11s ago
Main PID: 743400 (gunicorn)
Tasks: 5 (limit: 4678)
Memory: 92.8M
CPU: 804ms
CGroup: /system.slice/b2-iconik-plugin.service
├─743400 /usr/bin/python3 /home/pat/.local/bin/gunicorn -b localhost:8000 -w 4 plugin:app
├─743401 /usr/bin/python3 /home/pat/.local/bin/gunicorn -b localhost:8000 -w 4 plugin:app
├─743402 /usr/bin/python3 /home/pat/.local/bin/gunicorn -b localhost:8000 -w 4 plugin:app
├─743403 /usr/bin/python3 /home/pat/.local/bin/gunicorn -b localhost:8000 -w 4 plugin:app
└─743404 /usr/bin/python3 /home/pat/.local/bin/gunicorn -b localhost:8000 -w 4 plugin:app
Aug 16 19:06:36 vultr systemd[1]: Started Backblaze B2 iconik Plugin.
Aug 16 19:06:36 vultr gunicorn[743400]: [2022-08-16 19:06:36 +0000] [743400] [INFO] Starting gunicorn 20.1.0
Aug 16 19:06:36 vultr gunicorn[743400]: [2022-08-16 19:06:36 +0000] [743400] [INFO] Listening at: http://127.0.0.1:8000 (743400)
Aug 16 19:06:36 vultr gunicorn[743400]: [2022-08-16 19:06:36 +0000] [743400] [INFO] Using worker: sync
Aug 16 19:06:36 vultr gunicorn[743401]: [2022-08-16 19:06:36 +0000] [743401] [INFO] Booting worker with pid: 743401
Aug 16 19:06:36 vultr gunicorn[743402]: [2022-08-16 19:06:36 +0000] [743402] [INFO] Booting worker with pid: 743402
Aug 16 19:06:36 vultr gunicorn[743403]: [2022-08-16 19:06:36 +0000] [743403] [INFO] Booting worker with pid: 743403
Aug 16 19:06:36 vultr gunicorn[743404]: [2022-08-16 19:06:36 +0000] [743404] [INFO] Booting worker with pid: 743404
Your plugin's endpoint comprises the instance's public IP address or hostname plus the Gunicorn port number. For example, if your plugin is running at 1.2.3.4, and you left the Gunicorn port as the default 8000, your plugin endpoint is http://1.2.3.4:8000/
You can test connectivity to the plugin by opening http://1.2.3.4:8000/add
in a browser. You should see an error response similar to:
{"message": "The method is not allowed for the requested URL."}
Note - for production deployment, you should deploy Nginx as an HTTP proxy for Gunicorn and configure Nginx as an HTTPS server.
-
Ensure that billing is enabled for the project.
-
Enable the Cloud Functions, Cloud Build and Secret Manager APIs.
-
If necessary, install and initialize the gcloud CLI.
-
Update and install gcloud components:
gcloud components update
-
If necessary, set up a Python development environment.
-
Clone this repository to a directory on your local machine:
git clone [email protected]:backblaze-b2-samples/b2-iconik-plugin.git
Create the file .env.yaml
in the project directory, with the following content:
ICONIK_ID: '<required: your iconik application token id>'
FORMAT_NAMES: '<optional: defaults to ORIGINAL,PPRO_PROXY>'
Create the following secret in Google Secret Manager:
bz-shared-secret: '<your shared secret>'
From the command line in the project directory, run
gcloud functions deploy iconik_handler \
--runtime python39 --trigger-http --allow-unauthenticated --env-vars-file .env.yaml
Make a note of the httpsTrigger.url
property, or find it with:
gcloud functions describe iconik_handler
It should look like this:
https://<GCP_REGION-PROJECT_ID>.cloudfunctions.net/iconik_handler
You'll use this endpoint when you create the custom actions in iconik in the next step.
You can test connectivity to the plugin by opening https://<GCP_REGION-PROJECT_ID>.cloudfunctions.net/iconik_handler/add
in a browser. You should see an error response similar to:
{"message": "The method is not allowed for the requested URL."}
You can view logs for the function with:
gcloud functions logs read
Run the included create_custom_actions.py
script with the endpoint of the plugin and the two storage IDs as arguments:
ICONIK_TOKEN = '<your iconik application token value>' \
python create_custom_actions.py --endpoint <your plugin endpoint> \
--b2_storage_id <your B2 storage ID in iconik> \
--ll_storage_id <your LucidLink storage ID in iconik>
You can delete the custom actions, if necessary, with:
ICONIK_TOKEN = '<your iconik application token value>' \
python delete_custom_actions.py --endpoint <your plugin endpoint>
Test adding a file to B2, then LucidLink:
- Copy a media file to B2. If you copy the file directly to B2, you will need to wait for iconik to scan the bucket. If you use iconik to do so, you will see the file in iconik immediately.
- Right click the file in the search page and select 'Add to LucidLink'
- Click the 'Admin' tab. You should see a new job with a name like 'Transfer asset myvideo.mp4 to LucidLink ISG'. The job should complete successfully.
- Verify that the file is available in the LucidLink Filespace.
Note that, as well as right-clicking the file in the search space, you can right click a collection containing the file, or click on the gear in the upper right corner of the asset page or collection page.
You can now manipulate the file via LucidLink as you normally would.
Test deleting the file from LucidLink:
- Right click the file in the search page and select 'Remove from LucidLink'
- Click the 'Admin' tab. You should see a new job with a name like 'Delete myvideo.mp4 from LucidLink ISG'. The job should complete successfully.
- Verify that the file has been deleted from the LucidLink Filespace.
- Verify that the file is still available in B2.
You are free to modify the code for your own purposes.
There is a set of tests covering expected operation and various errors.
% pytest
================================================================== test session starts ==================================================================
platform darwin -- Python 3.10.2, pytest-7.1.2, pluggy-1.0.0
rootdir: /Users/ppatterson/src/b2-iconik-plugin
collected 29 items
app_test.py ........... [ 37%]
gcp_test.py .. [ 44%]
iconik_test.py ..... [ 62%]
main_test.py ........... [100%]
================================================================== 29 passed in 4.62s ===================================================================