UNPKG

@docker/actions-toolkit

Version:
199 lines 7.29 kB
/** * Copyright 2024 actions-toolkit authors * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ import fs from 'fs'; import os from 'os'; import gunzip from 'gunzip-maybe'; import * as path from 'path'; import * as tar from 'tar-stream'; import { IMAGE_BLOBS_DIR_V1, IMAGE_INDEX_FILE_V1, IMAGE_LAYOUT_FILE_V1 } from '../types/oci/layout.js'; import { MEDIATYPE_IMAGE_INDEX_V1, MEDIATYPE_IMAGE_MANIFEST_V1 } from '../types/oci/mediatype.js'; export class OCI { static defaultPlatform() { const nodePlatform = os.platform(); const nodeArch = os.arch(); const goosMap = { win32: 'windows', sunos: 'solaris' // others (linux, darwin, freebsd, openbsd, netbsd, aix, android) match Go already }; const goArchMap = { x64: 'amd64', ia32: '386', arm: 'arm', arm64: 'arm64', ppc64: 'ppc64le', s390x: 's390x', riscv64: 'riscv64', loong64: 'loong64', mips: 'mips', mipsel: 'mipsle', mips64: 'mips64', mips64el: 'mips64le' }; const goos = goosMap[nodePlatform] ?? nodePlatform; const goarch = goArchMap[nodeArch] ?? nodeArch; let variant; if (goarch === 'arm') { // eslint-disable-next-line @typescript-eslint/no-explicit-any const armVersionRaw = process.config.variables?.arm_version; const armVersion = Number(armVersionRaw); // Go only recognizes v5/v6/v7 for GOARM. Do not emit v8+ (that would be arm64). if ([5, 6, 7].includes(armVersion)) { variant = `v${armVersion}`; } } return { architecture: goarch, os: goos, variant: variant }; } static loadArchive(opts) { return new Promise((resolve, reject) => { const tarex = tar.extract(); let rootIndex; let rootLayout; const indexes = {}; const manifests = {}; const images = {}; const blobs = {}; tarex.on('entry', async (header, stream, next) => { if (header.type === 'file') { const filename = path.normalize(header.name); if (filename === IMAGE_INDEX_FILE_V1) { rootIndex = await OCI.streamToJson(stream); } else if (filename === IMAGE_LAYOUT_FILE_V1) { rootLayout = await OCI.streamToJson(stream); } else if (filename.startsWith(path.join(IMAGE_BLOBS_DIR_V1, path.sep))) { const blob = await OCI.extractBlob(stream); const digest = `${filename.split(path.sep)[1]}:${filename.split(path.sep)[filename.split(path.sep).length - 1]}`; if (OCI.isIndex(blob)) { indexes[digest] = JSON.parse(blob); } else if (OCI.isManifest(blob)) { manifests[digest] = JSON.parse(blob); } else if (OCI.isImage(blob)) { images[digest] = JSON.parse(blob); } else { blobs[digest] = blob; } } else { reject(new Error(`Invalid OCI archive: unexpected file ${filename}`)); } } stream.resume(); next(); }); tarex.on('finish', () => { if (!rootIndex || !rootLayout) { reject(new Error('Invalid OCI archive: missing index or layout')); } resolve({ root: { index: rootIndex, layout: rootLayout }, indexes: indexes, manifests: manifests, images: images, blobs: blobs }); }); tarex.on('error', error => { reject(error); }); fs.createReadStream(opts.file).pipe(gunzip()).pipe(tarex); }); } // eslint-disable-next-line @typescript-eslint/no-explicit-any static isIndex(blob) { try { const index = JSON.parse(blob); return index.mediaType === MEDIATYPE_IMAGE_INDEX_V1; } catch { return false; } } // eslint-disable-next-line @typescript-eslint/no-explicit-any static isManifest(blob) { try { const manifest = JSON.parse(blob); return manifest.mediaType === MEDIATYPE_IMAGE_MANIFEST_V1 && manifest.layers.length > 0; } catch { return false; } } // eslint-disable-next-line @typescript-eslint/no-explicit-any static isImage(blob) { try { const image = JSON.parse(blob); return image.rootfs.type !== ''; } catch { return false; } } // eslint-disable-next-line @typescript-eslint/no-explicit-any static extractBlob(stream) { return new Promise((resolve, reject) => { const chunks = []; const dstream = stream.pipe(gunzip()); dstream.on('data', chunk => { chunks.push(chunk); }); dstream.on('end', () => { resolve(Buffer.concat(chunks).toString('utf8')); }); dstream.on('error', async (error) => { reject(error); }); }); } static async streamToJson(stream) { return new Promise((resolve, reject) => { const chunks = []; let bytes = 0; stream.on('data', chunk => { bytes += chunk.length; if (bytes <= 2 * 1024 * 1024) { chunks.push(chunk.toString('utf8')); } else { reject(new Error('The data stream exceeds the size limit for JSON parsing.')); } }); stream.on('end', () => { try { resolve(JSON.parse(chunks.join(''))); } catch (error) { reject(error); } }); stream.on('error', async (error) => { reject(error); }); }); } } //# sourceMappingURL=oci.js.map