2024-07-12 09:39:47 +00:00
|
|
|
// GoToSocial
|
|
|
|
// Copyright (C) GoToSocial Authors admin@gotosocial.org
|
|
|
|
// SPDX-License-Identifier: AGPL-3.0-or-later
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU Affero General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
package ffmpeg
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2024-08-28 16:32:09 +00:00
|
|
|
"sync"
|
|
|
|
|
|
|
|
ffprobelib "codeberg.org/gruf/go-ffmpreg/embed/ffprobe"
|
|
|
|
"github.com/superseriousbusiness/gotosocial/internal/log"
|
|
|
|
"github.com/tetratelabs/wazero"
|
|
|
|
"github.com/tetratelabs/wazero/sys"
|
2024-07-12 09:39:47 +00:00
|
|
|
)
|
|
|
|
|
2024-08-28 16:32:09 +00:00
|
|
|
var (
|
|
|
|
// ffprobeRunner limits the number of
|
|
|
|
// ffprobe WebAssembly instances that
|
|
|
|
// may be concurrently running, in
|
|
|
|
// order to reduce memory usage.
|
|
|
|
ffprobeRunner runner
|
|
|
|
|
|
|
|
// ffprobe compiled WASM.
|
|
|
|
ffprobe wazero.CompiledModule
|
|
|
|
|
|
|
|
// Number of times ffprobe
|
|
|
|
// compiled WASM has run.
|
|
|
|
ffprobeRunCount int
|
|
|
|
|
|
|
|
// Sync for updating run count
|
|
|
|
// and recompiling ffprobe.
|
|
|
|
ffprobeM sync.Mutex
|
|
|
|
)
|
2024-08-23 15:15:35 +00:00
|
|
|
|
|
|
|
// InitFfprobe precompiles the ffprobe WebAssembly source into memory and
|
|
|
|
// prepares the runner to only allow max given concurrent running instances.
|
2024-07-12 09:39:47 +00:00
|
|
|
func InitFfprobe(ctx context.Context, max int) error {
|
2024-08-28 16:32:09 +00:00
|
|
|
|
|
|
|
// Ensure runner initialized.
|
2024-08-23 15:15:35 +00:00
|
|
|
ffprobeRunner.Init(max)
|
2024-08-28 16:32:09 +00:00
|
|
|
|
|
|
|
// Ensure runtime initialized.
|
|
|
|
if err := initRuntime(ctx); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ensure ffprobe compiled.
|
|
|
|
if ffprobe == nil {
|
|
|
|
return compileFfprobe(ctx)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// compileFfprobe ensures the ffprobe WebAssembly
|
|
|
|
// module has been pre-compiled into memory.
|
|
|
|
func compileFfprobe(ctx context.Context) error {
|
|
|
|
var err error
|
|
|
|
ffprobe, err = runtime.CompileModule(ctx, ffprobelib.B)
|
|
|
|
return err
|
2024-07-12 09:39:47 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Ffprobe runs the given arguments with an instance of ffprobe.
|
2024-08-23 15:15:35 +00:00
|
|
|
func Ffprobe(ctx context.Context, args Args) (uint32, error) {
|
2024-08-28 16:32:09 +00:00
|
|
|
return ffprobeRunner.Run(ctx, func() (uint32, error) {
|
|
|
|
|
|
|
|
// Update run count + check if we
|
|
|
|
// need to recompile the module.
|
|
|
|
ffprobeM.Lock()
|
|
|
|
{
|
|
|
|
ffprobeRunCount++
|
|
|
|
if ffprobeRunCount > 500 {
|
|
|
|
// Over our threshold of runs, close
|
|
|
|
// current compiled module and recompile.
|
|
|
|
if err := ffprobe.Close(ctx); err != nil {
|
|
|
|
ffprobeM.Unlock()
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
if err := compileFfprobe(ctx); err != nil {
|
|
|
|
ffprobeM.Unlock()
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
|
|
|
|
ffprobeRunCount = 0
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ffprobeM.Unlock()
|
|
|
|
|
|
|
|
// Prefix module name as argv0 to args.
|
|
|
|
cargs := make([]string, len(args.Args)+1)
|
|
|
|
copy(cargs[1:], args.Args)
|
|
|
|
cargs[0] = "ffprobe"
|
|
|
|
|
|
|
|
// Create base module config.
|
|
|
|
modcfg := wazero.NewModuleConfig()
|
|
|
|
modcfg = modcfg.WithArgs(cargs...)
|
|
|
|
modcfg = modcfg.WithStdin(args.Stdin)
|
|
|
|
modcfg = modcfg.WithStdout(args.Stdout)
|
|
|
|
modcfg = modcfg.WithStderr(args.Stderr)
|
|
|
|
|
|
|
|
if args.Config != nil {
|
|
|
|
// Pass through config fn.
|
|
|
|
modcfg = args.Config(modcfg)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Instantiate the module from precompiled wasm module data.
|
|
|
|
mod, err := runtime.InstantiateModule(ctx, ffprobe, modcfg)
|
|
|
|
|
|
|
|
if mod != nil {
|
|
|
|
// Ensure closed.
|
|
|
|
if err := mod.Close(ctx); err != nil {
|
|
|
|
log.Errorf(ctx, "error closing: %v", err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Try extract exit code.
|
|
|
|
switch err := err.(type) {
|
|
|
|
case *sys.ExitError:
|
|
|
|
return err.ExitCode(), nil
|
|
|
|
default:
|
|
|
|
return 0, err
|
|
|
|
}
|
|
|
|
})
|
2024-07-12 09:39:47 +00:00
|
|
|
}
|