• Home
  • Features
  • Pricing
  • Docs
  • Announcements
  • Sign In

supabase / storage / 23951363366

03 Apr 2026 03:21PM UTC coverage: 80.45% (-0.03%) from 80.479%
23951363366

Pull #965

github

web-flow
Merge ca6a1e7dd into 1587bf2e8
Pull Request #965: fix: report errors in orphan and test entity expansion

3126 of 4067 branches covered (76.86%)

Branch coverage included in aggregate %.

104 of 153 new or added lines in 4 files covered. (67.97%)

7 existing lines in 1 file now uncovered.

30012 of 37124 relevant lines covered (80.84%)

311.36 hits per line

Source File
Press 'n' to go to next uncovered line, 'b' for previous

44.0
/src/http/routes/admin/objects.ts
1
import { render } from '@internal/errors'
1✔
2
import { ObjectScanner } from '@storage/scanner/scanner'
1✔
3
import { FastifyInstance, RequestGenericInterface } from 'fastify'
1✔
4
import { FastifyReply } from 'fastify/types/reply'
1✔
5
import { dbSuperUser, storage } from '../../plugins'
1✔
6
import apiKey from '../../plugins/apikey'
1✔
7

1✔
8
const listOrphanedObjects = {
1✔
9
  description: 'List Orphaned Objects',
1✔
10
  params: {
1✔
11
    type: 'object',
1✔
12
    properties: {
1✔
13
      tenantId: { type: 'string' },
1✔
14
      bucketId: { type: 'string' },
1✔
15
    },
1✔
16
    required: ['tenantId', 'bucketId'],
1✔
17
  },
1✔
18
  query: {
1✔
19
    type: 'object',
1✔
20
    properties: {
1✔
21
      before: { type: 'string' },
1✔
22
      keepTmpTable: { type: 'boolean' },
1✔
23
    },
1✔
24
  },
1✔
25
} as const
1✔
26

1✔
27
const syncOrphanedObjects = {
1✔
28
  description: 'Sync Orphaned Objects',
1✔
29
  params: {
1✔
30
    type: 'object',
1✔
31
    properties: {
1✔
32
      tenantId: { type: 'string' },
1✔
33
      bucketId: { type: 'string' },
1✔
34
    },
1✔
35
    required: ['tenantId', 'bucketId'],
1✔
36
  },
1✔
37
  body: {
1✔
38
    type: 'object',
1✔
39
    properties: {
1✔
40
      deleteDbKeys: { type: 'boolean' },
1✔
41
      deleteS3Keys: { type: 'boolean' },
1✔
42
      tmpTable: { type: 'string' },
1✔
43
    },
1✔
44
  },
1✔
45
  optional: ['deleteDbKeys', 'deleteS3Keys'],
1✔
46
} as const
1✔
47

1✔
48
interface ListOrphanObjectsRequest extends RequestGenericInterface {
1✔
49
  Params: {
1✔
50
    tenantId: string
1✔
51
    bucketId: string
1✔
52
  }
1✔
53
  Querystring: {
1✔
54
    before?: string
1✔
55
    keepTmpTable?: boolean
1✔
56
  }
1✔
57
}
1✔
58

1✔
59
interface SyncOrphanObjectsRequest extends RequestGenericInterface {
1✔
60
  Params: {
1✔
61
    tenantId: string
1✔
62
    bucketId: string
1✔
63
  }
1✔
64
  Body: {
1✔
65
    deleteDbKeys?: boolean
1✔
66
    deleteS3Keys?: boolean
1✔
67
    before?: string
1✔
68
    tmpTable?: string
1✔
69
    keepTmpTable?: boolean
1✔
70
  }
1✔
71
}
1✔
72

