Skip to content

Commit

Permalink
FG-2933: Add recorder support for audio frames to MCAP recording demo
Browse files Browse the repository at this point in the history
  • Loading branch information
fgwt202412 committed Dec 15, 2024
1 parent 5db8c69 commit 5661220
Show file tree
Hide file tree
Showing 4 changed files with 72 additions and 4 deletions.
2 changes: 1 addition & 1 deletion website/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@
"@docusaurus/preset-classic": "2.4.1",
"@foxglove/eslint-plugin": "1.0.1",
"@foxglove/rostime": "1.1.2",
"@foxglove/schemas": "1.6.2",
"@foxglove/schemas": "github:foxglove/schemas#e2cd86e48762e9396737a00c56e1d45ebfae24e3",
"@foxglove/tsconfig": "2.0.0",
"@mcap/core": "workspace:*",
"@mdx-js/react": "1.6.22",
Expand Down
47 changes: 46 additions & 1 deletion website/src/components/McapRecordingDemo/Recorder.ts
Original file line number Diff line number Diff line change
@@ -1,8 +1,9 @@
import { Time, fromNanoSec } from "@foxglove/rostime";
import { Time, fromMillis, fromNanoSec } from "@foxglove/rostime";

Check failure on line 1 in website/src/components/McapRecordingDemo/Recorder.ts

View workflow job for this annotation

GitHub Actions / spellcheck

Unknown word (Millis)
import {
PoseInFrame,
CompressedImage,
CompressedVideo,
CompressedAudio,
} from "@foxglove/schemas";
import { foxgloveMessageSchemas } from "@foxglove/schemas/internal";
import zstd from "@foxglove/wasm-zstd";
Expand All @@ -11,6 +12,7 @@ import { EventEmitter } from "eventemitter3";
import Queue from "promise-queue";

import { ProtobufChannelInfo, addProtobufChannel } from "./addProtobufChannel";
import { CompressedAudioData } from "./audioCapture";
import { CompressedVideoFrame } from "./videoCapture";

export type ProtobufObject<Message> = {
Expand Down Expand Up @@ -60,6 +62,8 @@ export class Recorder extends EventEmitter<RecorderEvents> {
#vp9ChannelSeq = 0;
#av1Channel?: ProtobufChannelInfo;
#av1ChannelSeq = 0;
#opusChannel?: ProtobufChannelInfo;
#opusChannelSeq = 0;

#blobParts: Uint8Array[] = [];
bytesWritten = 0n;
Expand Down Expand Up @@ -112,6 +116,8 @@ export class Recorder extends EventEmitter<RecorderEvents> {
this.#h265ChannelSeq = 0;
this.#av1Channel = undefined;
this.#av1ChannelSeq = 0;
this.#opusChannel = undefined;
this.#opusChannelSeq = 0;
}

#time(): bigint {
Expand Down Expand Up @@ -277,6 +283,45 @@ export class Recorder extends EventEmitter<RecorderEvents> {
});
}

async addAudioData(data: CompressedAudioData): Promise<void> {
void this.#queue.add(async () => {
if (!this.#writer) {
return;
}
let channel: ProtobufChannelInfo;
let sequence: number;
switch (data.format) {
case "opus":
channel = this.#opusChannel ??= await addProtobufChannel(
this.#writer,
"microphone_opus",
foxgloveMessageSchemas.CompressedAudio,
);
sequence = this.#opusChannelSeq++;
break;
}
const { id, rootType } = channel;
const msg: ProtobufObject<CompressedAudio> = {
timestamp: toProtobufTime(fromMillis(data.timestamp)),

Check failure on line 305 in website/src/components/McapRecordingDemo/Recorder.ts

View workflow job for this annotation

GitHub Actions / spellcheck

Unknown word (Millis)
data: data.data,
format: data.format,
type: data.type,
sample_rate: data.sampleRate,
number_of_channels: data.numberOfChannels,
};
const now = this.#time();
await this.#writer.addMessage({
sequence,
channelId: id,
logTime: now,
publishTime: now,
data: rootType.encode(msg).finish(),
});
this.messageCount++;
this.#emit();
});
}

async closeAndRestart(): Promise<Blob> {
return await this.#queue.add(async () => {
await this.#writer?.end();
Expand Down
13 changes: 13 additions & 0 deletions website/src/components/McapRecordingDemo/audioCapture.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
type CompressedAudioFormat = "opus";
type CompressedAudioType = "key" | "delta";
export type CompressedAudioData = {
format: CompressedAudioFormat;
type: CompressedAudioType;
timestamp: number;
data: Uint8Array;
sampleRate: number;
numberOfChannels: number;

/** Call this function to release the buffer so it can be reused for new frames */
release: () => void;
};
14 changes: 12 additions & 2 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -3056,6 +3056,16 @@ __metadata:
languageName: node
linkType: hard

"@foxglove/schemas@github:foxglove/schemas#e2cd86e48762e9396737a00c56e1d45ebfae24e3":
version: 1.6.7
resolution: "@foxglove/schemas@https://github.com/foxglove/schemas.git#commit=e2cd86e48762e9396737a00c56e1d45ebfae24e3"
dependencies:
"@foxglove/rosmsg-msgs-common": "npm:^3.0.0"
tslib: "npm:^2.5.0"
checksum: 10c0/67b13339aa3f485e786c93ef22f45f67598db1c77c0fa062b30c22a85f3be8a655ae5c351279bccd98751dce64ba7c95ddbd0cc7ca77b3281795e195faaa042d
languageName: node
linkType: hard

"@foxglove/schemas@npm:1.3.0":
version: 1.3.0
resolution: "@foxglove/schemas@npm:1.3.0"
Expand All @@ -3066,7 +3076,7 @@ __metadata:
languageName: node
linkType: hard

"@foxglove/schemas@npm:1.6.2, @foxglove/schemas@npm:^1.0.0":
"@foxglove/schemas@npm:^1.0.0":
version: 1.6.2
resolution: "@foxglove/schemas@npm:1.6.2"
dependencies:
Expand Down Expand Up @@ -15610,7 +15620,7 @@ __metadata:
"@docusaurus/preset-classic": "npm:2.4.1"
"@foxglove/eslint-plugin": "npm:1.0.1"
"@foxglove/rostime": "npm:1.1.2"
"@foxglove/schemas": "npm:1.6.2"
"@foxglove/schemas": "github:foxglove/schemas#e2cd86e48762e9396737a00c56e1d45ebfae24e3"
"@foxglove/tsconfig": "npm:2.0.0"
"@mcap/core": "workspace:*"
"@mdx-js/react": "npm:1.6.22"
Expand Down

0 comments on commit 5661220

Please sign in to comment.