2 Answers
- Newest
- Most votes
- Most comments
1
Here is code that accepts a WAV file path as input and sends it to a Kinesis Video Streams stream:
// upload.js
const { KinesisVideoClient, GetDataEndpointCommand } = require("@aws-sdk/client-kinesis-video");
const ffmpeg = require('fluent-ffmpeg')
const aws4 = require('aws4')
const axios = require('axios')
const CancelToken = axios.CancelToken
const fs = require('fs')
const PipeViewer = require('pv')
function noop() { }
const HANGING_TIMEOUT = 20 * 1000
function getMkvStream(uri, onStart, onFinished) {
let processHanging = null
const audio = ffmpeg(uri)
.format('matroska')
.outputOptions(['-ar 8000', '-acodec pcm_s16le'
])
.on('progress', function (progress) {
console.log('Processing:', progress)
clearTimeout(processHanging)
processHanging = setTimeout(function () {
console.error('Hanging for more than 20 sec.')
audio.kill()
}, HANGING_TIMEOUT) // hanging for more than x sec
})
.on('start', function (commandLine) {
console.log('Spawned Ffmpeg with command: ' + commandLine)
onStart && onStart()
})
.on('end', function () {
console.log('Finished processing')
onFinished && onFinished()
})
.on('error', function (err, stdout, stderr) {
console.log('Cannot process video: ' + err.message)
clearTimeout(processHanging)
onFinished && onFinished(err) // eventually restart
})
return audio
}
function sendStream({ mkvFile, streamName, dataEndpoint, pwd }, onStart, onFinish) {
onFinish = onFinish || noop
const opts = {
service: 'kinesisvideo',
host: dataEndpoint,
path: '/putMedia',
method: 'POST', // NB. required!
headers: {
'Content-Type': 'application/json',
'x-amzn-stream-name': streamName,
"x-amzn-fragment-timecode-type": "ABSOLUTE",
"x-amz-content-sha256": "UNSIGNED-PAYLOAD"
}
}
const signed = aws4.sign(opts, pwd)
console.log(signed)
const reqUri = 'https://' + opts.host + opts.path
let cancel;
axios({
method: 'POST',
timeout: 40 * 1000,
url: reqUri,
headers: signed.headers,
data: mkvFile,
responseType: 'stream',
maxContentLength: Infinity, // required for stream!
cancelToken: new CancelToken(function executor(c) {
// An executor function receives a cancel function as a parameter
cancel = c
})
}).then(function (res) {
onStart && onStart(cancel)
res.data.on('data', function (fragRes) {
console.log(fragRes.toString())
fragRes = JSON.parse(fragRes.toString())
const EventType = fragRes.EventType
const ErrorCode = fragRes.ErrorCode
if (EventType === 'ERROR') {
// request aborted
onFinish(new Error('Fragment event type: ERROR - ' + ErrorCode))
}
})
}).catch(onFinish)
}
function startUpload({ audioUri, streamName, dataEndpoint, pwd }) {
// get stream
const ffmpegAudio = getMkvStream(audioUri, null, function (err) {
if (err) {
startUpload({ audioUri, streamName, dataEndpoint }) // start again
}
})
const pv = PipeViewer()
pv.on('info', function (str) {
console.log('Speed: ' + str.speed + ' - Transferred: ' + str.transferred)
})
// start send stream
sendStream({ mkvFile: ffmpegAudio.pipe(pv), streamName, dataEndpoint, pwd }, function started(cancel) {
setTimeout(function () {
cancel() // stop upload
ffmpegAudio.kill() // stop stream and raise err cb above
}, 40 * 60 * 1000) // 40 min
}, function finished(err) {
if (err) console.error(err)
})
}
// main
async function main() {
// node upload.js ./samples/input.wav
const args = process.argv.slice(2);
console.log(args)
// getMkvStream(args[0]).pipe().pipe(require('fs').createWriteStream('./test.mkv'), null, console.error)
const STREAM_NAME = 'my-kvs-stream';
const config = { region: 'us-east-1' };
const input = {
APIName: 'PUT_MEDIA',
StreamName: STREAM_NAME
}
const client = new KinesisVideoClient(config);
const command = new GetDataEndpointCommand(input);
const response = await client.send(command);
console.log({ response });
startUpload({
audioUri: args[0],
streamName: STREAM_NAME,
dataEndpoint: response.DataEndpoint.split('://')[1],
pwd: {
endpoint: 'kinesisvideo.us-east-1.amazonaws.com',
region: 'us-east-1',
accessKeyId: "xxx",
secretAccessKey: "xxx",
}
})
}
main();
To send the file, execute: node upload.js ./samples/input.wav
answered 2 years ago
0
Thought I never tried this out, I would suggest the following:
- Leverage JS api such as MediaDevices (https://developer.mozilla.org/en-US/docs/Web/API/MediaDevices/getUserMedia). You can refer to this tutorial to refer on how to use it in your browser: https://dobrian.github.io/cmp/topics/sample-recording-and-playback-with-web-audio-api/3.microphone-input-and-recording.html.
- Kinesis Video Streaming Javascript SDK(https://docs.aws.amazon.com/AWSJavaScriptSDK/v3/latest/clients/client-kinesis-video-signaling/index.html). Send audio to the stream and store it there and or in a separate database (DynamoDB), for later use if needed.
- Finally use Transcribe APIs. (https://docs.aws.amazon.com/transcribe/latest/dg/getting-started-sdk.html). You can take inspiration from this repo, specific for AmazonConnect, but still using a similar architecture: https://github.com/amazon-connect/amazon-connect-realtime-transcription which uses a Lambda function as glue between Kinesis stream and Transcribe service
Relevant content
- Accepted Answerasked a year ago
- asked 9 months ago
- AWS OFFICIALUpdated 10 months ago
- AWS OFFICIALUpdated 6 months ago
- AWS OFFICIALUpdated 7 months ago
- AWS OFFICIALUpdated 3 months ago
Can you provide more details on which commands to use from Kinesis Video Streaming Javascript SDK to PUT_MEDIA onto a KVS stream?
I believe that there is no direct putMedia support for JS. There is only for C++ and Java. Nevertheless I found this which may inspire you. It bypasses this limitation by issuing a rest call instead of using a putMedia method, which doesn’t exist in the Js sdk. https://gist.github.com/roccomuso/e65e80aecbe700db2576e751fa1a22c4.
Hope it helps