rodstream

package module
v0.0.1 Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Jan 9, 2024 License: MIT Imports: 16 Imported by: 0

README

Overview

An extension of go-rod to retrieve audio or video streams of a page

Installation

$ go get github.com/niklas-may/rod-stream

Usage

The first thing to do is to make sure that we have the lancher ready

package main

import (
	"github.com/go-rod/rod"
	"github.com/go-rod/stealth"

	rodstream "github.com/niklas-may/rod-stream"
)

func main() {
	l := rodstream.
		MustPrepareLauncher().
        // .. other options
		Set("no-sandbox").
		Devtools(false).
		XVFB().
		MustLaunch()

	browser := rod.New().ControlURL(l).
		NoDefaultDevice().
		MustConnect()
}

optional if you don't want to do it yourself, override permissions for URLs

// Example URLS to grant permissions to
urls := []string{"https://meet.google.com", "https://zoom.us"}
if err := rodstream.GrantPermissions(urls, browser); err != nil {
	log.Panicln(err)
}

Create a rod Page


// ... browser
// if using stealth, important to escape bot detection
page := stealth.MustPage(browser)

// Other Proto calls
proto.PageSetAdBlockingEnabled{
	Enabled: true,
}.Call(page)

page.MustNavigate(url)

Getting a stream

go func() {
	// ⚠ Note: the page returned from `MustStreamPage` is not navigatable
	// so don't replace MustPage() with this
	extensionTarget := rodstream.MustCreatePage(browser)
	constraints := &rodstream.StreamConstraints{
		Audio:              true,
		Video:              true,
		MimeType:           "video/webm;codecs=vp9,opus",
		AudioBitsPerSecond: 128000,
		VideoBitsPerSecond: 2500000,
		BitsPerSecond:      8000000, // 1080p https://support.google.com/youtube/answer/1722171?hl=en#zippy=%2Cbitrate
		FrameSize:          1000,    // option passed to mediaRecorder.start(frameSize)
		MinWidth:           1080,
		MinHeight:          1920,
		MaxWidth:           1080,
		MaxHeight:          1920,	
	}
	// Example: Saving to filesystem
	videoFile, err := os.Create("./videos/video.webm")
	if err != nil {
		log.Panicln(err)
	}
	// channel to receive the stream
	ch := make(chan string)
	if err := rodstream.MustGetStream(extensionTarget, constraints, ch); err != nil {
		log.Panicln(err)
	}

	for b64Str := range ch {
		//[important] remove base64 prefix
		buff := rodstream.Parseb64(b64Str)
		// write to File video
		videoFile.Write(buff)
	}
}()

Piping to FFMPEG


// Example: Piping the stream to ffmpeg
stdin, err := rodstream.GetStdInWriter("./videos/video.mp4")
if err != nil {
    log.Panicln(err)
}

defer func() {
    err := stdin.Close()
    log.Panicln(err)
}()

for b64Str := range ch {
    buff := rodstream.Parseb64(b64Str)
    _, err = stdin.Write(buff)
    log.Panicln(err)
}

Closing a stream

go func() {
	if err := rodstream.MustStopStream(extensionTarget); err != nil {
		log.Panicln(err)
	}
}()

Running headfull in Docker

This plugin requires that you run in a headfull mode, so when using docker start rod with or use XVFB() while creating a launcher

RUN apt update && apt install -y xvfb
CMD xvfb-run --server-args="-screen 0 1024x768x24" ./cmd-name

TODOS

  • Add test cases
  • Auto calculate stream bitrate constraints based on window dimensions

Documentation

Index

Constants

This section is empty.

Variables

View Source
var (
	ExtensionId = "jjndjgheafjngoipoacpjgeicjeomjli"
)

Functions

func GetStdInWriter

func GetStdInWriter(filename string) (io.WriteCloser, error)

GetStdInWriter returns a writer that writes to stdin

func GrantPermissions

func GrantPermissions(urls []string, browser *rod.Browser) error

GrantPermissions grants Video & Audio permissions to the urls

func MustGetStream

func MustGetStream(page *PageInfo, streamConstraints StreamConstraints, ch chan string) error

MustGetStream Gets a stream from the browser's page

func MustPrepareLauncher

func MustPrepareLauncher(args LauncherArgs) *launcher.Launcher

MustPrepareLauncher loads the extension and sets required parameters

func MustStopStream

func MustStopStream(page *PageInfo) error

MustStopStream Stops the Stream

func Parseb64

func Parseb64(data string) []byte

Parseb64 removes b64 prefix and returns decoded data

Types

type LauncherArgs

type LauncherArgs struct {
	UserMode bool
}

type PageInfo

type PageInfo struct {
	CapturePage *rod.Page
	StopStream  bool

	Chan chan string // recording channel
}

func MustCreatePage

func MustCreatePage(browser *rod.Browser) *PageInfo

MustCreatePage Must call the browser to capture the extension first handshake returns a page that can be used to capture video

type StreamConstraints

type StreamConstraints struct {
	Audio              bool   `json:"audio"`
	Video              bool   `json:"video"`
	MimeType           string `json:"mimeType,omitempty"`
	AudioBitsPerSecond int    `json:"audioBitsPerSecond,omitempty"`
	VideoBitsPerSecond int    `json:"videoBitsPerSecond,omitempty"`
	BitsPerSecond      int    `json:"bitsPerSecond,omitempty"`
	FrameSize          int    `json:"frameSize,omitempty"`
	MinWidth           int    `json:"minWidth,omitempty"`
	MinHeight          int    `json:"minHeight,omitempty"`
	MaxWidth           int    `json:"maxWidth,omitempty"`
	MaxHeight          int    `json:"maxHeight,omitempty"`
}

Directories

Path Synopsis

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL