2020-07-31 18:22:09 +00:00
|
|
|
import { ABC, Base64 } from "./deps.ts";
|
2020-07-28 12:39:54 +00:00
|
|
|
import config from "./config.ts";
|
|
|
|
|
|
|
|
const packageNameRegex = /^[@]?[a-zA-Z][\d\w\-\_]*$/g.compile();
|
|
|
|
const packageVersionRegex = /^\d(\.\d)?(\.\d)?$/g.compile();
|
|
|
|
const packageFullVersionRegex = /^\d(\.\d)(\.\d)$/g.compile();
|
|
|
|
|
|
|
|
export const isValidPackageName = (name: string) => packageNameRegex.test(name);
|
|
|
|
export const isValidVersion = (version: string) =>
|
|
|
|
packageVersionRegex.test(version);
|
|
|
|
export const isValidFullVersion = (version: string) =>
|
|
|
|
packageFullVersionRegex.test(version);
|
|
|
|
|
|
|
|
const ALg = 1;
|
|
|
|
const ASm = -ALg;
|
|
|
|
const Equ = 0;
|
|
|
|
|
|
|
|
export const sortVersions = (a: string, b: string) => {
|
|
|
|
const [a1, a2, a3] = a.split(".").map(Number);
|
|
|
|
const [b1, b2, b3] = b.split(".").map(Number);
|
|
|
|
|
|
|
|
if (a1 > b1) return ALg;
|
|
|
|
if (a1 < b1) return ASm;
|
|
|
|
|
|
|
|
if (a2 > b2) return ALg;
|
|
|
|
if (a2 < b2) return ASm;
|
|
|
|
|
|
|
|
if (a3 > b3) return ALg;
|
|
|
|
if (a3 < b3) return ASm;
|
|
|
|
|
|
|
|
return Equ;
|
|
|
|
};
|
|
|
|
|
|
|
|
export const basicauth = (realm: string) => (next: ABC.HandlerFunc) => (
|
|
|
|
ctx: ABC.Context
|
|
|
|
) => {
|
|
|
|
const value = ctx.request.headers.get("authorization");
|
|
|
|
|
|
|
|
console.log("Header:", value);
|
|
|
|
|
|
|
|
if (value && value.toLowerCase().startsWith("basic ")) {
|
|
|
|
const credentials = value.slice(6);
|
|
|
|
const [username, passwd] = new TextDecoder()
|
2020-07-31 18:22:09 +00:00
|
|
|
.decode(Base64.decode(credentials))
|
2020-07-28 12:39:54 +00:00
|
|
|
.split(":", 2);
|
|
|
|
|
|
|
|
if (config?.user[username]?.password === passwd) {
|
|
|
|
console.log("User authenticated!");
|
2020-07-31 18:16:12 +00:00
|
|
|
if (!ctx.customContext) ctx.customContext = {};
|
|
|
|
ctx.customContext.user = username;
|
2020-07-28 12:39:54 +00:00
|
|
|
return next(ctx);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
console.log("Authentication required");
|
|
|
|
ctx.response.status = 401;
|
|
|
|
ctx.response.headers.set("WWW-Authenticate", "Basic realm=" + realm);
|
|
|
|
return {
|
|
|
|
statusCode: 401,
|
|
|
|
error: "Authentication required",
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
export function extractPackagePath(path: string): [string, string | undefined] {
|
|
|
|
let packageName = "";
|
2020-07-31 18:16:12 +00:00
|
|
|
path = path.toLowerCase();
|
2020-08-02 20:33:18 +00:00
|
|
|
|
2020-07-28 12:39:54 +00:00
|
|
|
if (path.startsWith("@")) {
|
|
|
|
packageName = "@";
|
|
|
|
path = path.slice(1);
|
|
|
|
}
|
|
|
|
|
|
|
|
let parts = path.split("@");
|
|
|
|
if (parts.length > 2) throw new Error("Invalid package name!");
|
|
|
|
|
|
|
|
packageName += parts[0];
|
2020-08-02 20:39:05 +00:00
|
|
|
let packageVersion: string | undefined = parts[1];
|
2020-07-28 12:39:54 +00:00
|
|
|
|
|
|
|
if (!isValidPackageName(packageName))
|
|
|
|
throw new Error("Invalid package name!");
|
|
|
|
|
|
|
|
if (packageVersion !== "") {
|
|
|
|
if (!isValidVersion(packageVersion))
|
|
|
|
throw new Error("Invalid package version!");
|
|
|
|
}
|
|
|
|
|
|
|
|
return [packageName, packageVersion];
|
|
|
|
}
|
2020-07-31 18:16:12 +00:00
|
|
|
|
2020-10-14 00:52:02 +00:00
|
|
|
import type { IPackage } from "./db.ts";
|
2020-07-31 18:16:12 +00:00
|
|
|
|
|
|
|
import bucket from "./s3.ts";
|
|
|
|
|
2020-10-14 00:52:02 +00:00
|
|
|
export function getAbsolutePackageVersion(
|
|
|
|
pkg?: IPackage | null,
|
|
|
|
version?: string
|
|
|
|
) {
|
|
|
|
if (!pkg || pkg.versions.length < 1) return undefined;
|
2020-07-31 18:16:12 +00:00
|
|
|
|
2020-10-14 00:52:02 +00:00
|
|
|
const versions = pkg.versions.sort(sortVersions).reverse();
|
2020-07-31 18:16:12 +00:00
|
|
|
|
|
|
|
if (!version) {
|
|
|
|
version = versions[0];
|
|
|
|
} else {
|
|
|
|
const v = versions.filter((e) => e.startsWith(version as string));
|
2020-10-14 00:52:02 +00:00
|
|
|
if (v.length < 1) return undefined;
|
2020-07-31 18:16:12 +00:00
|
|
|
version = v[0];
|
|
|
|
}
|
|
|
|
|
2020-10-14 00:52:02 +00:00
|
|
|
return version;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getBucketFilePath(
|
|
|
|
pkgName: string,
|
|
|
|
version: string,
|
|
|
|
file: string
|
|
|
|
) {
|
|
|
|
if (file.startsWith("/")) {
|
|
|
|
file = file.substr(1);
|
|
|
|
}
|
|
|
|
|
2020-07-31 18:16:12 +00:00
|
|
|
const bucketPath = (
|
|
|
|
"packages/" +
|
|
|
|
pkgName +
|
|
|
|
"/" +
|
|
|
|
version +
|
|
|
|
"/" +
|
|
|
|
file
|
|
|
|
).replace(/@/g, "§");
|
|
|
|
|
2020-10-14 00:52:02 +00:00
|
|
|
return bucketPath;
|
|
|
|
}
|
|
|
|
|
|
|
|
export async function getFile(
|
|
|
|
pkgName: string,
|
|
|
|
version: string | undefined,
|
|
|
|
file: string
|
|
|
|
): Promise<{ etag: string; data: Uint8Array } | null | undefined> {
|
|
|
|
if (!version) return undefined;
|
|
|
|
const bucketPath = await getBucketFilePath(pkgName, version, file);
|
|
|
|
if (!bucketPath) return null;
|
|
|
|
|
2020-07-31 18:16:12 +00:00
|
|
|
console.log("Getting file from:", bucketPath);
|
|
|
|
|
|
|
|
try {
|
2020-08-02 14:33:41 +00:00
|
|
|
const res = await bucket.getObject(bucketPath);
|
2020-10-14 00:52:02 +00:00
|
|
|
if (!res || res.body.byteLength === 0) return undefined;
|
|
|
|
|
2020-08-02 14:33:41 +00:00
|
|
|
return {
|
|
|
|
etag: res.etag,
|
|
|
|
data: res.body,
|
|
|
|
};
|
2020-07-31 18:16:12 +00:00
|
|
|
} catch (err) {
|
|
|
|
const msg = err.message as string;
|
|
|
|
if (msg.indexOf("404") >= 0) return null;
|
|
|
|
throw err;
|
|
|
|
}
|
|
|
|
}
|