summaryrefslogtreecommitdiffstats
path: root/modules/log/color_console_other.go
blob: c08b38c674b92ca790296f2013c81af8eb498f12 (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
// Copyright 2022 The Gitea Authors. All rights reserved.
// SPDX-License-Identifier: MIT

//go:build !windows

package log

import (
	"os"
	"strconv"
	"strings"
	"syscall"

	"github.com/mattn/go-isatty"
)

func journaldDevIno() (uint64, uint64, bool) {
	journaldStream := os.Getenv("JOURNAL_STREAM")
	if len(journaldStream) == 0 {
		return 0, 0, false
	}
	deviceStr, inodeStr, ok := strings.Cut(journaldStream, ":")
	device, err1 := strconv.ParseUint(deviceStr, 10, 64)
	inode, err2 := strconv.ParseUint(inodeStr, 10, 64)
	if !ok || err1 != nil || err2 != nil {
		return 0, 0, false
	}
	return device, inode, true
}

func fileStatDevIno(file *os.File) (uint64, uint64, bool) {
	info, err := file.Stat()
	if err != nil {
		return 0, 0, false
	}

	stat, ok := info.Sys().(*syscall.Stat_t)
	if !ok {
		return 0, 0, false
	}

	// Do a type conversion to uint64, because Dev isn't always uint64
	// on every operating system + architecture combination.
	return uint64(stat.Dev), stat.Ino, true //nolint:unconvert
}

func fileIsDevIno(file *os.File, dev, ino uint64) bool {
	fileDev, fileIno, ok := fileStatDevIno(file)
	return ok && dev == fileDev && ino == fileIno
}

func init() {
	// When forgejo is running under service supervisor (e.g. systemd) with logging
	// set to console, the output streams are typically captured into some logging
	// system (e.g. journald or syslog) instead of going to the terminal. Disable
	// usage of ANSI escape sequences if that's the case to avoid spamming
	// the journal or syslog with garbled mess e.g. `#033[0m#033[32mcmd/web.go:102:#033[32m`.
	CanColorStdout = isatty.IsTerminal(os.Stdout.Fd())
	CanColorStderr = isatty.IsTerminal(os.Stderr.Fd())

	// Furthermore, check if we are running under journald specifically so that
	// further output adjustments can be applied. Specifically, this changes
	// the console logger defaults to disable duplication of date/time info and
	// enable emission of special control sequences understood by journald
	// instead of ANSI colors.
	journalDev, journalIno, ok := journaldDevIno()
	JournaldOnStdout = ok && !CanColorStdout && fileIsDevIno(os.Stdout, journalDev, journalIno)
	JournaldOnStderr = ok && !CanColorStderr && fileIsDevIno(os.Stderr, journalDev, journalIno)
}