summaryrefslogtreecommitdiff
path: root/rhimport/normalizer.go
blob: 9687cedca3b6f0775a34de8ffb38bd021f29078e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
//
//  rhimportd
//
//  The Radio Helsinki Rivendell Import Daemon
//
//
//  Copyright (C) 2015-2016 Christian Pointner <equinox@helsinki.at>
//
//  This file is part of rhimportd.
//
//  rhimportd is free software: you can redistribute it and/or modify
//  it under the terms of the GNU General Public License as published by
//  the Free Software Foundation, either version 3 of the License, or
//  any later version.
//
//  rhimportd 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 General Public License for more details.
//
//  You should have received a copy of the GNU General Public License
//  along with rhimportd. If not, see <http://www.gnu.org/licenses/>.
//

package rhimport

import (
	"fmt"
	"io"
	"net/http"
	"os"
	"os/exec"
	"path/filepath"
	"strings"
)

type ffmpegResult struct {
	output string
	err    error
}

func runNormalizer(ctx *Context, res *Result, src *os.File, size int64) (err error) {
	ctx.reportProgress(2, "normalizing", 0.0, float64(size))

	basepath, filename := filepath.Split(src.Name())
	ext := filepath.Ext(filename)
	destName := strings.TrimSuffix(filename, ext) + "_normalized.flac"
	ctx.stdlog.Printf("NormalizeFile: '%s' -> '%s', using gain = %.2f dB", filename, destName, ctx.LoudnessCorr)

	ctx.SourceFile = basepath + destName

	ffmpeg := exec.Command("ffmpeg", "-loglevel", "warning", "-i", "-", "-filter:a", fmt.Sprintf("volume=%fdB", ctx.LoudnessCorr), "-map_metadata", "0", "-f", "flac", ctx.SourceFile)
	var ffstdin io.WriteCloser
	if ffstdin, err = ffmpeg.StdinPipe(); err != nil {
		return
	}

	ffresult := make(chan ffmpegResult)
	go func() {
		output, err := ffmpeg.CombinedOutput()
		ffresult <- ffmpegResult{strings.TrimSpace(string(output)), err}
	}()

	var written int64
	for {
		if ctx.isCanceled() {
			ffmpeg.Process.Kill()
			res.ResponseCode = http.StatusNoContent
			res.ErrorString = "canceled"
			return nil
		}

		var w int64
		w, err = io.CopyN(ffstdin, src, 512*1024)
		if err != nil {
			break
		}
		written += w
		ctx.reportProgress(2, "normalizing", float64(written), float64(size))
	}
	ffstdin.Close()
	r := <-ffresult
	if r.err != nil {
		return fmt.Errorf("normalizer exited with error: %q, ffmpeg output: %s", r.err, r.output)
	}
	if err != nil && err != io.EOF {
		return
	}
	ctx.reportProgress(2, "normalizing", float64(size), float64(size))
	return nil
}

func NormalizeFile(ctx *Context) (res *Result, err error) {
	res = &Result{ResponseCode: http.StatusOK}

	if ctx.LoudnessCorr == 0.0 {
		ctx.stdlog.Println("NormalizeFile: skipping normalization since the gain = 0.0dB")
		ctx.reportProgress(2, "normalizing", 1.0, 1.0)
		return
	}

	var src *os.File
	if src, err = os.Open(ctx.SourceFile); err != nil {
		return
	}
	defer src.Close()

	size := int64(0)
	if info, err := src.Stat(); err != nil {
		return res, err
	} else {
		size = info.Size()
	}

	if err = runNormalizer(ctx, res, src, size); err != nil {
		ctx.stdlog.Println("NormalizeFile error:", err)
		return
	}
	return
}