Skip to content

acringest

Extract data from loaded ACRCloud records.

app(bootstrap_servers, security_protocol, tls_cafile, tls_certfile, tls_keyfile, consumer_topic, consumer_group, consumer_auto_offset_reset, minio_url, minio_access_key, minio_secret_key, minio_bucket_raw, minio_bucket_music, minio_secure, minio_cert_reqs, minio_ca_certs)

Consume MinIO CloudEvents and create deduped and historized objects.

Source code in acringest.py
def app(  # noqa: PLR0912,PLR0913,C901
    bootstrap_servers: list[str],
    security_protocol: str,
    tls_cafile: str,
    tls_certfile: str,
    tls_keyfile: str,
    consumer_topic: str,
    consumer_group: str,
    consumer_auto_offset_reset: str,
    minio_url: str,
    minio_access_key: str,
    minio_secret_key: str,
    minio_bucket_raw: str,
    minio_bucket_music: str,
    minio_secure: bool,  # noqa: FBT001
    minio_cert_reqs: str,
    minio_ca_certs: str,
) -> None:
    """Consume MinIO CloudEvents and create deduped and historized objects."""
    consumer = KafkaConsumer(
        consumer_topic,
        bootstrap_servers=bootstrap_servers,
        security_protocol=security_protocol,
        group_id=consumer_group,
        auto_offset_reset=consumer_auto_offset_reset,
        ssl_cafile=tls_cafile,
        ssl_certfile=tls_certfile,
        ssl_keyfile=tls_keyfile,
    )

    def on_sigint(*_: Any) -> NoReturn:  # noqa: ANN401 # pragma: no cover
        consumer.close()
        sys.exit(0)

    signal.signal(signal.SIGINT, on_sigint)

    mc = Minio(
        minio_url,
        minio_access_key,
        minio_secret_key,
        secure=minio_secure,
        http_client=urllib3.PoolManager(
            cert_reqs=minio_cert_reqs,
            ca_certs=minio_ca_certs,
        ),
    )
    for bucket in [minio_bucket_raw, minio_bucket_music]:
        if not mc.bucket_exists(bucket):  # pragma: no cover
            mc.make_bucket(bucket)

    for msg in consumer:
        headers: list[tuple[str, bytes]] = msg.headers if msg.headers else []
        ce: CloudEvent = from_structured(
            message=KafkaMessage(
                key=msg.key,
                value=msg.value,
                headers={h[0]: h[1] for h in headers},
            ),
        )
        if (
            ce["source"] == "minio:s3..acrcloud.raw"
            and ce["type"] == "com.amazonaws.s3.s3:ObjectCreated:Put"
        ):
            bucket = ce.data.get("s3", {}).get("bucket", {}).get("name")
            name = ce.data.get("s3", {}).get("object", {}).get("key")
            obj = mc.get_object(bucket, name)
            for data in obj.json():
                for music in data.get("metadata", {}).get("music", []):
                    acrid = music.get("acrid")
                    for key in list(music.keys()):
                        if key not in _ACR_EXPECTED_KEYS:  # pragma: no cover
                            logger.error("Unexpected key %s in acr results", key)
                    for to_del in _ACR_IGNORED_KEYS:
                        if to_del in music:
                            del music[to_del]
                    minio_data = None
                    try:
                        minio_data = mc.get_object(minio_bucket_music, acrid).json()
                    except S3Error as ex:  # pragma: no cover
                        if ex.code == "NoSuchKey":
                            put_data(mc, minio_bucket_music, acrid, json.dumps(music))
                            minio_data = music
                    changes = diff(minio_data, music)
                    if changes:
                        put_data(mc, minio_bucket_music, acrid, json.dumps(music))
                        logger.info("Applied changes to acrid=%s: %s", acrid, changes)

main()

Run main.

