The place where random ideas get written down and lost in time.

2023-06-26 - Web Stats Alternatives - MQTT?

Category DEV

The Looker+GA4 behavior is currently limiting. It also exposes the typical issue when relying on external “free” services. Are there ways I can bring my stats in-house?

2 aspects come to mind:

  • Sending the event data…
    • With the usual caveat of outage reliability, the data would be better hosted on my own server.
    • An option would be to use MQTT to transport that data around.
  • Viewing the data…
    • Can I write my own Looker connector pulling data from my server?
    • Are there other MQTT-based or similar web-based stats pages I can reuse?

I’m not particularly interested in writing my own graphs, even though I’ve certainly done that.

There are Linux packages that can pre-render graphs and I can expose them via ssl.

Can MQTT be used as either a transport mechanism, storage, or a data source?


2023-06-17 - BitBucket SSH Host Key Migration

Category DEV

What: https://bitbucket.org/blog/ssh-host-key-changes

Verify by running this command on each host:

$ ssh git@bitbucket.org host_key_info

> You are using host key with fingerprint:

> ssh-ed25519 SHA256:ybgmFkzwOSotHTHLJgHO0QN8L0xErw6vd0VhFA9m3SM

> ecdsa-sha2-nistp256 SHA256:FC73VB6C4OQLSCrjEayhMp9UMxS97caD/Yyi2bhW/J0

ECDA or ED25519 are GOOD.

Got “Permission denied”?

https://confluence.atlassian.com/bbkb/permission-denied-publickey-302811860.html

⇒ This should happen for any machine where I did not load an SSH Key in my Bitbucket account.

⇒ If I have an ssh-agent key, load it first.

⇒ If I don’t expect to have ssh auth, then it’s fine. In this case, grep bitbucket ~/.ssh/known_hosts ⇒ vim ~/.ssh/known_hosts and remove any reference to bitbucket).

To accept the new host on a machine that needs git access:

$ ssh-keygen -R bitbucket.org && curl https://bitbucket.org/site/ssh >> ~/.ssh/known_hosts

then rerun the host_key_info cmd, access the key with “yes”, and repeat again to remove any duplicated host entry.


2023-06-10 - Brace Yourself, Google Analytics 4 is Coming

Category DEV

No choice, GA is giving me more and more warnings about the incoming GA4 migration.

Tasks:

  • Build an exhaustive list of projects actually exporting to GA4.
  • Find how to send GA4 events from my bash scripts.

Taking a reverse approach to that, the Stats Pages I look daily are:

  • The Randall Camera Access stats. Data comes from wazz.js.
    • ⇒ This seems like a good contender to learn the “official” JS API.
  • The Train Activation stats generated by Conductor.
    • This uses ad-hoc URL pings.
    • And also uses bash-based wget pings.
  • The Servers Ping stats.  This uses bash-based wget pings.
    • ⇒ This seems like the best candidate to look for a wget alternative.
  • The Inkblot Access stats. This uses the official Flutter library.

Migration guide: https://support.google.com/analytics/answer/10759417

  • Universal Analytics (UA) is the old one.
  • Setup Assistant: https://support.google.com/analytics/answer/9744165
  • Account Structure: https://support.google.com/analytics/answer/9679158
    • UA structure: Account ⇒ N * Properties (e.g. website) ⇒ M * Views.
      • The Property number starts with UA-
    • GA4 Structure: Account ⇒ N * Properties (e.g. website) ⇒ Data Streams.
      • The property number is a long number.
      • Looks like GA4 numbers have already been created for each one.
  • Events are changing: https://support.google.com/analytics/answer/11091422
    • UA events have events category > event action > event label.
    • GA4 events are firebase events: event name + event parameters.
    • UA event action becomes the new GA4 event name
    • UA event category becomes a GA4 event custom parameter
    • UA event label becomes a GA4 event custom parameter


2023-06-09 - Rig4j Needs a Rewrite

Category DEV

Back there on the RIG 4 Spec [Idea] project page, I have a different angle for a rewrite of Rig4.

