File: manager.go

package info (click to toggle)
gitlab-shell 14.35.0%2Bds1-2
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 23,652 kB
  • sloc: ruby: 1,129; makefile: 583; sql: 391; sh: 384
file content (113 lines) | stat: -rw-r--r-- 3,751 bytes parent folder | download | duplicates (4)
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
package trace2

import (
	"context"
	"fmt"
	"os"
)

// Manager is responsible for enabling Trace2 for a Git command. It manages the list of hooks who
// are interested in trace2 data. Before the command starts, the manager opens a tempfile. It
// injects the path to this file and some other conventional environment variables to the ENV list
// of the command by calling Inject. After the command exits, the caller is expected to call Finish.
// Finally, the transformed trace2 tree is passed into handlers of registered hooks.
type Manager struct {
	sid   string
	hooks []Hook
	fd    *os.File
	err   error
}

// NewManager returns a Manager object that manages the registered hooks
func NewManager(sid string, hooks []Hook) (*Manager, error) {
	if len(hooks) == 0 {
		return nil, fmt.Errorf("input hooks are empty")
	}
	return &Manager{sid: sid, hooks: hooks}, nil
}

// HookNames return names of hooks
func (m *Manager) HookNames() []string {
	var names []string
	for _, hook := range m.hooks {
		names = append(names, hook.Name())
	}
	return names
}

// Error returns the error occurs after the manager finishes
func (m *Manager) Error() error {
	return m.err
}

// Inject injects the path to the tempfile used to store trace2 events and conventional environment
// variables to the input ENV list.
func (m *Manager) Inject(env []string) []string {
	fd, err := os.CreateTemp("", "gitaly-trace2")
	if err != nil {
		m.err = fmt.Errorf("trace2 create tempfile: %w", err)
		return env
	}
	m.fd = fd

	env = append(
		env,
		// GIT_TRACE2_EVENT is the key ENV variable. It enables git to dump event format
		// target as JSON-based format. When the path to the file is supplied, it *appends*
		// the events to the appointed file. Child processes inherits the same ENV set.
		// Thus, their events are dumped to the same file. This file is cleaned up when
		// calling Finish().
		fmt.Sprintf("GIT_TRACE2_EVENT=%s", fd.Name()),
		// GIT_TRACE2_PARENT_SID is the unique identifier of a process. As PID number is
		// re-used, Git uses SID number to identify the owner of an event
		fmt.Sprintf("GIT_TRACE2_PARENT_SID=%s", m.sid),
		// GIT_TRACE2_BRIEF strips redundant information, such as time, file, line, etc.
		// This variable makes the output data compact enough to use on production. One
		// notable stripped field is time. The time information is available in some key
		// events. Subsequent events must infer their time from relative float time diff.
		"GIT_TRACE2_BRIEF=true",
		// Apart from the above variables, there are some non-documented interesting
		// variables, such as GIT_TRACE2_CONFIG_PARAMS or GIT_TRACE2_ENV_VARS. We can
		// consider adding them in the future. The full list can be found at:
		// https://github.com/git/git/blob/master/trace2.h
	)
	return env
}

// Finish reads the events, parses them to a tree, triggers hook handlers, and clean up the fd.
func (m *Manager) Finish(ctx context.Context) {
	if m.Error() != nil {
		return
	}

	defer func() {
		if err := m.fd.Close(); err != nil {
			if m.err == nil {
				m.err = fmt.Errorf("trace2: close tempfile: %w", err)
			}
			// Even if the manager fails to close the tempfile, it should fallthrough to
			// remove it from the file system
		}
		if err := os.Remove(m.fd.Name()); err != nil {
			if m.err == nil {
				m.err = fmt.Errorf("trace2: remove tempfile: %w", err)
			}
		}
	}()

	trace, err := Parse(ctx, m.fd)
	if err != nil {
		m.err = fmt.Errorf("trace2: parsing events: %w", err)
		return
	}
	if trace == nil {
		m.err = fmt.Errorf("trace2: no events to handle")
		return
	}
	for _, hook := range m.hooks {
		if err := hook.Handle(ctx, trace); err != nil {
			m.err = fmt.Errorf("trace2: executing %q handler: %w", hook.Name(), err)
			return
		}
	}
}