Skip to content

Commit

Permalink
fix: type errors and lints in listManifests
Browse files Browse the repository at this point in the history
  • Loading branch information
gabivlj committed Mar 4, 2024
1 parent 7e58b84 commit d4b262b
Show file tree
Hide file tree
Showing 5 changed files with 110 additions and 30 deletions.
50 changes: 49 additions & 1 deletion index.test.ts
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import { Env } from ".";
import * as fetchAuth from "./index";
import { RegistryTokens } from "./src/token";
import { RegistryAuthProtocolTokenPayload } from "./src/auth";
import { registries } from "./src/registry/registry";
import { ListRepositoriesResponse, registries } from "./src/registry/registry";

Check failure on line 8 in index.test.ts

View workflow job for this annotation

GitHub Actions / Unit Tests (20.x)

'ListRepositoriesResponse' is declared but its value is never read.
import { RegistryHTTPClient } from "./src/registry/http";

function createRequest(method: string, path: string, body: ReadableStream | null, headers = {}) {
Expand Down Expand Up @@ -112,6 +112,7 @@ describe("v2 manifests", () => {
"content-type": "application/gzip",
"docker-content-digest": sha256,
});
await bindings.REGISTRY.delete(`${name}/manifests/${reference}`);
});

test("PUT /v2/:name/manifests/:reference works", () => createManifest("hello-world-main", "{}", "hello"));
Expand Down Expand Up @@ -371,3 +372,50 @@ describe("http client", () => {
expect("exists" in res && res.exists).toBe(false);
});
});

describe("push and catalog", () => {
test("push and then use the catalog", async () => {
await createManifest("hello-world-main", "{}", "hello");
await createManifest("hello-world-main", "{}", "latest");
await createManifest("hello-world-main", "{}", "hello-2");
await createManifest("hello", "{}", "hello");
await createManifest("hello/hello", "{}", "hello");

const response = await fetchUnauth(createRequest("GET", "/v2/_catalog", null));
expect(response.ok).toBeTruthy();
const body = (await response.json()) as { repositories: string[] };
expect(body).toEqual({
repositories: ["hello-world-main", "hello/hello", "hello"],
});
const expectedRepositories = body.repositories;
const tagsRes = await fetchUnauth(createRequest("GET", `/v2/hello-world-main/tags/list?n=1000`, null));
const tags = (await tagsRes.json()) as TagsList;
expect(tags.name).toEqual("hello-world-main");
expect(tags.tags).toEqual([
"hello",
"hello-2",
"latest",
"sha256:44136fa355b3678a1146ad16f7e8649e94fb4fc21fe77e8310c060f61caaff8a",
]);

const repositoryBuildUp: string[] = [];
let currentPath = "/v2/_catalog?n=1";
for (let i = 0; i < 3; i++) {
const response = await fetchUnauth(createRequest("GET", currentPath, null));
expect(response.ok).toBeTruthy();
const body = (await response.json()) as { repositories: string[] };
if (body.repositories.length === 0) {
break;
}
expect(body.repositories).toHaveLength(1);

repositoryBuildUp.push(...body.repositories);
const url = new URL(response.headers.get("Link")!.split(";")[0].trim());
console.log(url.pathname, url.search, response.headers.get("Link"));
currentPath = url.pathname + url.search;
console.log(currentPath);
}

expect(repositoryBuildUp).toEqual(expectedRepositories);
});
});
5 changes: 5 additions & 0 deletions src/registry/http.ts
Original file line number Diff line number Diff line change
Expand Up @@ -7,6 +7,7 @@ import {
FinishedUploadObject,
GetLayerResponse,
GetManifestResponse,
ListRepositoriesResponse,
PutManifestResponse,
Registry,
RegistryConfiguration,
Expand Down Expand Up @@ -467,6 +468,10 @@ export class RegistryHTTPClient implements Registry {
): Promise<RegistryError | FinishedUploadObject> {
throw new Error("unimplemented");
}

async listRepositories(_limit?: number, _last?: string): Promise<RegistryError | ListRepositoriesResponse> {
throw new Error("unimplemented");
}
}

// AuthType defined the supported auth types
Expand Down
48 changes: 32 additions & 16 deletions src/registry/r2.ts
Original file line number Diff line number Diff line change
Expand Up @@ -132,30 +132,46 @@ export class R2Registry implements Registry {
}

