Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Astroport #42

Open
wants to merge 33 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
33 commits
Select commit Hold shift + click to select a range
8dd779c
record and post treatments
May 28, 2024
0f239db
record and post treatments
May 28, 2024
e84f568
record a visio room into mp4 or wav file
papiche May 28, 2024
996c3ec
Merge branch 'steveseguin:main' into main
papiche May 29, 2024
d4a2610
Merge branch 'steveseguin:main' into main
papiche May 30, 2024
1a51733
record.py http://0.0.0.0:8000
papiche May 30, 2024
dfea305
run me
papiche May 30, 2024
cf8b185
generate
papiche May 30, 2024
cea5a3c
import random
papiche May 30, 2024
125188c
wait for "DATA CHANNEL: CLOSE" in publish.py's logs
papiche May 30, 2024
a774c53
generate_record_id
papiche May 30, 2024
aeb6654
DATA CHANNEL: CLOSE detection
papiche May 30, 2024
929c658
stt/ record.py whisper trasncriptions output directory
papiche May 30, 2024
b633b99
stt
papiche May 30, 2024
1ad10cb
stt
papiche May 30, 2024
a28d176
what is happening
papiche May 30, 2024
0c427d8
bug
papiche May 30, 2024
54ca8d6
last stable
papiche May 30, 2024
e3623db
stop recording file
papiche May 30, 2024
3073901
record audio.ts, but does not detect DATA CHANNEL: CLOSE to engage wh…
papiche May 30, 2024
a9b8068
record & transcribe microApp
Jun 8, 2024
963814a
Record transcribe and translate Vdo.Ninja Room http://localhost:8000
Jun 8, 2024
e2adf1c
WebRTC vdo.ninja record to whisper & IA service
Jun 9, 2024
ac2477e
record you
Jun 11, 2024
3ef8345
UX². manual & automatic audio to text through whisper
Jun 13, 2024
7d72476
record.py + README
Jun 13, 2024
9849114
https://vdo.ninja/?push=myRecord&room=myRoom&password=false&effects&r…
Jun 13, 2024
f2df5eb
record.py 1.0
Jun 13, 2024
9240a5e
typo
Jun 13, 2024
f3bef07
typo²
Jun 13, 2024
135ba01
trying medium whisper model needs 6GB VRAM
Jun 17, 2024
55afb02
setup systemd record.py --host 0.0.0.0 --port 9000
Jun 18, 2024
2ca7887
Merge branch 'main' into astroport
steveseguin Jun 18, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -127,3 +127,7 @@ dmypy.json

# Pyre type checker
.pyre/

# record.py whisper trasncriptions output directory
*_audio.ts
stt/
107 changes: 107 additions & 0 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -392,6 +392,113 @@ If using a virtual MIDI device on the remote viewer's computer, such as `loopMID
Please note, the raspberry_ninja publish.py script can both send and recieve MIDI commands over a single peer connection, which is a bit different than how video/audio work currently. It's also different than how browser to browser currently is setup, where a sender won't ever request MIDI data, yet the raspberry_ninja code does allow the sender to both send and receive MIDI data.

midi demo video: https://youtu.be/Gry9UFtOTmQ
Based on the provided information, here is a section for the README that details the `record.py` microservice, focusing on its audio recording and Whisper transcription capabilities, as well as process monitoring.

---

## `record.py`

The `record.py` microservice is designed to handle audio recording and transcription tasks using Whisper AI. It provides a streamlined way to capture audio streams, transcribe them, and manage recording processes efficiently.

### Features

- **Automatic Transcription**: Transcribe audio recordings using Whisper AI.
- **Process Monitoring**: Automatically monitor and manage recording processes to ensure they do not exceed a specified duration.

### Prerequisites

Ensure you have the following installed on your system:

- Python 3.x
- Whisper AI model
- Required Python packages (install via `pip`)

### Installation

A. **Install Dependencies**:

```bash
sudo apt-get update
sudo apt-get install -y python3-pip
pip3 install whisper
```

### Usage

#### Starting the FastAPI Server

To start the FastAPI server, use the following command:

```bash
python3 record.py --host 0.0.0.0 --port 8000
```

This will start the server on the specified host and port.

#### Starting an Audio Recording

To start an audio recording via the FastAPI endpoint, use the following `curl` command:

```bash
curl -X POST -F "room=myRoom" -F "record=myRecord" http://localhost:8000/rec
```

Replace `myRoom` and `myRecord` with your desired room name and record ID.

#### Stopping an Audio Recording

