mirror of https://github.com/grafana/grafana
Chore: Introduce playlist service (#52252)
* Store: Introduce playlist service * Integrate playlist service * Update swaggerpull/52389/head
parent
332639ce43
commit
fb379ae436
@ -0,0 +1,95 @@ |
|||||||
|
package playlist |
||||||
|
|
||||||
|
import ( |
||||||
|
"errors" |
||||||
|
) |
||||||
|
|
||||||
|
// Typed errors
|
||||||
|
var ( |
||||||
|
ErrPlaylistNotFound = errors.New("Playlist not found") |
||||||
|
ErrPlaylistFailedGenerateUniqueUid = errors.New("failed to generate unique playlist UID") |
||||||
|
ErrCommandValidationFailed = errors.New("command missing required fields") |
||||||
|
) |
||||||
|
|
||||||
|
// Playlist model
|
||||||
|
type Playlist struct { |
||||||
|
Id int64 `json:"id"` |
||||||
|
UID string `json:"uid" xorm:"uid"` |
||||||
|
Name string `json:"name"` |
||||||
|
Interval string `json:"interval"` |
||||||
|
OrgId int64 `json:"-"` |
||||||
|
} |
||||||
|
|
||||||
|
type PlaylistDTO struct { |
||||||
|
Id int64 `json:"id"` |
||||||
|
UID string `json:"uid"` |
||||||
|
Name string `json:"name"` |
||||||
|
Interval string `json:"interval"` |
||||||
|
OrgId int64 `json:"-"` |
||||||
|
Items []PlaylistItemDTO `json:"items"` |
||||||
|
} |
||||||
|
|
||||||
|
type PlaylistItemDTO struct { |
||||||
|
Id int64 `json:"id"` |
||||||
|
PlaylistId int64 `json:"playlistid"` |
||||||
|
Type string `json:"type"` |
||||||
|
Title string `json:"title"` |
||||||
|
Value string `json:"value"` |
||||||
|
Order int `json:"order"` |
||||||
|
} |
||||||
|
|
||||||
|
type PlaylistItem struct { |
||||||
|
Id int64 |
||||||
|
PlaylistId int64 |
||||||
|
Type string |
||||||
|
Value string |
||||||
|
Order int |
||||||
|
Title string |
||||||
|
} |
||||||
|
|
||||||
|
type Playlists []*Playlist |
||||||
|
|
||||||
|
//
|
||||||
|
// COMMANDS
|
||||||
|
//
|
||||||
|
|
||||||
|
type UpdatePlaylistCommand struct { |
||||||
|
OrgId int64 `json:"-"` |
||||||
|
UID string `json:"uid"` |
||||||
|
Name string `json:"name" binding:"Required"` |
||||||
|
Interval string `json:"interval"` |
||||||
|
Items []PlaylistItemDTO `json:"items"` |
||||||
|
} |
||||||
|
|
||||||
|
type CreatePlaylistCommand struct { |
||||||
|
Name string `json:"name" binding:"Required"` |
||||||
|
Interval string `json:"interval"` |
||||||
|
Items []PlaylistItemDTO `json:"items"` |
||||||
|
|
||||||
|
OrgId int64 `json:"-"` |
||||||
|
} |
||||||
|
|
||||||
|
type DeletePlaylistCommand struct { |
||||||
|
UID string |
||||||
|
OrgId int64 |
||||||
|
} |
||||||
|
|
||||||
|
//
|
||||||
|
// QUERIES
|
||||||
|
//
|
||||||
|
|
||||||
|
type GetPlaylistsQuery struct { |
||||||
|
Name string |
||||||
|
Limit int |
||||||
|
OrgId int64 |
||||||
|
} |
||||||
|
|
||||||
|
type GetPlaylistByUidQuery struct { |
||||||
|
UID string |
||||||
|
OrgId int64 |
||||||
|
} |
||||||
|
|
||||||
|
type GetPlaylistItemsByUidQuery struct { |
||||||
|
PlaylistUID string |
||||||
|
OrgId int64 |
||||||
|
} |
@ -0,0 +1,14 @@ |
|||||||
|
package playlist |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
) |
||||||
|
|
||||||
|
type Service interface { |
||||||
|
Create(context.Context, *CreatePlaylistCommand) (*Playlist, error) |
||||||
|
Update(context.Context, *UpdatePlaylistCommand) (*PlaylistDTO, error) |
||||||
|
Get(context.Context, *GetPlaylistByUidQuery) (*Playlist, error) |
||||||
|
GetItems(context.Context, *GetPlaylistItemsByUidQuery) ([]PlaylistItem, error) |
||||||
|
Search(context.Context, *GetPlaylistsQuery) (Playlists, error) |
||||||
|
Delete(ctx context.Context, cmd *DeletePlaylistCommand) error |
||||||
|
} |
@ -0,0 +1,44 @@ |
|||||||
|
package playlistimpl |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
|
||||||
|
"github.com/grafana/grafana/pkg/services/playlist" |
||||||
|
"github.com/grafana/grafana/pkg/services/sqlstore/db" |
||||||
|
) |
||||||
|
|
||||||
|
type Service struct { |
||||||
|
store store |
||||||
|
} |
||||||
|
|
||||||
|
func ProvideService(db db.DB) playlist.Service { |
||||||
|
return &Service{ |
||||||
|
store: &sqlStore{ |
||||||
|
db: db, |
||||||
|
}, |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) Create(ctx context.Context, cmd *playlist.CreatePlaylistCommand) (*playlist.Playlist, error) { |
||||||
|
return s.store.Insert(ctx, cmd) |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) Update(ctx context.Context, cmd *playlist.UpdatePlaylistCommand) (*playlist.PlaylistDTO, error) { |
||||||
|
return s.store.Update(ctx, cmd) |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) Get(ctx context.Context, q *playlist.GetPlaylistByUidQuery) (*playlist.Playlist, error) { |
||||||
|
return s.store.Get(ctx, q) |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) GetItems(ctx context.Context, q *playlist.GetPlaylistItemsByUidQuery) ([]playlist.PlaylistItem, error) { |
||||||
|
return s.store.GetItems(ctx, q) |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) Search(ctx context.Context, q *playlist.GetPlaylistsQuery) (playlist.Playlists, error) { |
||||||
|
return s.store.List(ctx, q) |
||||||
|
} |
||||||
|
|
||||||
|
func (s *Service) Delete(ctx context.Context, cmd *playlist.DeletePlaylistCommand) error { |
||||||
|
return s.store.Delete(ctx, cmd) |
||||||
|
} |
@ -0,0 +1,226 @@ |
|||||||
|
package playlistimpl |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
|
||||||
|
"github.com/grafana/grafana/pkg/models" |
||||||
|
"github.com/grafana/grafana/pkg/services/playlist" |
||||||
|
"github.com/grafana/grafana/pkg/services/sqlstore" |
||||||
|
"github.com/grafana/grafana/pkg/services/sqlstore/db" |
||||||
|
"github.com/grafana/grafana/pkg/util" |
||||||
|
) |
||||||
|
|
||||||
|
type store interface { |
||||||
|
Insert(context.Context, *playlist.CreatePlaylistCommand) (*playlist.Playlist, error) |
||||||
|
Delete(context.Context, *playlist.DeletePlaylistCommand) error |
||||||
|
Get(context.Context, *playlist.GetPlaylistByUidQuery) (*playlist.Playlist, error) |
||||||
|
GetItems(context.Context, *playlist.GetPlaylistItemsByUidQuery) ([]playlist.PlaylistItem, error) |
||||||
|
List(context.Context, *playlist.GetPlaylistsQuery) (playlist.Playlists, error) |
||||||
|
Update(context.Context, *playlist.UpdatePlaylistCommand) (*playlist.PlaylistDTO, error) |
||||||
|
} |
||||||
|
|
||||||
|
type sqlStore struct { |
||||||
|
db db.DB |
||||||
|
} |
||||||
|
|
||||||
|
func (s *sqlStore) Insert(ctx context.Context, cmd *playlist.CreatePlaylistCommand) (*playlist.Playlist, error) { |
||||||
|
p := playlist.Playlist{} |
||||||
|
err := s.db.WithTransactionalDbSession(ctx, func(sess *sqlstore.DBSession) error { |
||||||
|
uid, err := generateAndValidateNewPlaylistUid(sess, cmd.OrgId) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
p = playlist.Playlist{ |
||||||
|
Name: cmd.Name, |
||||||
|
Interval: cmd.Interval, |
||||||
|
OrgId: cmd.OrgId, |
||||||
|
UID: uid, |
||||||
|
} |
||||||
|
|
||||||
|
_, err = sess.Insert(&p) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
playlistItems := make([]playlist.PlaylistItem, 0) |
||||||
|
for _, item := range cmd.Items { |
||||||
|
playlistItems = append(playlistItems, playlist.PlaylistItem{ |
||||||
|
PlaylistId: p.Id, |
||||||
|
Type: item.Type, |
||||||
|
Value: item.Value, |
||||||
|
Order: item.Order, |
||||||
|
Title: item.Title, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
_, err = sess.Insert(&playlistItems) |
||||||
|
|
||||||
|
return err |
||||||
|
}) |
||||||
|
return &p, err |
||||||
|
} |
||||||
|
|
||||||
|
func (s *sqlStore) Update(ctx context.Context, cmd *playlist.UpdatePlaylistCommand) (*playlist.PlaylistDTO, error) { |
||||||
|
dto := playlist.PlaylistDTO{} |
||||||
|
err := s.db.WithTransactionalDbSession(ctx, func(sess *sqlstore.DBSession) error { |
||||||
|
p := playlist.Playlist{ |
||||||
|
UID: cmd.UID, |
||||||
|
OrgId: cmd.OrgId, |
||||||
|
Name: cmd.Name, |
||||||
|
Interval: cmd.Interval, |
||||||
|
} |
||||||
|
|
||||||
|
existingPlaylist := playlist.Playlist{UID: cmd.UID, OrgId: cmd.OrgId} |
||||||
|
_, err := sess.Get(&existingPlaylist) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
p.Id = existingPlaylist.Id |
||||||
|
|
||||||
|
dto = playlist.PlaylistDTO{ |
||||||
|
|
||||||
|
Id: p.Id, |
||||||
|
UID: p.UID, |
||||||
|
OrgId: p.OrgId, |
||||||
|
Name: p.Name, |
||||||
|
Interval: p.Interval, |
||||||
|
} |
||||||
|
|
||||||
|
_, err = sess.Where("id=?", p.Id).Cols("name", "interval").Update(&p) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
rawSQL := "DELETE FROM playlist_item WHERE playlist_id = ?" |
||||||
|
_, err = sess.Exec(rawSQL, p.Id) |
||||||
|
|
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
playlistItems := make([]models.PlaylistItem, 0) |
||||||
|
|
||||||
|
for index, item := range cmd.Items { |
||||||
|
playlistItems = append(playlistItems, models.PlaylistItem{ |
||||||
|
PlaylistId: p.Id, |
||||||
|
Type: item.Type, |
||||||
|
Value: item.Value, |
||||||
|
Order: index + 1, |
||||||
|
Title: item.Title, |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
_, err = sess.Insert(&playlistItems) |
||||||
|
return err |
||||||
|
}) |
||||||
|
return &dto, err |
||||||
|
} |
||||||
|
|
||||||
|
func (s *sqlStore) Get(ctx context.Context, query *playlist.GetPlaylistByUidQuery) (*playlist.Playlist, error) { |
||||||
|
if query.UID == "" || query.OrgId == 0 { |
||||||
|
return nil, playlist.ErrCommandValidationFailed |
||||||
|
} |
||||||
|
|
||||||
|
p := playlist.Playlist{} |
||||||
|
err := s.db.WithDbSession(ctx, func(sess *sqlstore.DBSession) error { |
||||||
|
p = playlist.Playlist{UID: query.UID, OrgId: query.OrgId} |
||||||
|
exists, err := sess.Get(&p) |
||||||
|
if !exists { |
||||||
|
return playlist.ErrPlaylistNotFound |
||||||
|
} |
||||||
|
|
||||||
|
return err |
||||||
|
}) |
||||||
|
return &p, err |
||||||
|
} |
||||||
|
|
||||||
|
func (s *sqlStore) Delete(ctx context.Context, cmd *playlist.DeletePlaylistCommand) error { |
||||||
|
if cmd.UID == "" || cmd.OrgId == 0 { |
||||||
|
return playlist.ErrCommandValidationFailed |
||||||
|
} |
||||||
|
|
||||||
|
return s.db.WithTransactionalDbSession(ctx, func(sess *sqlstore.DBSession) error { |
||||||
|
playlist := playlist.Playlist{UID: cmd.UID, OrgId: cmd.OrgId} |
||||||
|
_, err := sess.Get(&playlist) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
var rawPlaylistSQL = "DELETE FROM playlist WHERE uid = ? and org_id = ?" |
||||||
|
_, err = sess.Exec(rawPlaylistSQL, cmd.UID, cmd.OrgId) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
var rawItemSQL = "DELETE FROM playlist_item WHERE playlist_id = ?" |
||||||
|
_, err = sess.Exec(rawItemSQL, playlist.Id) |
||||||
|
|
||||||
|
return err |
||||||
|
}) |
||||||
|
} |
||||||
|
|
||||||
|
func (s *sqlStore) List(ctx context.Context, query *playlist.GetPlaylistsQuery) (playlist.Playlists, error) { |
||||||
|
playlists := make(playlist.Playlists, 0) |
||||||
|
if query.OrgId == 0 { |
||||||
|
return playlists, playlist.ErrCommandValidationFailed |
||||||
|
} |
||||||
|
|
||||||
|
err := s.db.WithDbSession(ctx, func(dbSess *sqlstore.DBSession) error { |
||||||
|
sess := dbSess.Limit(query.Limit) |
||||||
|
|
||||||
|
if query.Name != "" { |
||||||
|
sess.Where("name LIKE ?", "%"+query.Name+"%") |
||||||
|
} |
||||||
|
|
||||||
|
sess.Where("org_id = ?", query.OrgId) |
||||||
|
err := sess.Find(&playlists) |
||||||
|
|
||||||
|
return err |
||||||
|
}) |
||||||
|
return playlists, err |
||||||
|
} |
||||||
|
|
||||||
|
func (s *sqlStore) GetItems(ctx context.Context, query *playlist.GetPlaylistItemsByUidQuery) ([]playlist.PlaylistItem, error) { |
||||||
|
var playlistItems = make([]playlist.PlaylistItem, 0) |
||||||
|
err := s.db.WithDbSession(ctx, func(sess *sqlstore.DBSession) error { |
||||||
|
if query.PlaylistUID == "" || query.OrgId == 0 { |
||||||
|
return models.ErrCommandValidationFailed |
||||||
|
} |
||||||
|
|
||||||
|
// getQuery the playlist Id
|
||||||
|
getQuery := &playlist.GetPlaylistByUidQuery{UID: query.PlaylistUID, OrgId: query.OrgId} |
||||||
|
p, err := s.Get(ctx, getQuery) |
||||||
|
if err != nil { |
||||||
|
return err |
||||||
|
} |
||||||
|
|
||||||
|
err = sess.Where("playlist_id=?", p.Id).Find(&playlistItems) |
||||||
|
|
||||||
|
return err |
||||||
|
}) |
||||||
|
return playlistItems, err |
||||||
|
} |
||||||
|
|
||||||
|
// generateAndValidateNewPlaylistUid generates a playlistUID and verifies that
|
||||||
|
// the uid isn't already in use. This is deliberately overly cautious, since users
|
||||||
|
// can also specify playlist uids during provisioning.
|
||||||
|
func generateAndValidateNewPlaylistUid(sess *sqlstore.DBSession, orgId int64) (string, error) { |
||||||
|
for i := 0; i < 3; i++ { |
||||||
|
uid := generateNewUid() |
||||||
|
|
||||||
|
playlist := models.Playlist{OrgId: orgId, UID: uid} |
||||||
|
exists, err := sess.Get(&playlist) |
||||||
|
if err != nil { |
||||||
|
return "", err |
||||||
|
} |
||||||
|
|
||||||
|
if !exists { |
||||||
|
return uid, nil |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
return "", models.ErrPlaylistFailedGenerateUniqueUid |
||||||
|
} |
||||||
|
|
||||||
|
var generateNewUid func() string = util.GenerateShortUID |
@ -0,0 +1,82 @@ |
|||||||
|
package playlistimpl |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
"testing" |
||||||
|
|
||||||
|
"github.com/grafana/grafana/pkg/services/playlist" |
||||||
|
"github.com/grafana/grafana/pkg/services/sqlstore" |
||||||
|
"github.com/stretchr/testify/require" |
||||||
|
) |
||||||
|
|
||||||
|
func TestIntegrationPlaylistDataAccess(t *testing.T) { |
||||||
|
if testing.Short() { |
||||||
|
t.Skip("skipping integration test") |
||||||
|
} |
||||||
|
ss := sqlstore.InitTestDB(t) |
||||||
|
playlistStore := sqlStore{db: ss} |
||||||
|
|
||||||
|
t.Run("Can create playlist", func(t *testing.T) { |
||||||
|
items := []playlist.PlaylistItemDTO{ |
||||||
|
{Title: "graphite", Value: "graphite", Type: "dashboard_by_tag"}, |
||||||
|
{Title: "Backend response times", Value: "3", Type: "dashboard_by_id"}, |
||||||
|
} |
||||||
|
cmd := playlist.CreatePlaylistCommand{Name: "NYC office", Interval: "10m", OrgId: 1, Items: items} |
||||||
|
p, err := playlistStore.Insert(context.Background(), &cmd) |
||||||
|
require.NoError(t, err) |
||||||
|
uid := p.UID |
||||||
|
|
||||||
|
t.Run("Can get playlist items", func(t *testing.T) { |
||||||
|
get := &playlist.GetPlaylistItemsByUidQuery{PlaylistUID: uid, OrgId: 1} |
||||||
|
storedPlaylistItems, err := playlistStore.GetItems(context.Background(), get) |
||||||
|
require.NoError(t, err) |
||||||
|
require.Equal(t, len(storedPlaylistItems), len(items)) |
||||||
|
}) |
||||||
|
|
||||||
|
t.Run("Can update playlist", func(t *testing.T) { |
||||||
|
items := []playlist.PlaylistItemDTO{ |
||||||
|
{Title: "influxdb", Value: "influxdb", Type: "dashboard_by_tag"}, |
||||||
|
{Title: "Backend response times", Value: "2", Type: "dashboard_by_id"}, |
||||||
|
} |
||||||
|
query := playlist.UpdatePlaylistCommand{Name: "NYC office ", OrgId: 1, UID: uid, Interval: "10s", Items: items} |
||||||
|
_, err = playlistStore.Update(context.Background(), &query) |
||||||
|
require.NoError(t, err) |
||||||
|
}) |
||||||
|
|
||||||
|
t.Run("Can remove playlist", func(t *testing.T) { |
||||||
|
deleteQuery := playlist.DeletePlaylistCommand{UID: uid, OrgId: 1} |
||||||
|
err = playlistStore.Delete(context.Background(), &deleteQuery) |
||||||
|
require.NoError(t, err) |
||||||
|
|
||||||
|
getQuery := playlist.GetPlaylistByUidQuery{UID: uid, OrgId: 1} |
||||||
|
p, err := playlistStore.Get(context.Background(), &getQuery) |
||||||
|
require.Error(t, err) |
||||||
|
require.Equal(t, uid, p.UID, "playlist should've been removed") |
||||||
|
require.ErrorIs(t, err, playlist.ErrPlaylistNotFound) |
||||||
|
}) |
||||||
|
}) |
||||||
|
|
||||||
|
t.Run("Delete playlist that doesn't exist", func(t *testing.T) { |
||||||
|
deleteQuery := playlist.DeletePlaylistCommand{UID: "654312", OrgId: 1} |
||||||
|
err := playlistStore.Delete(context.Background(), &deleteQuery) |
||||||
|
require.NoError(t, err) |
||||||
|
}) |
||||||
|
|
||||||
|
t.Run("Delete playlist with invalid command yields error", func(t *testing.T) { |
||||||
|
testCases := []struct { |
||||||
|
desc string |
||||||
|
cmd playlist.DeletePlaylistCommand |
||||||
|
}{ |
||||||
|
{desc: "none", cmd: playlist.DeletePlaylistCommand{}}, |
||||||
|
{desc: "no OrgId", cmd: playlist.DeletePlaylistCommand{UID: "1"}}, |
||||||
|
{desc: "no Uid", cmd: playlist.DeletePlaylistCommand{OrgId: 1}}, |
||||||
|
} |
||||||
|
|
||||||
|
for _, tc := range testCases { |
||||||
|
t.Run(tc.desc, func(t *testing.T) { |
||||||
|
err := playlistStore.Delete(context.Background(), &tc.cmd) |
||||||
|
require.EqualError(t, err, playlist.ErrCommandValidationFailed.Error()) |
||||||
|
}) |
||||||
|
} |
||||||
|
}) |
||||||
|
} |
@ -0,0 +1,43 @@ |
|||||||
|
package playlisttest |
||||||
|
|
||||||
|
import ( |
||||||
|
"context" |
||||||
|
|
||||||
|
"github.com/grafana/grafana/pkg/services/playlist" |
||||||
|
) |
||||||
|
|
||||||
|
type FakePlaylistService struct { |
||||||
|
ExpectedPlaylist *playlist.Playlist |
||||||
|
ExpectedPlaylistDTO *playlist.PlaylistDTO |
||||||
|
ExpectedPlaylistItems []playlist.PlaylistItem |
||||||
|
ExpectedPlaylists playlist.Playlists |
||||||
|
ExpectedError error |
||||||
|
} |
||||||
|
|
||||||
|
func NewPlaylistServiveFake() *FakePlaylistService { |
||||||
|
return &FakePlaylistService{} |
||||||
|
} |
||||||
|
|
||||||
|
func (f *FakePlaylistService) Create(context.Context, *playlist.CreatePlaylistCommand) (*playlist.Playlist, error) { |
||||||
|
return f.ExpectedPlaylist, f.ExpectedError |
||||||
|
} |
||||||
|
|
||||||
|
func (f *FakePlaylistService) Update(context.Context, *playlist.UpdatePlaylistCommand) (*playlist.PlaylistDTO, error) { |
||||||
|
return f.ExpectedPlaylistDTO, f.ExpectedError |
||||||
|
} |
||||||
|
|
||||||
|
func (f *FakePlaylistService) Get(context.Context, *playlist.GetPlaylistByUidQuery) (*playlist.Playlist, error) { |
||||||
|
return f.ExpectedPlaylist, f.ExpectedError |
||||||
|
} |
||||||
|
|
||||||
|
func (f *FakePlaylistService) GetItems(context.Context, *playlist.GetPlaylistItemsByUidQuery) ([]playlist.PlaylistItem, error) { |
||||||
|
return f.ExpectedPlaylistItems, f.ExpectedError |
||||||
|
} |
||||||
|
|
||||||
|
func (f *FakePlaylistService) Search(context.Context, *playlist.GetPlaylistsQuery) (playlist.Playlists, error) { |
||||||
|
return f.ExpectedPlaylists, f.ExpectedError |
||||||
|
} |
||||||
|
|
||||||
|
func (f *FakePlaylistService) Delete(ctx context.Context, cmd *playlist.DeletePlaylistCommand) error { |
||||||
|
return f.ExpectedError |
||||||
|
} |
Loading…
Reference in new issue