async listRepositories(limit?: number, last?: string): Promise<RegistryError | ListRepositoriesResponse> {
const env = this.env;
console.log("cursor", last, "/");
const options = {
limit: limit ? limit : 1000,
delimiter: "/",
startAfter: last,
}
const r2Objects = (await env.REGISTRY.list(options));
limit: limit ?? 1000,
cursor: last ?? undefined,
};
const paths: Record<string, {}> = {};
let totalRecords = 0;
const addObjectPath = (object: R2Object) => {
if (totalRecords >= options.limit) return;
// skip either 'manifests' or 'blobs'
const parts = object.key.split("/");
const key = parts.slice(0, parts.length - 2).join("/");
if (!(key in paths)) {
totalRecords++;
}

let truncated = r2Objects.truncated;
let cursor = truncated ? r2Objects.cursor : undefined;
paths[key] = {};
};

while (truncated) {
const next = await env.REGISTRY.list({
const r2Objects = await this.env.REGISTRY.list({
limit: options.limit,
cursor: options.cursor,
});
r2Objects.objects.forEach((path) => addObjectPath(path));
//TODO: identify the key that we stopped adding and set it as cursor
let cursor = r2Objects.truncated ? r2Objects.cursor : undefined;
while (r2Objects.truncated && totalRecords < options.limit) {
const next = await this.env.REGISTRY.list({
...options,
cursor: cursor,
cursor,
});
r2Objects.objects.push(...next.objects);

truncated = next.truncated;
cursor = next.cursor
next.objects.forEach((path) => addObjectPath(path));
if (next.truncated) {
cursor = next.cursor;
}
}

return {
repositories: r2Objects.delimitedPrefixes.map((name)=> name.endsWith('/') ? name.slice(0, -1) : name)
repositories: Object.keys(paths),
cursor,
};
}

Expand Down
9 changes: 4 additions & 5 deletions src/registry/registry.ts
Original file line number Diff line number Diff line change
Expand Up @@ -43,11 +43,10 @@ export type CheckManifestResponse =
exists: false;
};

export type ListRepositoriesResponse =
{
repositories: string[];
}

export type ListRepositoriesResponse = {
repositories: string[];
cursor?: string;
};

// Response layerExists call
export type CheckLayerResponse =
Expand Down
28 changes: 20 additions & 8 deletions src/router.ts
Original file line number Diff line number Diff line change
Expand Up @@ -27,15 +27,27 @@ v2Router.get("/", async (_req, _env: Env) => {

v2Router.get("/_catalog", async (req, env: Env) => {
const { n, last } = req.query;
const res = await env.REGISTRY_CLIENT.listRepositories(
const response = await env.REGISTRY_CLIENT.listRepositories(
n ? parseInt(n?.toString()) : undefined,
last?.toString()
last?.toString(),
);
if ("response" in response) {
return response.response;
}

return new Response(JSON.stringify(res));
const url = new URL(req.url);
return new Response(
JSON.stringify({
repositories: response.repositories,
}),
{
headers: {
Link: `${url.protocol}//${url.hostname}${url.pathname}?n=${n ?? 1000}&last=${response.cursor ?? ""}; rel=next`,
},
},
);
});


v2Router.delete("/:name+/manifests/:reference", async (req, env: Env) => {
// deleting a manifest works by retrieving the """main""" manifest that its key is a sha,
// and then going through every tag and removing it
Expand All @@ -49,17 +61,17 @@ v2Router.delete("/:name+/manifests/:reference", async (req, env: Env) => {
//
// If somehow we need to remove by paginating, we accept a last query param

const { last } = req.query;
const { last, limit } = req.query;
const { name, reference } = req.params;
// Reference is ALWAYS a sha256
const manifest = await env.REGISTRY.head(`${name}/manifests/${reference}`);
if (manifest === null) {
return new Response(JSON.stringify(ManifestUnknownError), { status: 404 });
}

const limitInt = parseInt(limit?.toString() ?? "1000", 10);
const tags = await env.REGISTRY.list({
prefix: `${name}/manifests`,
limit: 1000,
limit: isNaN(limitInt) ? 1000 : limitInt,
startAfter: last?.toString(),
});
for (const tag of tags.objects) {
Expand All @@ -76,7 +88,7 @@ v2Router.delete("/:name+/manifests/:reference", async (req, env: Env) => {
return new Response(JSON.stringify(ManifestTagsListTooBigError), {
status: 400,
headers: {
"Link": `${req.url}/last=${tags.objects[tags.objects.length - 1]}; rel=next`,
"Link": `${req.url}/?last=${tags.truncated ? tags.cursor : ""}; rel=next`,
"Content-Type": "application/json",
},
});
Expand Down

0 comments on commit d4b262b

Please sign in to comment.