To stop an audio recording and trigger transcription, use the following `curl` command:

```bash
curl -X POST -F "record=myRecord" -F "process_pid=<PROCESS_PID>" -F "language=en" http://localhost:8000/stop
```

Replace `myRecord` with your record ID and `<PROCESS_PID>` with the process ID of the recording process.

#### Command-Line Usage

You can also start and stop recordings directly from the command line.

**Start Recording:**

```bash
python3 record.py --room myRoom --record myRecord
```

Join room [https://vdo.ninja/?push=myRecord&room=myRoom&password=false&effects&record](https://vdo.ninja/?push=myRecord&room=myRoom&password=false&effects&record)

**Stop Recording:**

```bash
python3 record.py --stop --pid <PROCESS_PID> --record myRecord --language en
```

### Process Monitoring

The `record.py` script includes a process monitoring feature that automatically kills any recording process that exceeds one hour of execution time. This ensures that long-running processes do not consume system resources indefinitely.

### Transcription

After stopping the recording, the audio file is automatically transcribed using Whisper AI. The transcription is saved to a text file in the `stt/` directory.

### Example

To start a recording session with specific parameters:

```bash
python3 record.py --room myRoom --record myRecord
```

To stop the recording and transcribe the audio:

```bash
python3 record.py --stop --pid <PROCESS_PID> --record myRecord --language en
```

### Auto-Starting on Boot

To configure the `record.py` script to start automatically on boot, follow the instructions in the Raspberry Pi folder for setting up auto-start scripts.


### Note:

Expand Down
65 changes: 34 additions & 31 deletions publish.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@
from multiprocessing import shared_memory
except Exception as e:
pass

try:
from cryptography.hazmat.primitives.ciphers import Cipher, algorithms, modes
from cryptography.hazmat.primitives.kdf.pbkdf2 import PBKDF2HMAC
Expand Down Expand Up @@ -567,6 +567,10 @@ def on_data_channel_open(channel):

def on_data_channel_close(channel):
printc('DATA CHANNEL: CLOSE', "F44")
if self.save_file:
# Stop the pipeline and set it to NULL state
self.pipe.set_state(Gst.State.NULL)
print("RECORDING TO DISK STOPPED")

def on_data_channel_message(channel, msg_raw):
try:
Expand Down Expand Up @@ -817,51 +821,51 @@ def on_incoming_stream( _, pad):
name = caps.to_string()

print(name)

if "video" in name:
if self.novideo:
printc('Ignoring incoming video track', "F88")
out = Gst.parse_bin_from_description("queue ! fakesink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)
return;

if self.ndiout:
print("NDI OUT")
if "VP8" in name:
out = Gst.parse_bin_from_description("queue ! rtpvp8depay ! decodebin ! videoconvert ! queue ! video/x-raw,format=UYVY ! ndisinkcombiner name=mux1 ! ndisink ndi-name='" + self.streamin + "'", True)
elif "H264" in name:
out = Gst.parse_bin_from_description("queue ! rtph264depay ! h264parse ! queue max-size-buffers=0 max-size-time=0 ! decodebin ! queue max-size-buffers=0 max-size-time=0 ! videoconvert ! queue max-size-buffers=0 max-size-time=0 ! video/x-raw,format=UYVY ! ndisinkcombiner name=mux1 ! queue ! ndisink ndi-name='" + self.streamin + "'", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)
elif self.view:
print("DISPLAY OUTPUT MODE BEING SETUP")

outsink = "autovideosink"
if check_drm_displays():
printc('\nThere is at least one connected display.',"00F")
else:
printc('\n ! No connected displays found. Will try to use glimagesink instead of autovideosink',"F60")
outsink = "glimagesink sync=true"

if "VP8" in name:
out = Gst.parse_bin_from_description(
"queue ! rtpvp8depay ! decodebin ! queue max-size-buffers=0 max-size-time=0 ! videoconvert ! video/x-raw,format=RGB ! queue max-size-buffers=0 max-size-time=0 ! "+outsink, True)
elif "H264" in name:
out = Gst.parse_bin_from_description(
"queue ! rtph264depay ! h264parse ! openh264dec ! queue max-size-buffers=0 max-size-time=0 ! videoconvert ! video/x-raw,format=RGB ! queue max-size-buffers=0 max-size-time=0 ! "+outsink, True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)
pad.link(sink)

elif self.fdsink:
print("FD SINK OUT")
if "VP8" in name:
Expand All @@ -870,24 +874,24 @@ def on_incoming_stream( _, pad):
elif "H264" in name:
out = Gst.parse_bin_from_description(
"queue ! rtph264depay ! h264parse ! openh264dec ! videoconvert ! video/x-raw,format=BGR ! queue max-size-buffers=0 max-size-time=0 ! fdsink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)

elif self.framebuffer: ## send raw data to ffmpeg or something I guess, using the stdout?
print("APP SINK OUT")
if "VP8" in name:
out = Gst.parse_bin_from_description("queue ! rtpvp8depay ! queue max-size-buffers=0 max-size-time=0 ! decodebin ! videoconvert ! video/x-raw,format=BGR ! queue max-size-buffers=2 leaky=downstream ! appsink name=appsink", True)
elif "H264" in name:
out = Gst.parse_bin_from_description("queue ! rtph264depay ! h264parse ! queue max-size-buffers=0 max-size-time=0 ! openh264dec ! videoconvert ! video/x-raw,format=BGR ! queue max-size-buffers=2 leaky=downstream ! appsink name=appsink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)

else:
printc('VIDEO record setup', "88F")
if self.pipe.get_by_name('filesink'):
Expand All @@ -896,7 +900,7 @@ def on_incoming_stream( _, pad):
out = Gst.parse_bin_from_description("queue ! rtpvp8depay", True)
elif "H264" in name:
out = Gst.parse_bin_from_description("queue ! rtph264depay ! h264parse", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
Expand All @@ -916,7 +920,6 @@ def on_incoming_stream( _, pad):
+ "target-duration=10 playlist-length=0 "
+ "location=" + "./" + self.streamin + "_"+str(int(time.time()))+"_segment_%05d.ts "
+ "playlist-location=" + "./" + self.streamin + "_"+str(int(time.time()))+".video.m3u8 " , True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
Expand All @@ -937,19 +940,19 @@ def on_incoming_stream( _, pad):
elif "audio" in name:
if self.noaudio:
printc('Ignoring incoming audio track', "F88")

out = Gst.parse_bin_from_description("queue ! fakesink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)
return;

if self.ndiout:
# if "OPUS" in name:
out = Gst.parse_bin_from_description("queue ! rtpopusdepay ! opusparse ! opusdec ! audioconvert ! audioresample ! audio/x-raw,format=S16LE,rate=48000 ! ndisink name=ndi-audio ndi-name='" + self.streamin + "'", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
Expand All @@ -958,41 +961,41 @@ def on_incoming_stream( _, pad):
# if "OPUS" in name:
print("decode and play out the incoming audio")
out = Gst.parse_bin_from_description("queue ! rtpopusdepay ! opusparse ! opusdec ! audioconvert ! audioresample ! audio/x-raw,format=S16LE,rate=48000 ! autoaudiosink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)
elif self.fdsink:
#if "OPUS" in name:
out = Gst.parse_bin_from_description("queue ! rtpopusdepay ! opusparse ! opusdec ! audioconvert ! audioresample ! audio/x-raw,format=S16LE,rate=48000 ! fdsink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)

elif self.framebuffer:
out = Gst.parse_bin_from_description("queue ! fakesink", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)

else:

if self.pipe.get_by_name('filesink'):
print("Audio being added after video")
if "OPUS" in name:
out = Gst.parse_bin_from_description("queue rtpopusdepay ! opusparse ! audio/x-opus,channel-mapping-family=0,channels=2,rate=48000", True)

self.pipe.add(out)
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
out.link(self.pipe.get_by_name('filesink'))
pad.link(sink)

else:
print("audio being saved...")
if "OPUS" in name:
Expand All @@ -1002,7 +1005,7 @@ def on_incoming_stream( _, pad):
out.sync_state_with_parent()
sink = out.get_static_pad('sink')
pad.link(sink)

print("success audio?")

except Exception as E:
Expand Down Expand Up @@ -1552,7 +1555,7 @@ async def main():
parser.add_argument('--audio-pipeline', type=str, default=None, help='Custom GStreamer audio source pipeline')
parser.add_argument('--timestamp', action='store_true', help='Add a timestamp to the video output, if possible')
parser.add_argument('--clockstamp', action='store_true', help='Add a clock overlay to the video output, if possible')

args = parser.parse_args()

Gst.init(None)
Expand Down Expand Up @@ -1607,7 +1610,7 @@ async def main():
args.streamin = args.record
elif args.view:
args.streamin = args.view

elif args.fdsink:
args.streamin = args.fdsink
elif args.framebuffer:
Expand Down
Loading