Skip to content

Commit cdeb293

Browse files
committed
chore: add templates search query to a filter
1 parent 9ee53e5 commit cdeb293

File tree

4 files changed

+198
-2
lines changed

4 files changed

+198
-2
lines changed

coderd/httpapi/queryparams.go

Lines changed: 48 additions & 2 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package httpapi
22

33
import (
4+
"database/sql"
45
"errors"
56
"fmt"
67
"net/url"
@@ -104,6 +105,27 @@ func (p *QueryParamParser) PositiveInt32(vals url.Values, def int32, queryParam
104105
return v
105106
}
106107

108+
// NullableBoolean will return a null sql value if no input is provided.
109+
// SQLc still uses sql.NullBool rather than the generic type. So converting from
110+
// the generic type is required.
111+
func (p *QueryParamParser) NullableBoolean(vals url.Values, def sql.NullBool, queryParam string) sql.NullBool {
112+
v, err := parseNullableQueryParam[bool](p, vals, strconv.ParseBool, sql.Null[bool]{
113+
V: def.Bool,
114+
Valid: def.Valid,
115+
}, queryParam)
116+
if err != nil {
117+
p.Errors = append(p.Errors, codersdk.ValidationError{
118+
Field: queryParam,
119+
Detail: fmt.Sprintf("Query param %q must be a valid boolean: %s", queryParam, err.Error()),
120+
})
121+
}
122+
123+
return sql.NullBool{
124+
Bool: v.V,
125+
Valid: v.Valid,
126+
}
127+
}
128+
107129
func (p *QueryParamParser) Boolean(vals url.Values, def bool, queryParam string) bool {
108130
v, err := parseQueryParam(p, vals, strconv.ParseBool, def, queryParam)
109131
if err != nil {
@@ -294,9 +316,34 @@ func ParseCustomList[T any](parser *QueryParamParser, vals url.Values, def []T,
294316
return v
295317
}
296318

319+
func parseNullableQueryParam[T any](parser *QueryParamParser, vals url.Values, parse func(v string) (T, error), def sql.Null[T], queryParam string) (sql.Null[T], error) {
320+
setParse := parseSingle(parser, parse, def.V, queryParam)
321+
return parseQueryParamSet[sql.Null[T]](parser, vals, func(set []string) (sql.Null[T], error) {
322+
if len(set) == 0 {
323+
return sql.Null[T]{
324+
Valid: false,
325+
}, nil
326+
}
327+
328+
value, err := setParse(set)
329+
if err != nil {
330+
return sql.Null[T]{}, err
331+
}
332+
return sql.Null[T]{
333+
V: value,
334+
Valid: true,
335+
}, nil
336+
}, def, queryParam)
337+
}
338+
297339
// parseQueryParam expects just 1 value set for the given query param.
298340
func parseQueryParam[T any](parser *QueryParamParser, vals url.Values, parse func(v string) (T, error), def T, queryParam string) (T, error) {
299-
setParse := func(set []string) (T, error) {
341+
setParse := parseSingle(parser, parse, def, queryParam)
342+
return parseQueryParamSet(parser, vals, setParse, def, queryParam)
343+
}
344+
345+
func parseSingle[T any](parser *QueryParamParser, parse func(v string) (T, error), def T, queryParam string) func(set []string) (T, error) {
346+
return func(set []string) (T, error) {
300347
if len(set) > 1 {
301348
// Set as a parser.Error rather than return an error.
302349
// Returned errors are errors from the passed in `parse` function, and
@@ -311,7 +358,6 @@ func parseQueryParam[T any](parser *QueryParamParser, vals url.Values, parse fun
311358
}
312359
return parse(set[0])
313360
}
314-
return parseQueryParamSet(parser, vals, setParse, def, queryParam)
315361
}
316362

317363
func parseQueryParamSet[T any](parser *QueryParamParser, vals url.Values, parse func(set []string) (T, error), def T, queryParam string) (T, error) {

coderd/httpapi/queryparams_test.go

Lines changed: 60 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
package httpapi_test
22

33
import (
4+
"database/sql"
45
"fmt"
56
"net/http"
67
"net/url"
@@ -220,6 +221,65 @@ func TestParseQueryParams(t *testing.T) {
220221
testQueryParams(t, expParams, parser, parser.Boolean)
221222
})
222223

224+
t.Run("NullableBoolean", func(t *testing.T) {
225+
t.Parallel()
226+
expParams := []queryParamTestCase[sql.NullBool]{
227+
{
228+
QueryParam: "valid_true",
229+
Value: "true",
230+
Expected: sql.NullBool{
231+
Bool: true,
232+
Valid: true,
233+
},
234+
},
235+
{
236+
QueryParam: "no_value_true_def",
237+
NoSet: true,
238+
Default: sql.NullBool{
239+
Bool: true,
240+
Valid: true,
241+
},
242+
Expected: sql.NullBool{
243+
Bool: true,
244+
Valid: true,
245+
},
246+
},
247+
{
248+
QueryParam: "no_value",
249+
NoSet: true,
250+
Expected: sql.NullBool{
251+
Bool: false,
252+
Valid: false,
253+
},
254+
},
255+
256+
{
257+
QueryParam: "invalid_boolean",
258+
Value: "yes",
259+
Expected: sql.NullBool{
260+
Bool: false,
261+
Valid: false,
262+
},
263+
ExpectedErrorContains: "must be a valid boolean",
264+
},
265+
{
266+
QueryParam: "unexpected_list",
267+
Values: []string{"true", "false"},
268+
ExpectedErrorContains: multipleValuesError,
269+
// Expected value is a bit strange, but the error is raised
270+
// in the parser, not as a parse failure. Maybe this should be
271+
// fixed, but is how it is done atm.
272+
Expected: sql.NullBool{
273+
Bool: false,
274+
Valid: true,
275+
},
276+
},
277+
}
278+
279+
parser := httpapi.NewQueryParamParser()
280+
testQueryParams(t, expParams, parser, parser.NullableBoolean)
281+
})
282+
223283
t.Run("Int", func(t *testing.T) {
224284
t.Parallel()
225285
expParams := []queryParamTestCase[int]{

coderd/searchquery/search.go

Lines changed: 47 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -184,6 +184,53 @@ func Workspaces(query string, page codersdk.Pagination, agentInactiveDisconnectT
184184
return filter, parser.Errors
185185
}
186186

187+
func Templates(ctx context.Context, db database.Store, query string) (database.GetTemplatesWithFilterParams, []codersdk.ValidationError) {
188+
// Always lowercase for all searches.
189+
query = strings.ToLower(query)
190+
values, errors := searchTerms(query, func(term string, values url.Values) error {
191+
// Default to the template name
192+
values.Add("name", term)
193+
return nil
194+
})
195+
if len(errors) > 0 {
196+
return database.GetTemplatesWithFilterParams{}, errors
197+
}
198+
199+
const dateLayout = "2006-01-02"
200+
parser := httpapi.NewQueryParamParser()
201+
filter := database.GetTemplatesWithFilterParams{
202+
Deleted: parser.Boolean(values, false, "deleted"),
203+
// TODO: Should name be a fuzzy search?
204+
ExactName: parser.String(values, "", "name"),
205+
IDs: parser.UUIDs(values, []uuid.UUID{}, "ids"),
206+
Deprecated: parser.NullableBoolean(values, sql.NullBool{}, "deprecated"),
207+
}
208+
209+
// Convert the "organization" parameter to an organization uuid. This can require
210+
// a database lookup.
211+
organizationArg := parser.String(values, "", "organization")
212+
if organizationArg != "" {
213+
organizationID, err := uuid.Parse(organizationArg)
214+
if err == nil {
215+
filter.OrganizationID = organizationID
216+
} else {
217+
// Organization could be a name
218+
organization, err := db.GetOrganizationByName(ctx, organizationArg)
219+
if err != nil {
220+
parser.Errors = append(parser.Errors, codersdk.ValidationError{
221+
Field: "organization",
222+
Detail: fmt.Sprintf("Organization %q either does not exist, or you are unauthorized to view it", organizationArg),
223+
})
224+
} else {
225+
filter.OrganizationID = organization.ID
226+
}
227+
}
228+
}
229+
230+
parser.ErrorExcessParams(values)
231+
return filter, parser.Errors
232+
}
233+
187234
func searchTerms(query string, defaultKey func(term string, values url.Values) error) (url.Values, []codersdk.ValidationError) {
188235
searchValues := make(url.Values)
189236

coderd/searchquery/search_test.go

Lines changed: 43 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ import (
88
"testing"
99
"time"
1010

11+
"github.com/google/uuid"
1112
"github.com/stretchr/testify/assert"
1213
"github.com/stretchr/testify/require"
1314

@@ -454,3 +455,45 @@ func TestSearchUsers(t *testing.T) {
454455
})
455456
}
456457
}
458+
459+
func TestSearchTemplates(t *testing.T) {
460+
t.Parallel()
461+
testCases := []struct {
462+
Name string
463+
Query string
464+
Expected database.GetTemplatesWithFilterParams
465+
ExpectedErrorContains string
466+
}{
467+
{
468+
Name: "Empty",
469+
Query: "",
470+
Expected: database.GetTemplatesWithFilterParams{},
471+
},
472+
}
473+
474+
for _, c := range testCases {
475+
c := c
476+
t.Run(c.Name, func(t *testing.T) {
477+
t.Parallel()
478+
// Do not use a real database, this is only used for an
479+
// organization lookup.
480+
db := dbmem.New()
481+
values, errs := searchquery.Templates(context.Background(), db, c.Query)
482+
if c.ExpectedErrorContains != "" {
483+
require.True(t, len(errs) > 0, "expect some errors")
484+
var s strings.Builder
485+
for _, err := range errs {
486+
_, _ = s.WriteString(fmt.Sprintf("%s: %s\n", err.Field, err.Detail))
487+
}
488+
require.Contains(t, s.String(), c.ExpectedErrorContains)
489+
} else {
490+
require.Len(t, errs, 0, "expected no error")
491+
if c.Expected.IDs == nil {
492+
// Nil and length 0 are the same
493+
c.Expected.IDs = []uuid.UUID{}
494+
}
495+
require.Equal(t, c.Expected, values, "expected values")
496+
}
497+
})
498+
}
499+
}

0 commit comments

Comments
 (0)