mirror of
https://github.com/navidrome/navidrome.git
synced 2025-04-04 21:17:37 +03:00
Complete User CRUD
This commit is contained in:
parent
1c04a19910
commit
2ab0cecd48
11 changed files with 201 additions and 45 deletions
8
Makefile
8
Makefile
|
@ -1,14 +1,14 @@
|
||||||
GO_VERSION=1.13
|
GO_VERSION=1.13
|
||||||
NODE_VERSION=12.14.1
|
NODE_VERSION=12.14.1
|
||||||
|
|
||||||
.PHONY: run
|
|
||||||
run: check_go_env data
|
|
||||||
@reflex -d none -c reflex.conf
|
|
||||||
|
|
||||||
.PHONY: dev
|
.PHONY: dev
|
||||||
dev: check_env data
|
dev: check_env data
|
||||||
@goreman -f Procfile.dev -b 4533 start
|
@goreman -f Procfile.dev -b 4533 start
|
||||||
|
|
||||||
|
.PHONY: server
|
||||||
|
server: check_go_env data
|
||||||
|
@reflex -d none -c reflex.conf
|
||||||
|
|
||||||
.PHONY: watch
|
.PHONY: watch
|
||||||
watch: check_go_env
|
watch: check_go_env
|
||||||
ginkgo watch -notify ./...
|
ginkgo watch -notify ./...
|
||||||
|
|
|
@ -32,7 +32,7 @@ the steps in the [Development Environment](#development-environment) section bel
|
||||||
|
|
||||||
```
|
```
|
||||||
$ export SONIC_MUSICFOLDER="/path/to/your/music/folder"
|
$ export SONIC_MUSICFOLDER="/path/to/your/music/folder"
|
||||||
$ make run
|
$ make
|
||||||
```
|
```
|
||||||
|
|
||||||
The server should start listening for requests. The default configuration is:
|
The server should start listening for requests. The default configuration is:
|
||||||
|
@ -58,7 +58,7 @@ Some useful commands:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Start local server (with hot reload)
|
# Start local server (with hot reload)
|
||||||
$ make run
|
$ make
|
||||||
|
|
||||||
# Run all tests
|
# Run all tests
|
||||||
$ make test
|
$ make test
|
||||||
|
|
4
go.mod
4
go.mod
|
@ -8,14 +8,12 @@ require (
|
||||||
github.com/astaxie/beego v1.12.0
|
github.com/astaxie/beego v1.12.0
|
||||||
github.com/bradleyjkemp/cupaloy v2.3.0+incompatible
|
github.com/bradleyjkemp/cupaloy v2.3.0+incompatible
|
||||||
github.com/deluan/rest v0.0.0-20200114062534-0653ffe9eab4
|
github.com/deluan/rest v0.0.0-20200114062534-0653ffe9eab4
|
||||||
github.com/dhowden/itl v0.0.0-20170329215456-9fbe21093131
|
|
||||||
github.com/dhowden/plist v0.0.0-20141002110153-5db6e0d9931a // indirect
|
|
||||||
github.com/dhowden/tag v0.0.0-20191122115059-7e5c04feccd8
|
github.com/dhowden/tag v0.0.0-20191122115059-7e5c04feccd8
|
||||||
github.com/fatih/camelcase v0.0.0-20160318181535-f6a740d52f96 // indirect
|
github.com/fatih/camelcase v0.0.0-20160318181535-f6a740d52f96 // indirect
|
||||||
github.com/fatih/structs v1.0.0 // indirect
|
github.com/fatih/structs v1.0.0 // indirect
|
||||||
github.com/go-chi/chi v4.0.3+incompatible
|
github.com/go-chi/chi v4.0.3+incompatible
|
||||||
github.com/go-chi/cors v1.0.0
|
github.com/go-chi/cors v1.0.0
|
||||||
github.com/google/uuid v1.1.1 // indirect
|
github.com/google/uuid v1.1.1
|
||||||
github.com/google/wire v0.4.0
|
github.com/google/wire v0.4.0
|
||||||
github.com/kennygrant/sanitize v0.0.0-20170120101633-6a0bfdde8629
|
github.com/kennygrant/sanitize v0.0.0-20170120101633-6a0bfdde8629
|
||||||
github.com/koding/multiconfig v0.0.0-20170327155832-26b6dfd3a84a
|
github.com/koding/multiconfig v0.0.0-20170327155832-26b6dfd3a84a
|
||||||
|
|
4
go.sum
4
go.sum
|
@ -24,10 +24,6 @@ github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/deluan/rest v0.0.0-20200114062534-0653ffe9eab4 h1:VSoAcWJvj656TSyWbJ5KuGsi/J8dO5+iO9+5/7I8wao=
|
github.com/deluan/rest v0.0.0-20200114062534-0653ffe9eab4 h1:VSoAcWJvj656TSyWbJ5KuGsi/J8dO5+iO9+5/7I8wao=
|
||||||
github.com/deluan/rest v0.0.0-20200114062534-0653ffe9eab4/go.mod h1:tSgDythFsl0QgS/PFWfIZqcJKnkADWneY80jaVRlqK8=
|
github.com/deluan/rest v0.0.0-20200114062534-0653ffe9eab4/go.mod h1:tSgDythFsl0QgS/PFWfIZqcJKnkADWneY80jaVRlqK8=
|
||||||
github.com/dhowden/itl v0.0.0-20170329215456-9fbe21093131 h1:siEGb+iB1Ea75U7BnkYVSqSRzE6QHlXCbqEXenxRmhQ=
|
|
||||||
github.com/dhowden/itl v0.0.0-20170329215456-9fbe21093131/go.mod h1:eVWQJVQ67aMvYhpkDwaH2Goy2vo6v8JCMfGXfQ9sPtw=
|
|
||||||
github.com/dhowden/plist v0.0.0-20141002110153-5db6e0d9931a h1:7MucP9rMAsQRcRE1sGpvMZoTxFYZlDmfDvCH+z7H+90=
|
|
||||||
github.com/dhowden/plist v0.0.0-20141002110153-5db6e0d9931a/go.mod h1:sLjdR6uwx3L6/Py8F+QgAfeiuY87xuYGwCDqRFrvCzw=
|
|
||||||
github.com/dhowden/tag v0.0.0-20191122115059-7e5c04feccd8 h1:nmFnmD8VZXkjDHIb1Gnfz50cgzUvGN72zLjPRXBW/hU=
|
github.com/dhowden/tag v0.0.0-20191122115059-7e5c04feccd8 h1:nmFnmD8VZXkjDHIb1Gnfz50cgzUvGN72zLjPRXBW/hU=
|
||||||
github.com/dhowden/tag v0.0.0-20191122115059-7e5c04feccd8/go.mod h1:SniNVYuaD1jmdEEvi+7ywb1QFR7agjeTdGKyFb0p7Rw=
|
github.com/dhowden/tag v0.0.0-20191122115059-7e5c04feccd8/go.mod h1:SniNVYuaD1jmdEEvi+7ywb1QFR7agjeTdGKyFb0p7Rw=
|
||||||
github.com/edsrzf/mmap-go v0.0.0-20170320065105-0bce6a688712 h1:aaQcKT9WumO6JEJcRyTqFVq4XUZiUcKR2/GI31TOcz8=
|
github.com/edsrzf/mmap-go v0.0.0-20170320065105-0bce6a688712 h1:aaQcKT9WumO6JEJcRyTqFVq4XUZiUcKR2/GI31TOcz8=
|
||||||
|
|
|
@ -7,8 +7,8 @@ type User struct {
|
||||||
Name string
|
Name string
|
||||||
Password string
|
Password string
|
||||||
IsAdmin bool
|
IsAdmin bool
|
||||||
LastLoginAt time.Time
|
LastLoginAt *time.Time
|
||||||
LastAccessAt time.Time
|
LastAccessAt *time.Time
|
||||||
CreatedAt time.Time
|
CreatedAt time.Time
|
||||||
UpdatedAt time.Time
|
UpdatedAt time.Time
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,12 +1,15 @@
|
||||||
package persistence
|
package persistence
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"fmt"
|
||||||
"reflect"
|
"reflect"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/astaxie/beego/orm"
|
"github.com/astaxie/beego/orm"
|
||||||
"github.com/cloudsonic/sonic-server/model"
|
"github.com/cloudsonic/sonic-server/model"
|
||||||
"github.com/deluan/rest"
|
"github.com/deluan/rest"
|
||||||
|
"github.com/google/uuid"
|
||||||
)
|
)
|
||||||
|
|
||||||
type resourceRepository struct {
|
type resourceRepository struct {
|
||||||
|
@ -20,19 +23,43 @@ type resourceRepository struct {
|
||||||
|
|
||||||
func NewResource(o orm.Ormer, model interface{}, mappedModel interface{}) model.ResourceRepository {
|
func NewResource(o orm.Ormer, model interface{}, mappedModel interface{}) model.ResourceRepository {
|
||||||
r := &resourceRepository{model: model, mappedModel: mappedModel, ormer: o}
|
r := &resourceRepository{model: model, mappedModel: mappedModel, ormer: o}
|
||||||
r.instanceType = reflect.TypeOf(mappedModel)
|
|
||||||
|
// Get type of mappedModel (which is a *struct)
|
||||||
|
rv := reflect.ValueOf(mappedModel)
|
||||||
|
for rv.Kind() == reflect.Ptr || rv.Kind() == reflect.Interface {
|
||||||
|
rv = rv.Elem()
|
||||||
|
}
|
||||||
|
r.instanceType = rv.Type()
|
||||||
r.sliceType = reflect.SliceOf(r.instanceType)
|
r.sliceType = reflect.SliceOf(r.instanceType)
|
||||||
return r
|
return r
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *resourceRepository) newQuery() orm.QuerySeter {
|
func (r *resourceRepository) EntityName() string {
|
||||||
return r.ormer.QueryTable(r.mappedModel)
|
return r.instanceType.Name()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) newQuery(options ...rest.QueryOptions) orm.QuerySeter {
|
||||||
|
qs := r.ormer.QueryTable(r.mappedModel)
|
||||||
|
if len(options) > 0 {
|
||||||
|
qs = r.addOptions(qs, options)
|
||||||
|
qs = r.addFilters(qs, r.buildFilters(qs, options))
|
||||||
|
}
|
||||||
|
return qs
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) NewInstance() interface{} {
|
||||||
|
return reflect.New(r.instanceType).Interface()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) NewSlice() interface{} {
|
||||||
|
slice := reflect.MakeSlice(r.sliceType, 0, 0)
|
||||||
|
x := reflect.New(slice.Type())
|
||||||
|
x.Elem().Set(slice)
|
||||||
|
return x.Interface()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *resourceRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
|
func (r *resourceRepository) ReadAll(options ...rest.QueryOptions) (interface{}, error) {
|
||||||
qs := r.newQuery()
|
qs := r.newQuery(options...)
|
||||||
qs = r.addOptions(qs, options)
|
|
||||||
//qs = r.addFilters(qs, r.buildFilters(qs, options), r.getRestriction())
|
|
||||||
dataSet := r.NewSlice()
|
dataSet := r.NewSlice()
|
||||||
_, err := qs.All(dataSet)
|
_, err := qs.All(dataSet)
|
||||||
if err == orm.ErrNoRows {
|
if err == orm.ErrNoRows {
|
||||||
|
@ -42,8 +69,7 @@ func (r *resourceRepository) ReadAll(options ...rest.QueryOptions) (interface{},
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *resourceRepository) Count(options ...rest.QueryOptions) (int64, error) {
|
func (r *resourceRepository) Count(options ...rest.QueryOptions) (int64, error) {
|
||||||
qs := r.newQuery()
|
qs := r.newQuery(options...)
|
||||||
//qs = r.addFilters(qs, r.buildFilters(qs, options), r.getRestriction())
|
|
||||||
count, err := qs.Count()
|
count, err := qs.Count()
|
||||||
if err == orm.ErrNoRows {
|
if err == orm.ErrNoRows {
|
||||||
err = rest.ErrNotFound
|
err = rest.ErrNotFound
|
||||||
|
@ -51,11 +77,44 @@ func (r *resourceRepository) Count(options ...rest.QueryOptions) (int64, error)
|
||||||
return count, err
|
return count, err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *resourceRepository) NewSlice() interface{} {
|
func (r *resourceRepository) Read(id string) (interface{}, error) {
|
||||||
slice := reflect.MakeSlice(r.sliceType, 0, 0)
|
qs := r.newQuery().Filter("id", id)
|
||||||
x := reflect.New(slice.Type())
|
data := r.NewInstance()
|
||||||
x.Elem().Set(slice)
|
err := qs.One(data)
|
||||||
return x.Interface()
|
if err == orm.ErrNoRows {
|
||||||
|
return data, rest.ErrNotFound
|
||||||
|
}
|
||||||
|
return data, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func setUUID(p interface{}) {
|
||||||
|
f := reflect.ValueOf(p).Elem().FieldByName("ID")
|
||||||
|
if f.Kind() == reflect.String {
|
||||||
|
id, _ := uuid.NewRandom()
|
||||||
|
f.SetString(id.String())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) Save(p interface{}) (string, error) {
|
||||||
|
setUUID(p)
|
||||||
|
id, err := r.ormer.Insert(p)
|
||||||
|
if err != nil {
|
||||||
|
if err.Error() != "LastInsertId is not supported by this driver" {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return strconv.FormatInt(id, 10), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) Update(p interface{}, cols ...string) error {
|
||||||
|
count, err := r.ormer.Update(p, cols...)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if count == 0 {
|
||||||
|
return rest.ErrNotFound
|
||||||
|
}
|
||||||
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
func (r *resourceRepository) addOptions(qs orm.QuerySeter, options []rest.QueryOptions) orm.QuerySeter {
|
func (r *resourceRepository) addOptions(qs orm.QuerySeter, options []rest.QueryOptions) orm.QuerySeter {
|
||||||
|
@ -87,3 +146,59 @@ func (r *resourceRepository) addOptions(qs orm.QuerySeter, options []rest.QueryO
|
||||||
}
|
}
|
||||||
return qs
|
return qs
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) addFilters(qs orm.QuerySeter, conditions ...*orm.Condition) orm.QuerySeter {
|
||||||
|
var cond *orm.Condition
|
||||||
|
for _, c := range conditions {
|
||||||
|
if c != nil {
|
||||||
|
if cond == nil {
|
||||||
|
cond = c
|
||||||
|
} else {
|
||||||
|
cond = cond.AndCond(c)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if cond != nil {
|
||||||
|
return qs.SetCond(cond)
|
||||||
|
}
|
||||||
|
return qs
|
||||||
|
}
|
||||||
|
|
||||||
|
func unmarshalValue(val interface{}) string {
|
||||||
|
switch v := val.(type) {
|
||||||
|
case float64:
|
||||||
|
return strconv.FormatFloat(v, 'f', -1, 64)
|
||||||
|
case string:
|
||||||
|
return v
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("%v", val)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (r *resourceRepository) buildFilters(qs orm.QuerySeter, options []rest.QueryOptions) *orm.Condition {
|
||||||
|
if len(options) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
cond := orm.NewCondition()
|
||||||
|
clauses := cond
|
||||||
|
for f, v := range options[0].Filters {
|
||||||
|
fn := strings.Replace(f, ".", "__", -1)
|
||||||
|
s := unmarshalValue(v)
|
||||||
|
|
||||||
|
if strings.HasSuffix(fn, "Id") || strings.HasSuffix(fn, "__id") {
|
||||||
|
clauses = IdFilter(clauses, fn, s)
|
||||||
|
} else {
|
||||||
|
clauses = StartsWithFilter(clauses, fn, s)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return clauses
|
||||||
|
}
|
||||||
|
|
||||||
|
func IdFilter(cond *orm.Condition, field, value string) *orm.Condition {
|
||||||
|
field = strings.TrimSuffix(field, "Id") + "__id"
|
||||||
|
return cond.And(field, value)
|
||||||
|
}
|
||||||
|
|
||||||
|
func StartsWithFilter(cond *orm.Condition, field, value string) *orm.Condition {
|
||||||
|
return cond.And(field+"__istartswith", value)
|
||||||
|
}
|
||||||
|
|
|
@ -7,14 +7,14 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
type user struct {
|
type user struct {
|
||||||
ID string `json:"id" orm:"pk;column(id)"`
|
ID string `json:"id" orm:"pk;column(id)"`
|
||||||
Name string `json:"name" orm:"index"`
|
Name string `json:"name" orm:"index"`
|
||||||
Password string `json:"-"`
|
Password string `json:"-"`
|
||||||
IsAdmin bool `json:"isAdmin"`
|
IsAdmin bool `json:"isAdmin"`
|
||||||
LastLoginAt time.Time `json:"lastLoginAt"`
|
LastLoginAt *time.Time `json:"lastLoginAt" orm:"null"`
|
||||||
LastAccessAt time.Time `json:"lastAccessAt"`
|
LastAccessAt *time.Time `json:"lastAccessAt" orm:"null"`
|
||||||
CreatedAt time.Time `json:"createdAt" orm:"auto_now_add;type(datetime)"`
|
CreatedAt time.Time `json:"createdAt" orm:"auto_now_add;type(datetime)"`
|
||||||
UpdatedAt time.Time `json:"updatedAt" orm:"auto_now;type(datetime)"`
|
UpdatedAt time.Time `json:"updatedAt" orm:"auto_now;type(datetime)"`
|
||||||
}
|
}
|
||||||
|
|
||||||
var _ = model.User(user{})
|
var _ = model.User(user{})
|
||||||
|
|
21
ui/src/user/UserCreate.js
Normal file
21
ui/src/user/UserCreate.js
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
import React from 'react'
|
||||||
|
import {
|
||||||
|
BooleanInput,
|
||||||
|
Create,
|
||||||
|
TextInput,
|
||||||
|
PasswordInput,
|
||||||
|
required,
|
||||||
|
SimpleForm
|
||||||
|
} from 'react-admin'
|
||||||
|
|
||||||
|
const UserCreate = (props) => (
|
||||||
|
<Create {...props}>
|
||||||
|
<SimpleForm redirect="list">
|
||||||
|
<TextInput source="name" validate={[required()]} />
|
||||||
|
<PasswordInput source="password" validate={[required()]} />
|
||||||
|
<BooleanInput source="isAdmin" initialValue={false} />
|
||||||
|
</SimpleForm>
|
||||||
|
</Create>
|
||||||
|
)
|
||||||
|
|
||||||
|
export default UserCreate
|
26
ui/src/user/UserEdit.js
Normal file
26
ui/src/user/UserEdit.js
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
import React from 'react'
|
||||||
|
import {
|
||||||
|
TextInput,
|
||||||
|
BooleanInput,
|
||||||
|
DateField,
|
||||||
|
PasswordInput,
|
||||||
|
Edit,
|
||||||
|
required,
|
||||||
|
SimpleForm
|
||||||
|
} from 'react-admin'
|
||||||
|
|
||||||
|
const UserEdit = (props) => (
|
||||||
|
<Edit {...props}>
|
||||||
|
<SimpleForm>
|
||||||
|
<TextInput source="name" validate={[required()]} />
|
||||||
|
<PasswordInput source="password" validate={[required()]} />
|
||||||
|
<BooleanInput source="isAdmin" initialValue={false} />
|
||||||
|
<DateField source="lastLoginAt" />
|
||||||
|
<DateField source="lastAccessAt" />
|
||||||
|
<DateField source="updatedAt" />
|
||||||
|
<DateField source="createdAt" />
|
||||||
|
</SimpleForm>
|
||||||
|
</Edit>
|
||||||
|
)
|
||||||
|
|
||||||
|
export default UserEdit
|
|
@ -2,8 +2,8 @@ import React from 'react'
|
||||||
import {
|
import {
|
||||||
BooleanField,
|
BooleanField,
|
||||||
Datagrid,
|
Datagrid,
|
||||||
DateField,
|
|
||||||
Filter,
|
Filter,
|
||||||
|
DateField,
|
||||||
List,
|
List,
|
||||||
SearchInput,
|
SearchInput,
|
||||||
SimpleList,
|
SimpleList,
|
||||||
|
@ -13,7 +13,7 @@ import { useMediaQuery } from '@material-ui/core'
|
||||||
|
|
||||||
const UserFilter = (props) => (
|
const UserFilter = (props) => (
|
||||||
<Filter {...props}>
|
<Filter {...props}>
|
||||||
<SearchInput source="q" alwaysOn />
|
<SearchInput source="name" alwaysOn />
|
||||||
</Filter>
|
</Filter>
|
||||||
)
|
)
|
||||||
|
|
||||||
|
@ -30,10 +30,10 @@ const UserList = (props) => {
|
||||||
{isXsmall ? (
|
{isXsmall ? (
|
||||||
<SimpleList
|
<SimpleList
|
||||||
primaryText={(record) => record.name}
|
primaryText={(record) => record.name}
|
||||||
secondaryText={(record) => record.email}
|
tertiaryText={(record) => (record.isAdmin ? '[admin]️' : '')}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<Datagrid>
|
<Datagrid rowClick="edit">
|
||||||
<TextField source="name" />
|
<TextField source="name" />
|
||||||
<BooleanField source="isAdmin" />
|
<BooleanField source="isAdmin" />
|
||||||
<DateField source="lastLoginAt" locales="pt-BR" />
|
<DateField source="lastLoginAt" locales="pt-BR" />
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
import SupervisedUserCircleIcon from '@material-ui/icons/SupervisedUserCircle'
|
import SupervisedUserCircleIcon from '@material-ui/icons/Group'
|
||||||
import UserList from './UserList'
|
import UserList from './UserList'
|
||||||
// import UserEdit from './UserEdit'
|
import UserEdit from './UserEdit'
|
||||||
// import UserCreate from './UserCreate'
|
import UserCreate from './UserCreate'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
list: UserList,
|
list: UserList,
|
||||||
// edit: UserEdit,
|
edit: UserEdit,
|
||||||
// create: UserCreate,
|
create: UserCreate,
|
||||||
icon: SupervisedUserCircleIcon
|
icon: SupervisedUserCircleIcon
|
||||||
}
|
}
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue