aboutsummaryrefslogtreecommitdiffstats
path: root/syz-cluster/pkg/db/session_test_repo.go
blob: 03316c38cc5cdb26dd9db84b490a6742daf5ca2b (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
121
122
// Copyright 2024 syzkaller project authors. All rights reserved.
// Use of this source code is governed by Apache 2 LICENSE that can be found in the LICENSE file.

package db

import (
	"context"

	"cloud.google.com/go/spanner"
)

type SessionTestRepository struct {
	client *spanner.Client
}

func NewSessionTestRepository(client *spanner.Client) *SessionTestRepository {
	return &SessionTestRepository{
		client: client,
	}
}

// If the beforeSave callback is specified, it will be called before saving the entity.
func (repo *SessionTestRepository) InsertOrUpdate(ctx context.Context, test *SessionTest,
	beforeSave func(*SessionTest)) error {
	_, err := repo.client.ReadWriteTransaction(ctx,
		func(ctx context.Context, txn *spanner.ReadWriteTransaction) error {
			// Check if the test already exists.
			dbTest, err := readEntity[SessionTest](ctx, txn, spanner.Statement{
				SQL: "SELECT * from `SessionTests` WHERE `SessionID`=@sessionID AND `TestName` = @testName",
				Params: map[string]interface{}{
					"sessionID": test.SessionID,
					"testName":  test.TestName,
				},
			})
			var stmts []*spanner.Mutation
			if err != nil {
				return err
			} else if dbTest != nil {
				if beforeSave != nil {
					beforeSave(test)
				}
				m, err := spanner.UpdateStruct("SessionTests", test)
				if err != nil {
					return err
				}
				stmts = append(stmts, m)
			} else {
				if beforeSave != nil {
					beforeSave(test)
				}
				m, err := spanner.InsertStruct("SessionTests", test)
				if err != nil {
					return err
				}
				stmts = append(stmts, m)
			}
			return txn.BufferWrite(stmts)
		})
	return err
}

func (repo *SessionTestRepository) Get(ctx context.Context, sessionID, testName string) (*SessionTest, error) {
	return readEntity[SessionTest](ctx, repo.client.Single(), spanner.Statement{
		SQL: "SELECT * FROM `SessionTests` WHERE `SessionID` = @session AND `TestName` = @name",
		Params: map[string]interface{}{
			"session": sessionID,
			"name":    testName,
		},
	})
}

type FullSessionTest struct {
	*SessionTest
	BaseBuild    *Build
	PatchedBuild *Build
}

func (repo *SessionTestRepository) BySession(ctx context.Context, sessionID string) ([]*FullSessionTest, error) {
	list, err := repo.BySessionRaw(ctx, sessionID)
	if err != nil {
		return nil, err
	}
	var ret []*FullSessionTest
	needBuilds := map[string][]**Build{}
	for _, obj := range list {
		full := &FullSessionTest{SessionTest: obj}
		ret = append(ret, full)
		if id := obj.BaseBuildID.String(); !obj.BaseBuildID.IsNull() {
			needBuilds[id] = append(needBuilds[id], &full.BaseBuild)
		}
		if id := obj.PatchedBuildID.String(); !obj.PatchedBuildID.IsNull() {
			needBuilds[id] = append(needBuilds[id], &full.PatchedBuild)
		}
	}
	if len(needBuilds) > 0 {
		var keys []string
		for key := range needBuilds {
			keys = append(keys, key)
		}
		builds, err := readEntities[Build](ctx, repo.client.Single(), spanner.Statement{
			SQL:    "SELECT * FROM `Builds` WHERE `ID` IN UNNEST(@ids)",
			Params: map[string]interface{}{"ids": keys},
		})
		if err != nil {
			return nil, err
		}
		for _, build := range builds {
			for _, patch := range needBuilds[build.ID] {
				*patch = build
			}
		}
	}
	return ret, nil
}

func (repo *SessionTestRepository) BySessionRaw(ctx context.Context, sessionID string) ([]*SessionTest, error) {
	return readEntities[SessionTest](ctx, repo.client.Single(), spanner.Statement{
		SQL: "SELECT * FROM `SessionTests` WHERE `SessionID` = @session" +
			" ORDER BY `UpdatedAt`",
		Params: map[string]interface{}{"session": sessionID},
	})
}