The summary is that the current version of Rig4j is still running the “early experiment”. Code base was mostly “feasibility”. When exploring the Rust-based Rig4r, I realized it would be better structured by separating the gdoc functionality from the “rig” generator functionality: one tool does the gdoc fetching, and then Rig works on purely local files.

The new angle is that the gdoc fetching should actually generate izu files, by parsing the gdoc html exports and rewriting them as izu/markdown text files. There would be some limitations on the format. Not all the formatting would make it through.

This would also involve looking at the format I use in the train blog, making an exhaustive list of features used, and try to match some level of parity. Luckily I don’t use too much stuff, but tables seem like they could be tricky, and I’d also want to support the linkable TOCs and their existing hrefs. Some things like usage of titles in article pages did not exist in izu.

I may also want to consider dropping my izu format, and focus on “official” markdown.

I can probably fairly simply translate old izu pages to markdown if such support is needed.

OTOH it would solve one of the current problems, namely that the page format is dictated by the gdoc html export, and that has been known to change substantially over the time.

The end goal is to not have such a strong dependence on gdoc. That makes gdoc merely a convenience to more easily write the izu files, and then the izu files become the “source of truth” that I need to backup.


2023-06-07 - Rig4j: "This file is too large to be exported."

Category DEV

This is new:

java.lang.RuntimeException: com.google.api.client.googleapis.json.GoogleJsonResponseException: 403 Forbidden

{

  "code" : 403,

  "errors" : [ {

    "domain" : "global",

    "message" : "This file is too large to be exported.",

    "reason" : "exportSizeLimitExceeded"

  } ],

  "message" : "This file is too large to be exported."

}

        at com.alflabs.rig4.gdoc.GDocHelper.lambda$getGDocAsync$0(GDocHelper.java:491)

        at com.alflabs.rig4.struct.GDocEntity.getContent(GDocEntity.java:54)

        at com.alflabs.rig4.blog.BlogGenerator.parseSource(BlogGenerator.java:135)

        at com.alflabs.rig4.blog.BlogGenerator.parseSources(BlogGenerator.java:120)

        at com.alflabs.rig4.blog.BlogGenerator.processEntries(BlogGenerator.java:69)

        at com.alflabs.rig4.exp.Exp.start(Exp.java:60)

        at com.alflabs.rig4.EntryPoint.main(EntryPoint.java:53)

As usual, that makes me reconsider my dependency on the GDrive API.


2023-06-04 - DaVinci Fusion Fuse

Category DEV

Here’s the end result of the Fuse to perform image removal on my train videos:

https://bitbucket.org/alf-labs/lightworks/src/master/fusion/fuses/RalfCamCarRodRemoval.fuse

and here’s an example of the result:

https://www.alfray.com/trains/blog/train/2023-06-04_davinci_resolve_plugin_for_t_4126bb12.html


2023-05-21 - Lua Optimization Tips

Category DEV

https://www.lua.org/gems/sample.pdf

  • Locals are faster to access than globals. Copy globals to locals especially for loop access.
  • Tables are composed of an array (storing integer keys from 1..n) and a hash map for any other keys.


2023-05-17 - DaVinci Fusion Plugin

Category DEV

How to write a Fusion video effect plugin?

This SO answer leads to “we suck less” (aka WSL), and “eyeon”.

That last name “eyeon” is the company which originally created Fusion before being acquired by BlackMagic to integrate it in DaVinci Resolve.

  • https://www.steakunderwater.com/wesuckless/ is the place to find a community behind plugins.
  • API should be available via Help > Documentation > Developer. There’s a PDF IIRC.
  • An older version (Fusion 8, 2016) can be found here: https://documents.blackmagicdesign.com/UserManuals/Fusion8_Scripting_Guide.pdf
  • Scripting:
    • “FusionScript”, to be used either in Lua or Python.
    • Uses LuaJIT for performance. Lua is the preferred choice.
    • Choice of Python 2 or 3.
  • Fuses:
    • OpenCL for tools & filters
  • Scripts:
    • Composition scripts (a “composition” is a full Fusion document)
    • Tool scripts (for a “single tool” aka a single node?)
    • Bin scripts (as in “media bin”)
    • Utility scripts (act on Fusion itself)
    • Script libraries (scriptlib, used by other scripts)
    • External command-line script which can act on a composition.
    • Event scripts.
      • Composition callbacks (load, save, render, etc)
      • Button callbacks (for UI)
    • InTool scripts (executed when evaluating each frame)