1✔
73
export default async function routes(fastify: FastifyInstance) {
1✔
74
  fastify.register(apiKey)
8✔
75
  fastify.register(dbSuperUser, {
8✔
76
    disableHostCheck: true,
8✔
77
    maxConnections: 5,
8✔
78
  })
8✔
79
  fastify.register(storage)
8✔
80

8✔
81
  fastify.get<ListOrphanObjectsRequest>(
8✔
82
    '/:tenantId/buckets/:bucketId/orphan-objects',
8✔
83
    {
8✔
84
      schema: { ...listOrphanedObjects, tags: ['object'] },
8✔
85
    },
8✔
86
    async (req, reply) => {
8✔
87
      const bucket = req.params.bucketId
×
88
      let before = req.query.before ? new Date(req.query.before as string) : undefined
×
89

×
90
      if (before && isNaN(before.getTime())) {
×
91
        return reply.status(400).send({
×
92
          error: 'Invalid date format',
×
93
        })
×
94
      }
×
95
      if (!before) {
×
96
        before = new Date()
×
97
        before.setHours(before.getHours() - 1)
×
98
      }
×
99

×
100
      const scanner = new ObjectScanner(req.storage)
×
101
      const orphanObjects = scanner.listOrphaned(bucket, {
×
102
        signal: req.signals.disconnect.signal,
×
103
        before,
×
104
        keepTmpTable: Boolean(req.query.keepTmpTable),
×
105
      })
×
106

×
107
      reply.header('Content-Type', 'application/x-ndjson; charset=utf-8')
×
108

×
109
      // Do not let the connection time out, periodically send
×
110
      // a ping message to keep the connection alive
×
111
      const respPing = ping(reply)
×
112

×
113
      try {
×
114
        for await (const result of orphanObjects) {
×
115
          if (result.value.length > 0) {
×
116
            respPing.update()
×
117
            reply.raw.write(
×
118
              JSON.stringify({
×
119
                ...result,
×
120
                event: 'data',
×
121
              }) + '\n'
×
122
            )
×
123
          }
×
124
        }
×
125
      } catch (e) {
×
NEW
126
        req.log.error({ err: e, bucket }, 'list orphaned objects stream failed')
×
NEW
127
        reply.raw.write(
×
NEW
128
          JSON.stringify({
×
NEW
129
            event: 'error',
×
NEW
130
            error: render(e),
×
NEW
131
          }) + '\n'
×
NEW
132
        )
×
NEW
133
        return
×
134
      } finally {
×
UNCOV
135
        respPing.clear()
×
UNCOV
136
        reply.raw.end()
×
UNCOV
137
      }
×
UNCOV
138
    }
×
139
  )
8✔
140

8✔
141
  fastify.delete<SyncOrphanObjectsRequest>(
8✔
142
    '/:tenantId/buckets/:bucketId/orphan-objects',
8✔
143
    {
8✔
144
      schema: { ...syncOrphanedObjects, tags: ['object'] },
8✔
145
    },
8✔
146
    async (req, reply) => {
8✔
147
      if (!req.body.deleteDbKeys && !req.body.deleteS3Keys) {
×
148
        return reply.status(400).send({
×
149
          error: 'At least one of deleteDbKeys or deleteS3Keys must be set to true',
×
150
        })
×
151
      }
×
152

×
153
      const bucket = `${req.params.bucketId}`
×
154
      let before = req.body.before ? new Date(req.body.before as string) : undefined
×
155

×
156
      if (!before) {
×
157
        before = new Date()
×
158
        before.setHours(before.getHours() - 1)
×
159
      }
×
160

×
161
      reply.header('Content-Type', 'application/x-ndjson; charset=utf-8')
×
162

×
163
      const respPing = ping(reply)
×
164

×
165
      try {
×
166
        const scanner = new ObjectScanner(req.storage)
×
167
        const result = scanner.deleteOrphans(bucket, {
×
168
          deleteDbKeys: req.body.deleteDbKeys,
×
169
          deleteS3Keys: req.body.deleteS3Keys,
×
170
          signal: req.signals.disconnect.signal,
×
171
          before,
×
172
          tmpTable: req.body.tmpTable,
×
173
        })
×
174

×
175
        for await (const deleted of result) {
×
176
          respPing.update()
×
177
          reply.raw.write(
×
178
            JSON.stringify({
×
179
              ...deleted,
×
180
              event: 'data',
×
181
            }) + '\n'
×
182
          )
×
183
        }
×
184
      } catch (e) {
×
NEW
185
        req.log.error({ err: e, bucket }, 'delete orphaned objects stream failed')
×
NEW
186
        reply.raw.write(
×
NEW
187
          JSON.stringify({
×
NEW
188
            event: 'error',
×
NEW
189
            error: render(e),
×
NEW
UNCOV
190
          }) + '\n'
×
NEW
UNCOV
191
        )
×
NEW
UNCOV
192
        return
×
193
      } finally {
×
194
        respPing.clear()
×
195
        reply.raw.end()
×
196
      }
×
197
    }
×
198
  )
8✔
199
}
8✔
200

1✔
201
// Occasionally write a ping message to the response stream
1✔
202
function ping(reply: FastifyReply) {
×
203
  let lastSend = undefined as Date | undefined
×
204
  const clearPing = setInterval(() => {
×
205
    const fiveSecondsEarly = new Date()
×
206
    fiveSecondsEarly.setSeconds(fiveSecondsEarly.getSeconds() - 5)
×
207

×
208
    if (!lastSend || (lastSend && lastSend < fiveSecondsEarly)) {
×
209
      lastSend = new Date()
×
210
      reply.raw.write(
×
211
        JSON.stringify({
×
212
          event: 'ping',
×
213
        }) + '\n'
×
214
      )
×
215
    }
×
216
  }, 1000 * 10)
×
217

×
218
  return {
×
219
    clear: () => clearInterval(clearPing),
×
220
    update: () => {
×
221
      lastSend = new Date()
×
222
    },
×
223
  }
×
224
}
×
STATUS · Troubleshooting · Open an Issue · Sales · Support · CAREERS · ENTERPRISE · START FREE · SCHEDULE DEMO
ANNOUNCEMENTS · TWITTER · TOS & SLA · Supported CI Services · What's a CI service? · Automated Testing

© 2026 Coveralls, Inc