Source code in acringest.py
def main() -> None:  # pragma: no cover
    """Run main."""
    parser = ArgumentParser(__name__)
    parser.add(
        "--kafka-bootstrap-servers",
        required=True,
        env_var="KAFKA_BOOTSTRAP_SERVERS",
    )
    parser.add(
        "--kafka-security-protocol",
        default="PLAINTEXT",
        env_var="KAFKA_SECURITY_PROTOCOL",
    )
    parser.add(
        "--kafka-tls-cafile",
        default=None,
        env_var="KAFKA_TLS_CAFILE",
    )
    parser.add(
        "--kafka-tls-certfile",
        default=None,
        env_var="KAFKA_TLS_CERTFILE",
    )
    parser.add(
        "--kafka-tls-keyfile",
        default=None,
        env_var="KAFKA_TLS_KEYFILE",
    )
    parser.add(
        "--kafka-consumer-topic",
        default="cloudevents",
        env_var="KAFKA_CONSUMER_TOPIC",
    )
    parser.add(
        "--kafka-consumer-group",
        default=__name__,
        env_var="KAFKA_CONSUMER_GROUP",
    )
    parser.add(
        "--kafka-consumer-auto-offset-reset",
        default="latest",
        env_var="KAFKA_CONSUMER_AUTO_OFFSET_RESET",
    )
    parser.add(
        "--minio-url",
        default="minio.service.int.rabe.ch:9000",
        env_var="MINIO_HOST",
        help="MinIO Hostname",
    )
    parser.add(
        "--minio-secure",
        default=True,
        env_var="MINIO_SECURE",
        help="MinIO Secure param",
    )
    parser.add(
        "--minio-cert-reqs",
        default="CERT_REQUIRED",
        env_var="MINIO_CERT_REQS",
        help="cert_reqs for urlib3.PoolManager used by MinIO",
    )
    parser.add(
        "--minio-ca-certs",
        default="/etc/pki/ca-trust/extracted/openssl/ca-bundle.trust.crt",
        env_var="MINIO_CA_CERTS",
        help="ca_certs for urlib3.PoolManager used by MinIO",
    )
    parser.add(
        "--minio-bucket-raw",
        default="acrcloud.raw",
        env_var="MINIO_BUCKET_RAW",
        help="MinIO Bucket with raw ACRCloud data",
    )
    parser.add(
        "--minio-bucket-music",
        default="acrcloud.music",
        env_var="MINIO_BUCKET_MUSIC",
        help="MinIO Bucket with music ACRCloud data",
    )
    parser.add(
        "--minio-access-key",
        default=None,
        env_var="MINIO_ACCESS_KEY",
        help="MinIO Access Key",
    )
    parser.add(
        "--minio-secret-key",
        default=None,
        env_var="MINIO_SECRET_KEY",
        help="MinIO Secret Key",
    )
    parser.add(
        "--quiet",
        "-q",
        default=False,
        action="store_true",
        env_var="ACRINGEST_QUIET",
    )
    parser.add(
        "--debug",
        "-d",
        default=False,
        action="store_true",
        env_var="ACRINGEST_DEBUG",
    )

    options = parser.parse_args()

    if not options.quiet:
        logging.basicConfig(level=logging.INFO)
    if options.debug:
        logging.basicConfig(level=logging.DEBUG)
    logger.info("Starting %s", __name__)

    app(
        bootstrap_servers=options.kafka_bootstrap_servers,
        security_protocol=options.kafka_security_protocol,
        tls_cafile=options.kafka_tls_cafile,
        tls_certfile=options.kafka_tls_certfile,
        tls_keyfile=options.kafka_tls_keyfile,
        consumer_topic=options.kafka_consumer_topic,
        consumer_group=options.kafka_consumer_group,
        consumer_auto_offset_reset=options.kafka_consumer_auto_offset_reset,
        minio_url=options.minio_url,
        minio_access_key=options.minio_access_key,
        minio_secret_key=options.minio_secret_key,
        minio_bucket_raw=options.minio_bucket_raw,
        minio_bucket_music=options.minio_bucket_music,
        minio_secure=options.minio_secure,
        minio_cert_reqs=options.minio_cert_reqs,
        minio_ca_certs=options.minio_ca_certs,
    )

put_data(mc, bucket, acrid, data)

Put data to MinIO.

Source code in acringest.py
def put_data(mc: Minio, bucket: str, acrid: str, data: str) -> None:
    """Put data to MinIO."""
    _as_bytes = data.encode("utf-8")
    mc.put_object(
        bucket,
        acrid,
        BytesIO(_as_bytes),
        length=len(_as_bytes),
        content_type="application/json",
    )