Fuses has its own SDK documentation:

  • https://documents.blackmagicdesign.com/UserManuals/Fusion_Fuse_SDK.pdf?_v=1658361162000
  • Locally as “Fusion Fuse Manual.pdf” in C:\ProgramData\Blackmagic Design\DaVinci Resolve\Support\Developer\Fusion Fuse
  • Plugin types:
    • Image Processing, with inspector and onscreen crosshair.
    • Metadata processing.
    • Modified plugins (affects number inputs)
    • View LUT plugins.
  • Using Lua with LuaJIT.
  • What I’d use is an Image Processing “Tool” plugin.
    • Plugin has code to register its name.
    • Code to register inputs (UI controls) but also Image input/output.
    • Callback NotifyChanged when UI controls change (e.g. to adjust UI).
    • Callback Process to process an image and compute an output.
    • Native methods that apply to the entire image:
      • ColorMatrixFull (RGBA offset, scale, etc)
      • RGBA Color Gain, Gamma, Saturate.
      • Color Space conversion (RGB, HSV, etc)
      • Clear/Fill
      • Channel operations on 1 image with + - * / and RGB coefficients.
      • Channel operations on 2 images with RGB fg vs bg: Copy, Add, Multiply, Subtract, Divide, Threshold, And, Or, Xor, Negative, Difference.
      • Transform or Merge (center, pivot, size, angle, edge wrapping)
      • Crop, Resize
      • Blur (also for Glow)
    • Pixel Processing : create functions that take pixels from 2 images and return the resulting pixel.
      • Functions are “pre declared” and stored in an array, then used in Process.
    • Processing has 8 channels: 4x RGBA and 4x Bg-RGBA (I guess for the background?)
    • Shapes: created using MoveTo/LineTo + matrix operations then filled and merged with image.
    • Text overlays: font management, then draw as shapes.
    • Per-pixel processing using for y/for x/getPixel/setPixel.
  • DCTL (DaVinci Color Transform Language)  is a shader-style processing.
    • C-like syntax, operates om x,y individual pixels.
    • Converted to GPU as needed.
    • Transform DCTL to change color data in one image.
    • Transition DCTL to compute transition between two images.
  • OpenFX
    • C/C++ shader with VS.Net SLN project.

Some example of Fuses here:

https://www.steakunderwater.com/VFXPedia/96.0.243.189/index8be6.html?title=Eyeon:Script/Reference/Applications/Fuse/Example_Fuses#Example_Fuses


2023-05-16 - Video Camera Object Removal

Category DEV

Filming trains with the video car pulled by my “extended” coupler leaves that coupler visible on screen. It’s a tad distracting.

One question I had is: Can I write a Fusion plugin that would “erase” this object?

Let’s look at the obvious part: how would I erase this object?

We can split the problem in two: how to detect the object, and how to erase it.

First, the “coupler” appears as a gray-ish rectangle. It always touches the bottom edge of the screen and extends upwards. It’s roughly centered horizontally, but can move left and right as the train takes on curves. If we had to modify a movie frame by frame, one optimization would be on a given frame the coupler will very close to its position in the previous frame.

The coupler is more or less a vertical rectangle.

If we were operating as a plugin in Fusion, ideally we could use a 2d ui crosshair to define its initial position, but we would also have to deal with the fact it moves slowly from frame to frame, and tracking would be ideal.

If we were operating as command-line script without UI, the initial position could be given as parameters.

Its width is constant. It tapers as it gets farther from the camera, but overall we could parametrize the max/min width we are looking for -- e.g. what looks like a vertical gray box of W pixels wide and at least H pixels height starting from the bottom of the screen, roughly centered?

If we have a UI giving us a starting pixel position, we can use a basic flood algorithm to detect the boundaries.

