live-plugin-manager
Version:
Install and uninstall any node package at runtime from npm registry
227 lines (173 loc) • 6.01 kB
text/typescript
import * as path from "path";
import * as fs from "./fileSystem";
import {Headers, headersBasicAuth, headersTokenAuth, httpJsonGet} from "./httpUtils";
import Debug from "debug";
import { downloadTarball, extractTarball } from "./tarballUtils";
import { PackageJsonInfo } from "./PackageInfo";
const debug = Debug("live-plugin-manager.GithubRegistryClient");
export class GithubRegistryClient {
private headers: Headers;
constructor(auth?: GithubAuth) {
if (auth) {
debug(`Authenticating github api with ${auth.type}...`);
switch (auth.type) {
case "token":
this.headers = {
...headersTokenAuth(auth.token),
"user-agent": "live-plugin-manager"
};
break;
case "basic":
this.headers = {
...headersBasicAuth(auth.username, auth.password),
"user-agent": "live-plugin-manager"
};
break;
default:
throw new Error("Auth type not supported");
}
} else {
this.headers = {};
}
}
async get(repository: string): Promise<PackageJsonInfo> {
const repoInfo = extractRepositoryInfo(repository);
debug("Repository info: ", repoInfo);
const urlPkg
= `https://raw.githubusercontent.com/${repoInfo.owner}/${repoInfo.repo}/${repoInfo.ref}/package.json`;
const pkgContent = await httpJsonGet<PackageJsonInfo>(
urlPkg,
{...this.headers, accept: "application/vnd.github.v3+json"});
if (!pkgContent || !pkgContent.name || !pkgContent.version) {
throw new Error("Invalid plugin github repository " + repository);
}
const urlArchiveLink
= `https://api.github.com/repos/${repoInfo.owner}/${repoInfo.repo}/tarball/${repoInfo.ref}`;
pkgContent.dist = { tarball: urlArchiveLink };
return pkgContent;
}
async download(
destinationDirectory: string,
packageInfo: PackageJsonInfo): Promise<string> {
if (!packageInfo.dist || !packageInfo.dist.tarball) {
throw new Error("Invalid dist.tarball property");
}
const tgzFile = await downloadTarball(packageInfo.dist.tarball);
const pluginDirectory = path.join(destinationDirectory, packageInfo.name);
try {
await extractTarball(tgzFile, pluginDirectory);
} finally {
await fs.remove(tgzFile);
}
return pluginDirectory;
}
isGithubRepo(version: string): boolean {
return version.indexOf("/") > 0;
}
}
function extractRepositoryInfo(repository: string) {
const parts = repository.split("/");
if (parts.length !== 2) {
throw new Error("Invalid repository name");
}
const repoParts = parts[1].split("#");
const repoInfo = {
owner: parts[0],
repo: repoParts[0],
ref: repoParts[1] || "master"
};
return repoInfo;
}
export interface GithubAuthUserToken {
type: "token";
token: string;
}
export interface GithubAuthBasic {
type: "basic";
username: string;
password: string;
}
export type GithubAuth = GithubAuthUserToken
| GithubAuthBasic;
// | AuthOAuthToken
// | AuthOAuthSecret
// | AuthUserToken
// | AuthJWT;
// Implementation using github api
// no more used because the new version doesn't have an easy way to get the download link
// https://github.com/octokit/discussions/issues/12
// import * as path from "path";
// import * as fs from "./fileSystem";
// import * as GitHubApi from "github";
// import * as Debug from "debug";
// import { downloadTarball, extractTarball } from "./tarballUtils";
// import { PackageJsonInfo } from "./PackageInfo";
// const debug = Debug("live-plugin-manager.GithubRegistryClient");
// export class GithubRegistryClient {
// private readonly gitHubApi = new GitHubApi({followRedirects: false});
// constructor(auth?: GitHubApi.Auth) {
// if (auth) {
// debug(`Authenticating github api with ${auth.type}...`);
// this.gitHubApi.authenticate(auth);
// }
// }
// async get(repository: string): Promise<PackageJsonInfo> {
// const repoInfo = extractRepositoryInfo(repository);
// debug("Repository info: ", repoInfo);
// const response = await this.gitHubApi.repos.getContent({
// ...repoInfo,
// path: "package.json"
// });
// const contentBuff = new Buffer(response.data.content, "base64");
// const contentString = contentBuff.toString("utf-8");
// const pkgContent = JSON.parse(contentString) as PackageJsonInfo;
// if (!pkgContent.name || !pkgContent.version) {
// throw new Error("Invalid plugin github repository " + repository);
// }
// debug("Repository package info: ", pkgContent.name, pkgContent.version);
// // https://github.com/jashkenas/underscore/archive/master.zip
// // https://codeload.github.com/jashkenas/underscore/legacy.tar.gz/master
// const archiveLinkResponse = await this.gitHubApi.repos.getArchiveLink({
// ...repoInfo,
// archive_format: "tarball"
// });
// const archiveLink = archiveLinkResponse.meta.location;
// if (!(typeof archiveLink === "string")) {
// throw new Error("Invalid archive link");
// }
// debug("Repository package archive: ", archiveLink);
// pkgContent.dist = { tarball: archiveLink };
// return pkgContent;
// }
// async download(
// destinationDirectory: string,
// packageInfo: PackageJsonInfo): Promise<string> {
// if (!packageInfo.dist || !packageInfo.dist.tarball) {
// throw new Error("Invalid dist.tarball property");
// }
// const tgzFile = await downloadTarball(packageInfo.dist.tarball);
// const pluginDirectory = path.join(destinationDirectory, packageInfo.name);
// try {
// await extractTarball(tgzFile, pluginDirectory);
// } finally {
// await fs.remove(tgzFile);
// }
// return pluginDirectory;
// }
// isGithubRepo(version: string): boolean {
// return version.indexOf("/") > 0;
// }
// }
// function extractRepositoryInfo(repository: string) {
// const parts = repository.split("/");
// if (parts.length !== 2) {
// throw new Error("Invalid repository name");
// }
// const repoParts = parts[1].split("#");
// const repoInfo = {
// owner: parts[0],
// repo: repoParts[0],
// ref: repoParts[1] || "master"
// };
// return repoInfo;
// }