When looking at a still image capture, the coupler edges are naturally fuzzy. We cannot expect a stark transition between the coupler and its background. However we can parametrize this, as typically the fuzzy border will have a predictable width.

Although the coupler is mostly of the same gray color, that color can vary depending on illumination, not to mention tunnels, and there’s bound to be shade variation across the length of the object.

One option is to look at the image in HSL -- the coupler being gray should not have a hue.

Assuming we can detect the coupler, how do we “erase” it?

One suggestion is to treat the coupler as a mask of width W:

  • Shift the image horizontally left and right by W pixels.
  • Merge this over the coupler using an averaging algorithm.
  • Using the luminance as the coefficient to merge it.

Assuming this works for one image, the obvious concern is how will it look once applied to a 30 fps movie? We may need to do some temporal averaging -- that is apply a percentage of the last frame to smooth the output.

Implementation wise, why kind of framework would be used?

One option is to make a standalone processing app, either in Java/Kotlin/C++, with its own movie decoding + UI rendering. The input movie decoding would take advantage of either ffmpeg or vlc as libraries or avcodec on linux. Similarly this can write a movie result on the fly.

In this mode, we would convert a raw footage upfront -- cam files ar 10 minutes long.

The other option is to use a Fusion plugin as the framework -- in this case Fusion provides the input and the output. Debugging seems potentially more challenging this way. The Fuse SDK doc indicates the Lua code can be reloaded “live” to ease debugging. This has the side benefit that we can judge performance immediately, and that we can take advantage of the fuse UI widgets to e.g. select the initial position to look at or to select some parameters (width, blending, etc).

In this mode, we could apply the tool to only the segments being rendered in the final production rather than on the entire raw footage.

One thing I don’t see in the Fuse SDK is a way for a “tool” to incrementally operate across frames in an animation. We want to do some basic tracking of the coupler position, and it would be nice if it could be automated. The other option is to simply animate the initial position, and/or to use a planar tracker to define it.


2023-02-24 - MQTT

Category DEV

Just leaving this here to explore it later: https://docs.espressif.com/projects/esp-idf/en/latest/esp32/api-reference/protocols/mqtt.html

 

Guides:

Spec for MQTT v5:

This would be useful for SDB: make it an MQTT client.

But potentially that can be useful for other projects.

For example, my current layout stats from Randall are sent via very hackish REST post (hackish in the sense it’s received via a BASH CGI :-p ⇒ not anymore, it’s a CGI py), and it may be better served by using this protocol: e.g. Conductor sending events to home via broker/gateways, and I'd want to better understand the possibilities.

From the adafruit guide above:

  • An MQTT broker is a “stable” node to which (potentially ephemeral) MQTT clients connect. It’s basically the “local server” of a bunch of clients.
    • Brokers typically have their own local database but they do not have to.
  • An MQTT client is any end-point sending or receiving messages.
    • Publish: A client sends a message to a broker.
    • Subscribe: A client receives messages from a broker.
    • A client keeps an open connection to a broker. The connection open/close is lightweight enough that it can be beneficial to open/close it when needed.
  • MQTT messages have a fairly lightweight header, designed for lightweight payloads.
    • The protocol is mostly transport-agnostic.
    • The spec above recommends TCP, TLS, or WebSockets, and recommends avoiding UDP due to lack of ordering guarantee.
    • QOS: 0=at most once delivery, 1=at least once delivery, 2=exactly once delivery.
  • Messages are sent to “topics”: a hierarchical string e.g. /root/part1/…/partn.
    • Concrete example: /building1/sensors/floor1/division2/temperature.
    • Subscription can be done using wildcards: + for any given part, and # for “anything till the end”.
    • Topics should be carefully defined upfront, as changing the schema later is not trivial especially with regards to wildcard subscriptions.
  • An MQTT gateway is a client/bridge translating data from sensors to MQTT messages (example: sensor ⇔ RF/BLE ⇔ gateway ⇔ MQTT messages ⇔  broker).

MQTT Libraries


 Generated on 2025-01-18 by Rig4j 0.1-Exp-f2c0035