mirror of
https://github.com/LukeHagar/vercel.git
synced 2025-12-11 12:57:46 +00:00
Compare commits
11 Commits
vercel-plu
...
@vercel/py
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
d3ef240f6e | ||
|
|
5b26ebc7b8 | ||
|
|
3427ad6ce0 | ||
|
|
4ab5e4326b | ||
|
|
d24a3ce3ab | ||
|
|
29a44db8d9 | ||
|
|
695f3a9212 | ||
|
|
3ff777b8ed | ||
|
|
d94b9806ab | ||
|
|
35c8fc2729 | ||
|
|
0a468fd6d7 |
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@vercel/build-utils",
|
||||
"version": "2.12.3-canary.40",
|
||||
"version": "2.12.3-canary.42",
|
||||
"license": "MIT",
|
||||
"main": "./dist/index.js",
|
||||
"types": "./dist/index.d.js",
|
||||
|
||||
@@ -100,8 +100,6 @@ export function convertRuntimeToPlugin(
|
||||
|
||||
await fs.ensureDir(traceDir);
|
||||
|
||||
let newPathsRuntime: Set<string> = new Set();
|
||||
|
||||
const entryRoot = join(outputPath, 'server', 'pages');
|
||||
|
||||
for (const entrypoint of Object.keys(entrypoints)) {
|
||||
@@ -118,17 +116,6 @@ export function convertRuntimeToPlugin(
|
||||
},
|
||||
});
|
||||
|
||||
// Legacy Runtimes tend to pollute the `workPath` with compiled results,
|
||||
// because the `workPath` used to be a place that was a place where they could
|
||||
// just put anything, but nowadays it's the working directory of the `vercel build`
|
||||
// command, which is the place where the developer keeps their source files,
|
||||
// so we don't want to pollute this space unnecessarily. That means we have to clean
|
||||
// up files that were created by the build, which is done further below.
|
||||
const sourceFilesAfterBuild = await getSourceFiles(
|
||||
workPath,
|
||||
ignoreFilter
|
||||
);
|
||||
|
||||
// @ts-ignore This symbol is a private API
|
||||
const lambdaFiles: Files = output[FILES_SYMBOL];
|
||||
|
||||
@@ -173,12 +160,9 @@ export function convertRuntimeToPlugin(
|
||||
const entryPath = join(dirname(entrypoint), entryBase);
|
||||
const entry = join(entryRoot, entryPath);
|
||||
|
||||
// We never want to link here, only copy, because the launcher
|
||||
// file often has the same name for every entrypoint, which means that
|
||||
// every build for every entrypoint overwrites the launcher of the previous
|
||||
// one, so linking would end with a broken reference.
|
||||
// Create the parent directory of the API Route that will be created
|
||||
// for the current entrypoint inside of `.output/server/pages/api`.
|
||||
await fs.ensureDir(dirname(entry));
|
||||
await fs.copy(handlerFile.fsPath, entry);
|
||||
|
||||
// For compiled languages, the launcher file will be binary and therefore
|
||||
// won't try to import a user-provided request handler (instead, it will
|
||||
@@ -199,8 +183,13 @@ export function convertRuntimeToPlugin(
|
||||
let handlerContent = await fs.readFile(fsPath, encoding);
|
||||
|
||||
const importPaths = [
|
||||
// This is the full entrypoint path, like `./api/test.py`
|
||||
`./${entrypoint}`,
|
||||
// This is the full entrypoint path, like `./api/test.py`. In our tests
|
||||
// Python didn't support importing from a parent directory without using different
|
||||
// code in the launcher that registers it as a location for modules and then changing
|
||||
// the importing syntax, but continuing to import it like before seems to work. If
|
||||
// other languages need this, we should consider excluding Python explicitly.
|
||||
// `./${entrypoint}`,
|
||||
|
||||
// This is the entrypoint path without extension, like `api/test`
|
||||
entrypoint.slice(0, -ext.length),
|
||||
];
|
||||
@@ -244,57 +233,15 @@ export function convertRuntimeToPlugin(
|
||||
await fs.copy(handlerFile.fsPath, entry);
|
||||
}
|
||||
|
||||
const newFilesEntrypoint: Array<string> = [];
|
||||
const newDirectoriesEntrypoint: Array<string> = [];
|
||||
|
||||
const preBuildFiles = Object.values(sourceFilesPreBuild).map(file => {
|
||||
return file.fsPath;
|
||||
});
|
||||
|
||||
// Generate a list of directories and files that weren't present
|
||||
// before the entrypoint was processed by the Legacy Runtime, so
|
||||
// that we can perform a cleanup later. We need to divide into files
|
||||
// and directories because only cleaning up files might leave empty
|
||||
// directories, and listing directories separately also speeds up the
|
||||
// build because we can just delete them, which wipes all of their nested
|
||||
// paths, instead of iterating through all files that should be deleted.
|
||||
for (const file in sourceFilesAfterBuild) {
|
||||
if (!sourceFilesPreBuild[file]) {
|
||||
const path = sourceFilesAfterBuild[file].fsPath;
|
||||
const dirPath = dirname(path);
|
||||
|
||||
// If none of the files that were present before the entrypoint
|
||||
// was processed are contained within the directory we're looking
|
||||
// at right now, then we know it's a newly added directory
|
||||
// and it can therefore be removed later on.
|
||||
const isNewDir = !preBuildFiles.some(filePath => {
|
||||
return dirname(filePath).startsWith(dirPath);
|
||||
});
|
||||
|
||||
// Check out the list of tracked directories that were
|
||||
// newly added and see if one of them contains the path
|
||||
// we're looking at.
|
||||
const hasParentDir = newDirectoriesEntrypoint.some(dir => {
|
||||
return path.startsWith(dir);
|
||||
});
|
||||
|
||||
// If we have already tracked a directory that was newly
|
||||
// added that sits above the file or directory that we're
|
||||
// looking at, we don't need to add more entries to the list
|
||||
// because when the parent will get removed in the future,
|
||||
// all of its children (and therefore the path we're looking at)
|
||||
// will automatically get removed anyways.
|
||||
if (hasParentDir) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if (isNewDir) {
|
||||
newDirectoriesEntrypoint.push(dirPath);
|
||||
} else {
|
||||
newFilesEntrypoint.push(path);
|
||||
}
|
||||
}
|
||||
}
|
||||
// Legacy Runtimes based on interpreted languages will create a new launcher file
|
||||
// for every entrypoint, but they will create each one inside `workPath`, which means that
|
||||
// the launcher for one entrypoint will overwrite the launcher provided for the previous
|
||||
// entrypoint. That's why, above, we copy the file contents into the new destination (and
|
||||
// optionally transform them along the way), instead of linking. We then also want to remove
|
||||
// the copy origin right here, so that the `workPath` doesn't contain a useless launcher file
|
||||
// once the build has finished running.
|
||||
await fs.remove(handlerFile.fsPath);
|
||||
debug(`Removed temporary file "${handlerFile.fsPath}"`);
|
||||
|
||||
const nft = `${entry}.nft.json`;
|
||||
|
||||
@@ -320,18 +267,8 @@ export function convertRuntimeToPlugin(
|
||||
.filter(Boolean),
|
||||
});
|
||||
|
||||
await fs.ensureDir(dirname(nft));
|
||||
await fs.writeFile(nft, json);
|
||||
|
||||
// Extend the list of directories and files that were created by the
|
||||
// Legacy Runtime with the list of directories and files that were
|
||||
// created for the entrypoint that was just processed above.
|
||||
newPathsRuntime = new Set([
|
||||
...newPathsRuntime,
|
||||
...newFilesEntrypoint,
|
||||
...newDirectoriesEntrypoint,
|
||||
]);
|
||||
|
||||
// Add an entry that will later on be added to the `functions-manifest.json`
|
||||
// file that is placed inside of the `.output` directory.
|
||||
pages[normalizePath(entryPath)] = {
|
||||
@@ -348,20 +285,6 @@ export function convertRuntimeToPlugin(
|
||||
};
|
||||
}
|
||||
|
||||
// A list of all the files that were created by the Legacy Runtime,
|
||||
// which we'd like to remove from the File System.
|
||||
const toRemove = Array.from(newPathsRuntime).map(path => {
|
||||
debug(`Removing ${path} as part of cleanup`);
|
||||
return fs.remove(path);
|
||||
});
|
||||
|
||||
// Once all the entrypoints have been processed, we'd like to
|
||||
// remove all the files from `workPath` that originally weren't present
|
||||
// before the Legacy Runtime began running, because the `workPath`
|
||||
// is nowadays the directory in which the user keeps their source code, since
|
||||
// we're no longer running separate parallel builds for every Legacy Runtime.
|
||||
await Promise.all(toRemove);
|
||||
|
||||
// Add any Serverless Functions that were exposed by the Legacy Runtime
|
||||
// to the `functions-manifest.json` file provided in `.output`.
|
||||
await updateFunctionsManifest({ workPath, pages });
|
||||
|
||||
@@ -34,7 +34,7 @@ Finally, [connect your Git repository to Vercel](https://vercel.com/docs/git) an
|
||||
|
||||
## Documentation
|
||||
|
||||
For details on how to use Vercel CLI, check out our [documentation](https://vercel.com/docs).
|
||||
For details on how to use Vercel CLI, check out our [documentation](https://vercel.com/docs/cli).
|
||||
|
||||
## Local Development
|
||||
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "vercel",
|
||||
"version": "23.1.3-canary.63",
|
||||
"version": "23.1.3-canary.67",
|
||||
"preferGlobal": true,
|
||||
"license": "Apache-2.0",
|
||||
"description": "The command-line interface for Vercel",
|
||||
@@ -43,14 +43,14 @@
|
||||
"node": ">= 12"
|
||||
},
|
||||
"dependencies": {
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@vercel/go": "1.2.4-canary.4",
|
||||
"@vercel/node": "1.12.2-canary.7",
|
||||
"@vercel/python": "2.1.2-canary.1",
|
||||
"@vercel/python": "2.1.2-canary.2",
|
||||
"@vercel/ruby": "1.2.10-canary.0",
|
||||
"update-notifier": "4.1.0",
|
||||
"vercel-plugin-middleware": "0.0.0-canary.16",
|
||||
"vercel-plugin-node": "1.12.2-canary.32"
|
||||
"vercel-plugin-middleware": "0.0.0-canary.19",
|
||||
"vercel-plugin-node": "1.12.2-canary.34"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@next/env": "11.1.2",
|
||||
|
||||
@@ -447,6 +447,7 @@ export default async (client: Client) => {
|
||||
forceNew: argv['--force'],
|
||||
withCache: argv['--with-cache'],
|
||||
prebuilt: argv['--prebuilt'],
|
||||
rootDirectory,
|
||||
quiet,
|
||||
wantsPublic: argv['--public'] || localConfig.public,
|
||||
isFile,
|
||||
|
||||
@@ -52,6 +52,7 @@ export default async function processDeployment({
|
||||
isSettingUpProject: boolean;
|
||||
skipAutoDetectionConfirmation?: boolean;
|
||||
cwd?: string;
|
||||
rootDirectory?: string;
|
||||
}) {
|
||||
let {
|
||||
now,
|
||||
@@ -64,6 +65,7 @@ export default async function processDeployment({
|
||||
nowConfig,
|
||||
quiet,
|
||||
prebuilt,
|
||||
rootDirectory,
|
||||
} = args;
|
||||
|
||||
const { debug } = output;
|
||||
@@ -86,6 +88,7 @@ export default async function processDeployment({
|
||||
force,
|
||||
withCache,
|
||||
prebuilt,
|
||||
rootDirectory,
|
||||
skipAutoDetectionConfirmation,
|
||||
};
|
||||
|
||||
|
||||
@@ -37,6 +37,7 @@ export interface CreateOptions {
|
||||
project?: string;
|
||||
wantsPublic: boolean;
|
||||
prebuilt?: boolean;
|
||||
rootDirectory?: string;
|
||||
meta: Dictionary<string>;
|
||||
regions?: string[];
|
||||
quiet?: boolean;
|
||||
@@ -113,6 +114,7 @@ export default class Now extends EventEmitter {
|
||||
name,
|
||||
project,
|
||||
prebuilt = false,
|
||||
rootDirectory,
|
||||
wantsPublic,
|
||||
meta,
|
||||
regions,
|
||||
@@ -168,6 +170,7 @@ export default class Now extends EventEmitter {
|
||||
skipAutoDetectionConfirmation,
|
||||
cwd,
|
||||
prebuilt,
|
||||
rootDirectory,
|
||||
});
|
||||
|
||||
if (deployment && deployment.warnings) {
|
||||
|
||||
2
packages/client/.gitignore
vendored
2
packages/client/.gitignore
vendored
@@ -6,3 +6,5 @@ node_modules
|
||||
!tests/fixtures/nowignore/node_modules
|
||||
!tests/fixtures/vercelignore-allow-nodemodules/node_modules
|
||||
!tests/fixtures/vercelignore-allow-nodemodules/sub/node_modules
|
||||
!tests/fixtures/file-system-api/.output
|
||||
!tests/fixtures/file-system-api-root-directory/**/.output
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@vercel/client",
|
||||
"version": "10.2.3-canary.41",
|
||||
"version": "10.2.3-canary.45",
|
||||
"main": "dist/index.js",
|
||||
"typings": "dist/index.d.ts",
|
||||
"homepage": "https://vercel.com",
|
||||
@@ -40,7 +40,7 @@
|
||||
]
|
||||
},
|
||||
"dependencies": {
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@zeit/fetch": "5.2.0",
|
||||
"async-retry": "1.2.3",
|
||||
"async-sema": "3.0.0",
|
||||
|
||||
@@ -1,12 +1,12 @@
|
||||
import { lstatSync } from 'fs-extra';
|
||||
|
||||
import { relative, isAbsolute } from 'path';
|
||||
import hashes, { mapToObject } from './utils/hashes';
|
||||
import { hashes, mapToObject, resolveNftJsonFiles } from './utils/hashes';
|
||||
import { upload } from './upload';
|
||||
import { buildFileTree, createDebug, parseVercelConfig } from './utils';
|
||||
import { DeploymentError } from './errors';
|
||||
import {
|
||||
NowConfig,
|
||||
VercelConfig,
|
||||
VercelClientOptions,
|
||||
DeploymentOptions,
|
||||
DeploymentEventType,
|
||||
@@ -16,7 +16,7 @@ export default function buildCreateDeployment() {
|
||||
return async function* createDeployment(
|
||||
clientOptions: VercelClientOptions,
|
||||
deploymentOptions: DeploymentOptions = {},
|
||||
nowConfig: NowConfig = {}
|
||||
nowConfig: VercelConfig = {}
|
||||
): AsyncIterableIterator<{ type: DeploymentEventType; payload: any }> {
|
||||
const { path } = clientOptions;
|
||||
|
||||
@@ -74,12 +74,7 @@ export default function buildCreateDeployment() {
|
||||
debug(`Provided 'path' is a single file`);
|
||||
}
|
||||
|
||||
let { fileList } = await buildFileTree(
|
||||
path,
|
||||
clientOptions.isDirectory,
|
||||
debug,
|
||||
clientOptions.prebuilt
|
||||
);
|
||||
let { fileList } = await buildFileTree(path, clientOptions, debug);
|
||||
|
||||
let configPath: string | undefined;
|
||||
if (!nowConfig) {
|
||||
@@ -114,7 +109,11 @@ export default function buildCreateDeployment() {
|
||||
};
|
||||
}
|
||||
|
||||
const files = await hashes(fileList);
|
||||
const hashedFileMap = await hashes(fileList);
|
||||
const nftFileList = clientOptions.prebuilt
|
||||
? await resolveNftJsonFiles(hashedFileMap)
|
||||
: [];
|
||||
const files = await hashes(nftFileList, hashedFileMap);
|
||||
|
||||
debug(`Yielding a 'hashes-calculated' event with ${files.size} hashes`);
|
||||
yield { type: 'hashes-calculated', payload: mapToObject(files) };
|
||||
|
||||
@@ -15,6 +15,7 @@ export interface VercelClientOptions {
|
||||
apiUrl?: string;
|
||||
force?: boolean;
|
||||
prebuilt?: boolean;
|
||||
rootDirectory?: string;
|
||||
withCache?: boolean;
|
||||
userAgent?: string;
|
||||
defaultName?: string;
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
import { createHash } from 'crypto';
|
||||
import fs from 'fs-extra';
|
||||
import { Sema } from 'async-sema';
|
||||
import { join, dirname } from 'path';
|
||||
|
||||
export interface DeploymentFile {
|
||||
names: string[];
|
||||
@@ -15,9 +16,7 @@ export interface DeploymentFile {
|
||||
* @return {String} hex digest
|
||||
*/
|
||||
function hash(buf: Buffer): string {
|
||||
return createHash('sha1')
|
||||
.update(buf)
|
||||
.digest('hex');
|
||||
return createHash('sha1').update(buf).digest('hex');
|
||||
}
|
||||
|
||||
/**
|
||||
@@ -39,34 +38,68 @@ export const mapToObject = (
|
||||
/**
|
||||
* Computes hashes for the contents of each file given.
|
||||
*
|
||||
* @param {Array} of {String} full paths
|
||||
* @return {Map}
|
||||
* @param files - absolute file paths
|
||||
* @param map - optional map of files to append
|
||||
* @return Map of hash digest to file object
|
||||
*/
|
||||
async function hashes(files: string[]): Promise<Map<string, DeploymentFile>> {
|
||||
const map = new Map<string, DeploymentFile>();
|
||||
export async function hashes(
|
||||
files: string[],
|
||||
map = new Map<string, DeploymentFile>()
|
||||
): Promise<Map<string, DeploymentFile>> {
|
||||
const semaphore = new Sema(100);
|
||||
|
||||
await Promise.all(
|
||||
files.map(
|
||||
async (name: string): Promise<void> => {
|
||||
await semaphore.acquire();
|
||||
const data = await fs.readFile(name);
|
||||
const { mode } = await fs.stat(name);
|
||||
files.map(async (name: string): Promise<void> => {
|
||||
await semaphore.acquire();
|
||||
const data = await fs.readFile(name);
|
||||
const { mode } = await fs.stat(name);
|
||||
|
||||
const h = hash(data);
|
||||
const entry = map.get(h);
|
||||
const h = hash(data);
|
||||
const entry = map.get(h);
|
||||
|
||||
if (entry) {
|
||||
if (entry) {
|
||||
if (entry.names[0] !== name) {
|
||||
entry.names.push(name);
|
||||
} else {
|
||||
map.set(h, { names: [name], data, mode });
|
||||
}
|
||||
|
||||
semaphore.release();
|
||||
} else {
|
||||
map.set(h, { names: [name], data, mode });
|
||||
}
|
||||
)
|
||||
|
||||
semaphore.release();
|
||||
})
|
||||
);
|
||||
return map;
|
||||
}
|
||||
|
||||
export default hashes;
|
||||
export async function resolveNftJsonFiles(
|
||||
hashedFiles: Map<string, DeploymentFile>
|
||||
): Promise<string[]> {
|
||||
const semaphore = new Sema(100);
|
||||
const existingFiles = Array.from(hashedFiles.values());
|
||||
const resolvedFiles = new Set<string>();
|
||||
|
||||
await Promise.all(
|
||||
existingFiles.map(async file => {
|
||||
await semaphore.acquire();
|
||||
const fsPath = file.names[0];
|
||||
if (fsPath.endsWith('.nft.json')) {
|
||||
const json = file.data.toString('utf8');
|
||||
const { version, files } = JSON.parse(json) as {
|
||||
version: number;
|
||||
files: string[] | { input: string; output: string }[];
|
||||
};
|
||||
if (version === 1 || version === 2) {
|
||||
for (let f of files) {
|
||||
const relPath = typeof f === 'string' ? f : f.input;
|
||||
resolvedFiles.add(join(dirname(fsPath), relPath));
|
||||
}
|
||||
} else {
|
||||
console.error(`Invalid nft.json version: ${version}`);
|
||||
}
|
||||
}
|
||||
semaphore.release();
|
||||
})
|
||||
);
|
||||
|
||||
return Array.from(resolvedFiles);
|
||||
}
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
import { DeploymentFile } from './hashes';
|
||||
import { FetchOptions } from '@zeit/fetch';
|
||||
import { nodeFetch, zeitFetch } from './fetch';
|
||||
import { join, sep, relative } from 'path';
|
||||
import { join, sep, relative, posix } from 'path';
|
||||
import { URL } from 'url';
|
||||
import ignore from 'ignore';
|
||||
type Ignore = ReturnType<typeof ignore>;
|
||||
@@ -81,13 +81,16 @@ const maybeRead = async function <T>(path: string, default_: T) {
|
||||
|
||||
export async function buildFileTree(
|
||||
path: string | string[],
|
||||
isDirectory: boolean,
|
||||
debug: Debug,
|
||||
prebuilt?: boolean
|
||||
{
|
||||
isDirectory,
|
||||
prebuilt,
|
||||
rootDirectory,
|
||||
}: Pick<VercelClientOptions, 'isDirectory' | 'prebuilt' | 'rootDirectory'>,
|
||||
debug: Debug
|
||||
): Promise<{ fileList: string[]; ignoreList: string[] }> {
|
||||
const ignoreList: string[] = [];
|
||||
let fileList: string[];
|
||||
let { ig, ignores } = await getVercelIgnore(path, prebuilt);
|
||||
let { ig, ignores } = await getVercelIgnore(path, prebuilt, rootDirectory);
|
||||
|
||||
debug(`Found ${ignores.length} rules in .vercelignore`);
|
||||
debug('Building file tree...');
|
||||
@@ -119,37 +122,50 @@ export async function buildFileTree(
|
||||
|
||||
export async function getVercelIgnore(
|
||||
cwd: string | string[],
|
||||
prebuilt?: boolean
|
||||
prebuilt?: boolean,
|
||||
rootDirectory?: string
|
||||
): Promise<{ ig: Ignore; ignores: string[] }> {
|
||||
const ignores: string[] = prebuilt
|
||||
? ['*', '!.output', '!.output/**']
|
||||
: [
|
||||
'.hg',
|
||||
'.git',
|
||||
'.gitmodules',
|
||||
'.svn',
|
||||
'.cache',
|
||||
'.next',
|
||||
'.now',
|
||||
'.vercel',
|
||||
'.npmignore',
|
||||
'.dockerignore',
|
||||
'.gitignore',
|
||||
'.*.swp',
|
||||
'.DS_Store',
|
||||
'.wafpicke-*',
|
||||
'.lock-wscript',
|
||||
'.env.local',
|
||||
'.env.*.local',
|
||||
'.venv',
|
||||
'npm-debug.log',
|
||||
'config.gypi',
|
||||
'node_modules',
|
||||
'__pycache__',
|
||||
'venv',
|
||||
'CVS',
|
||||
'.output',
|
||||
];
|
||||
let ignores: string[] = [];
|
||||
|
||||
const outputDir = posix.join(rootDirectory || '', '.output');
|
||||
|
||||
if (prebuilt) {
|
||||
ignores.push('*');
|
||||
const parts = outputDir.split('/');
|
||||
parts.forEach((_, i) => {
|
||||
const level = parts.slice(0, i + 1).join('/');
|
||||
ignores.push(`!${level}`);
|
||||
});
|
||||
ignores.push(`!${outputDir}/**`);
|
||||
} else {
|
||||
ignores = [
|
||||
'.hg',
|
||||
'.git',
|
||||
'.gitmodules',
|
||||
'.svn',
|
||||
'.cache',
|
||||
'.next',
|
||||
'.now',
|
||||
'.vercel',
|
||||
'.npmignore',
|
||||
'.dockerignore',
|
||||
'.gitignore',
|
||||
'.*.swp',
|
||||
'.DS_Store',
|
||||
'.wafpicke-*',
|
||||
'.lock-wscript',
|
||||
'.env.local',
|
||||
'.env.*.local',
|
||||
'.venv',
|
||||
'npm-debug.log',
|
||||
'config.gypi',
|
||||
'node_modules',
|
||||
'__pycache__',
|
||||
'venv',
|
||||
'CVS',
|
||||
`.output`,
|
||||
];
|
||||
}
|
||||
const cwds = Array.isArray(cwd) ? cwd : [cwd];
|
||||
|
||||
const files = await Promise.all(
|
||||
@@ -250,39 +266,31 @@ export const prepareFiles = (
|
||||
files: Map<string, DeploymentFile>,
|
||||
clientOptions: VercelClientOptions
|
||||
): PreparedFile[] => {
|
||||
const preparedFiles = [...files.keys()].reduce(
|
||||
(acc: PreparedFile[], sha: string): PreparedFile[] => {
|
||||
const next = [...acc];
|
||||
const preparedFiles: PreparedFile[] = [];
|
||||
for (const [sha, file] of files) {
|
||||
for (const name of file.names) {
|
||||
let fileName: string;
|
||||
|
||||
const file = files.get(sha) as DeploymentFile;
|
||||
|
||||
for (const name of file.names) {
|
||||
let fileName: string;
|
||||
|
||||
if (clientOptions.isDirectory) {
|
||||
// Directory
|
||||
fileName =
|
||||
typeof clientOptions.path === 'string'
|
||||
? relative(clientOptions.path, name)
|
||||
: name;
|
||||
} else {
|
||||
// Array of files or single file
|
||||
const segments = name.split(sep);
|
||||
fileName = segments[segments.length - 1];
|
||||
}
|
||||
|
||||
next.push({
|
||||
file: isWin ? fileName.replace(/\\/g, '/') : fileName,
|
||||
size: file.data.byteLength || file.data.length,
|
||||
mode: file.mode,
|
||||
sha,
|
||||
});
|
||||
if (clientOptions.isDirectory) {
|
||||
// Directory
|
||||
fileName =
|
||||
typeof clientOptions.path === 'string'
|
||||
? relative(clientOptions.path, name)
|
||||
: name;
|
||||
} else {
|
||||
// Array of files or single file
|
||||
const segments = name.split(sep);
|
||||
fileName = segments[segments.length - 1];
|
||||
}
|
||||
|
||||
return next;
|
||||
},
|
||||
[]
|
||||
);
|
||||
preparedFiles.push({
|
||||
file: isWin ? fileName.replace(/\\/g, '/') : fileName,
|
||||
size: file.data.byteLength || file.data.length,
|
||||
mode: file.mode,
|
||||
sha,
|
||||
});
|
||||
}
|
||||
}
|
||||
|
||||
return preparedFiles;
|
||||
};
|
||||
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/foo.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/foo.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
foo
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/root/.output/baz.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/root/.output/baz.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
baz
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/root/.output/sub/qux.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/root/.output/sub/qux.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
qux
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/root/bar.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/root/bar.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
bar
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/someother/.output/baz.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/someother/.output/baz.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
baz
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/someother/.output/sub/qux.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/someother/.output/sub/qux.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
qux
|
||||
1
packages/client/tests/fixtures/file-system-api-root-directory/someother/bar.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api-root-directory/someother/bar.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
bar
|
||||
1
packages/client/tests/fixtures/file-system-api/.output/baz.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api/.output/baz.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
baz
|
||||
1
packages/client/tests/fixtures/file-system-api/.output/sub/qux.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api/.output/sub/qux.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
qux
|
||||
1
packages/client/tests/fixtures/file-system-api/foo.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api/foo.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
foo
|
||||
1
packages/client/tests/fixtures/file-system-api/sub/bar.txt
vendored
Normal file
1
packages/client/tests/fixtures/file-system-api/sub/bar.txt
vendored
Normal file
@@ -0,0 +1 @@
|
||||
bar
|
||||
4
packages/client/tests/tsconfig.json
Normal file
4
packages/client/tests/tsconfig.json
Normal file
@@ -0,0 +1,4 @@
|
||||
{
|
||||
"extends": "../tsconfig.json",
|
||||
"include": ["*.test.ts"]
|
||||
}
|
||||
@@ -17,7 +17,11 @@ const toAbsolutePaths = (cwd: string, files: string[]) =>
|
||||
describe('buildFileTree()', () => {
|
||||
it('should exclude files using `.nowignore` blocklist', async () => {
|
||||
const cwd = fixture('nowignore');
|
||||
const { fileList, ignoreList } = await buildFileTree(cwd, true, noop);
|
||||
const { fileList, ignoreList } = await buildFileTree(
|
||||
cwd,
|
||||
{ isDirectory: true },
|
||||
noop
|
||||
);
|
||||
|
||||
const expectedFileList = toAbsolutePaths(cwd, ['.nowignore', 'index.txt']);
|
||||
expect(normalizeWindowsPaths(expectedFileList).sort()).toEqual(
|
||||
@@ -36,7 +40,11 @@ describe('buildFileTree()', () => {
|
||||
|
||||
it('should include the node_modules using `.vercelignore` allowlist', async () => {
|
||||
const cwd = fixture('vercelignore-allow-nodemodules');
|
||||
const { fileList, ignoreList } = await buildFileTree(cwd, true, noop);
|
||||
const { fileList, ignoreList } = await buildFileTree(
|
||||
cwd,
|
||||
{ isDirectory: true },
|
||||
noop
|
||||
);
|
||||
|
||||
const expected = toAbsolutePaths(cwd, [
|
||||
'node_modules/one.txt',
|
||||
@@ -54,4 +62,90 @@ describe('buildFileTree()', () => {
|
||||
normalizeWindowsPaths(ignoreList).sort()
|
||||
);
|
||||
});
|
||||
|
||||
it('should find root files but ignore .output files when prebuilt=false', async () => {
|
||||
const cwd = fixture('file-system-api');
|
||||
const { fileList, ignoreList } = await buildFileTree(
|
||||
cwd,
|
||||
{ isDirectory: true, prebuilt: false },
|
||||
noop
|
||||
);
|
||||
|
||||
const expectedFileList = toAbsolutePaths(cwd, ['foo.txt', 'sub/bar.txt']);
|
||||
expect(normalizeWindowsPaths(expectedFileList).sort()).toEqual(
|
||||
normalizeWindowsPaths(fileList).sort()
|
||||
);
|
||||
|
||||
const expectedIgnoreList = ['.output'];
|
||||
expect(normalizeWindowsPaths(expectedIgnoreList).sort()).toEqual(
|
||||
normalizeWindowsPaths(ignoreList).sort()
|
||||
);
|
||||
});
|
||||
|
||||
it('should find .output files but ignore other files when prebuilt=true', async () => {
|
||||
const cwd = fixture('file-system-api');
|
||||
const { fileList, ignoreList } = await buildFileTree(
|
||||
cwd,
|
||||
{ isDirectory: true, prebuilt: true },
|
||||
noop
|
||||
);
|
||||
|
||||
const expectedFileList = toAbsolutePaths(cwd, [
|
||||
'.output/baz.txt',
|
||||
'.output/sub/qux.txt',
|
||||
]);
|
||||
expect(normalizeWindowsPaths(expectedFileList).sort()).toEqual(
|
||||
normalizeWindowsPaths(fileList).sort()
|
||||
);
|
||||
|
||||
const expectedIgnoreList = ['foo.txt', 'sub'];
|
||||
expect(normalizeWindowsPaths(expectedIgnoreList).sort()).toEqual(
|
||||
normalizeWindowsPaths(ignoreList).sort()
|
||||
);
|
||||
});
|
||||
|
||||
it('should find root files but ignore all .output files when prebuilt=false and rootDirectory=root', async () => {
|
||||
const cwd = fixture('file-system-api-root-directory');
|
||||
const { fileList, ignoreList } = await buildFileTree(
|
||||
cwd,
|
||||
{ isDirectory: true, prebuilt: false, rootDirectory: 'root' },
|
||||
noop
|
||||
);
|
||||
|
||||
const expectedFileList = toAbsolutePaths(cwd, [
|
||||
'foo.txt',
|
||||
'root/bar.txt',
|
||||
'someother/bar.txt',
|
||||
]);
|
||||
expect(normalizeWindowsPaths(expectedFileList).sort()).toEqual(
|
||||
normalizeWindowsPaths(fileList).sort()
|
||||
);
|
||||
|
||||
const expectedIgnoreList = ['root/.output', 'someother/.output'];
|
||||
expect(normalizeWindowsPaths(expectedIgnoreList).sort()).toEqual(
|
||||
normalizeWindowsPaths(ignoreList).sort()
|
||||
);
|
||||
});
|
||||
|
||||
it('should find root/.output files but ignore other files when prebuilt=true and rootDirectory=root', async () => {
|
||||
const cwd = fixture('file-system-api-root-directory');
|
||||
const { fileList, ignoreList } = await buildFileTree(
|
||||
cwd,
|
||||
{ isDirectory: true, prebuilt: true, rootDirectory: 'root' },
|
||||
noop
|
||||
);
|
||||
|
||||
const expectedFileList = toAbsolutePaths(cwd, [
|
||||
'root/.output/baz.txt',
|
||||
'root/.output/sub/qux.txt',
|
||||
]);
|
||||
expect(normalizeWindowsPaths(expectedFileList).sort()).toEqual(
|
||||
normalizeWindowsPaths(fileList).sort()
|
||||
);
|
||||
|
||||
const expectedIgnoreList = ['foo.txt', 'root/bar.txt', 'someother'];
|
||||
expect(normalizeWindowsPaths(expectedIgnoreList).sort()).toEqual(
|
||||
normalizeWindowsPaths(ignoreList).sort()
|
||||
);
|
||||
});
|
||||
});
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "vercel-plugin-middleware",
|
||||
"version": "0.0.0-canary.16",
|
||||
"version": "0.0.0-canary.19",
|
||||
"license": "MIT",
|
||||
"main": "./dist/index",
|
||||
"homepage": "",
|
||||
@@ -30,7 +30,7 @@
|
||||
"@types/node-fetch": "^2",
|
||||
"@types/ua-parser-js": "0.7.36",
|
||||
"@types/uuid": "8.3.1",
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@vercel/ncc": "0.24.0",
|
||||
"cookie": "0.4.1",
|
||||
"formdata-node": "4.3.1",
|
||||
|
||||
52
packages/middleware/src/esbuild-plugins.ts
Normal file
52
packages/middleware/src/esbuild-plugins.ts
Normal file
@@ -0,0 +1,52 @@
|
||||
import path from 'path';
|
||||
import * as esbuild from 'esbuild';
|
||||
|
||||
const processInjectFile = `
|
||||
// envOverride is passed by esbuild plugin
|
||||
const env = envOverride
|
||||
function cwd() {
|
||||
return '/'
|
||||
}
|
||||
function chdir(dir) {
|
||||
throw new Error('process.chdir is not supported')
|
||||
}
|
||||
export const process = {
|
||||
argv: [],
|
||||
env,
|
||||
chdir,
|
||||
cwd,
|
||||
};
|
||||
`;
|
||||
|
||||
export function nodeProcessPolyfillPlugin({ env = {} } = {}): esbuild.Plugin {
|
||||
return {
|
||||
name: 'node-process-polyfill',
|
||||
setup({ initialOptions, onResolve, onLoad }) {
|
||||
onResolve({ filter: /_virtual-process-polyfill_\.js/ }, ({ path }) => {
|
||||
return {
|
||||
path,
|
||||
sideEffects: false,
|
||||
};
|
||||
});
|
||||
|
||||
onLoad({ filter: /_virtual-process-polyfill_\.js/ }, () => {
|
||||
const contents = `const envOverride = ${JSON.stringify(
|
||||
env
|
||||
)};\n${processInjectFile}`;
|
||||
return {
|
||||
loader: 'js',
|
||||
contents,
|
||||
};
|
||||
});
|
||||
|
||||
const polyfills = [
|
||||
path.resolve(__dirname, '_virtual-process-polyfill_.js'),
|
||||
];
|
||||
if (initialOptions.inject) {
|
||||
initialOptions.inject.push(...polyfills);
|
||||
} else {
|
||||
initialOptions.inject = [...polyfills];
|
||||
}
|
||||
},
|
||||
};
|
||||
}
|
||||
@@ -17,6 +17,7 @@ import {
|
||||
UrlWithParsedQuery,
|
||||
} from 'url';
|
||||
import { toNodeHeaders } from './websandbox/utils';
|
||||
import { nodeProcessPolyfillPlugin } from './esbuild-plugins';
|
||||
|
||||
const glob = util.promisify(libGlob);
|
||||
const SUPPORTED_EXTENSIONS = ['.js', '.ts'];
|
||||
@@ -80,6 +81,7 @@ export async function build({ workPath }: { workPath: string }) {
|
||||
banner: {
|
||||
js: '"use strict";',
|
||||
},
|
||||
plugins: [nodeProcessPolyfillPlugin({ env: process.env })],
|
||||
format: 'cjs',
|
||||
});
|
||||
// Create `_ENTRIES` wrapper
|
||||
|
||||
26
packages/middleware/test/build.test.ts
vendored
26
packages/middleware/test/build.test.ts
vendored
@@ -63,6 +63,32 @@ describe('build()', () => {
|
||||
).toEqual('1');
|
||||
});
|
||||
|
||||
it('should build simple middleware with env vars', async () => {
|
||||
const expectedEnvVar = 'expected-env-var';
|
||||
const fixture = join(__dirname, 'fixtures/env');
|
||||
process.env.ENV_VAR_SHOULD_BE_DEFINED = expectedEnvVar;
|
||||
await build({
|
||||
workPath: fixture,
|
||||
});
|
||||
// env var should be inlined in the output
|
||||
delete process.env.ENV_VAR_SHOULD_BE_DEFINED;
|
||||
|
||||
const outputFile = join(fixture, '.output/server/pages/_middleware.js');
|
||||
expect(await fsp.stat(outputFile)).toBeTruthy();
|
||||
|
||||
require(outputFile);
|
||||
//@ts-ignore
|
||||
const middleware = global._ENTRIES['middleware_pages/_middleware'].default;
|
||||
expect(typeof middleware).toStrictEqual('function');
|
||||
const handledResponse = await middleware({
|
||||
request: {},
|
||||
});
|
||||
expect(String(handledResponse.response.body)).toEqual(expectedEnvVar);
|
||||
expect(
|
||||
(handledResponse.response as Response).headers.get('x-middleware-next')
|
||||
).toEqual(null);
|
||||
});
|
||||
|
||||
it('should create a middleware that runs in strict mode', async () => {
|
||||
const { middleware } = await setupFixture('use-strict');
|
||||
const response = await middleware({
|
||||
|
||||
3
packages/middleware/test/fixtures/env/_middleware.js
vendored
Normal file
3
packages/middleware/test/fixtures/env/_middleware.js
vendored
Normal file
@@ -0,0 +1,3 @@
|
||||
export default req => {
|
||||
return new Response(process.env.ENV_VAR_SHOULD_BE_DEFINED);
|
||||
};
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"private": false,
|
||||
"name": "vercel-plugin-go",
|
||||
"version": "1.0.0-canary.28",
|
||||
"version": "1.0.0-canary.30",
|
||||
"main": "dist/index.js",
|
||||
"license": "MIT",
|
||||
"files": [
|
||||
@@ -17,7 +17,7 @@
|
||||
"prepublishOnly": "tsc"
|
||||
},
|
||||
"dependencies": {
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@vercel/go": "1.2.4-canary.4"
|
||||
},
|
||||
"devDependencies": {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "vercel-plugin-node",
|
||||
"version": "1.12.2-canary.32",
|
||||
"version": "1.12.2-canary.34",
|
||||
"license": "MIT",
|
||||
"main": "./dist/index",
|
||||
"homepage": "https://vercel.com/docs/runtimes#official-runtimes/node-js",
|
||||
@@ -34,7 +34,7 @@
|
||||
"@types/node-fetch": "2",
|
||||
"@types/test-listen": "1.1.0",
|
||||
"@types/yazl": "2.4.2",
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@vercel/fun": "1.0.3",
|
||||
"@vercel/ncc": "0.24.0",
|
||||
"@vercel/nft": "0.14.0",
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"private": false,
|
||||
"name": "vercel-plugin-python",
|
||||
"version": "1.0.0-canary.29",
|
||||
"version": "1.0.0-canary.31",
|
||||
"main": "dist/index.js",
|
||||
"license": "MIT",
|
||||
"files": [
|
||||
@@ -17,8 +17,8 @@
|
||||
"prepublishOnly": "tsc"
|
||||
},
|
||||
"dependencies": {
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/python": "2.1.2-canary.1"
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@vercel/python": "2.1.2-canary.2"
|
||||
},
|
||||
"devDependencies": {
|
||||
"@types/node": "*",
|
||||
|
||||
@@ -1,7 +1,7 @@
|
||||
{
|
||||
"private": false,
|
||||
"name": "vercel-plugin-ruby",
|
||||
"version": "1.0.0-canary.28",
|
||||
"version": "1.0.0-canary.30",
|
||||
"main": "dist/index.js",
|
||||
"license": "MIT",
|
||||
"files": [
|
||||
@@ -17,7 +17,7 @@
|
||||
"prepublishOnly": "tsc"
|
||||
},
|
||||
"dependencies": {
|
||||
"@vercel/build-utils": "2.12.3-canary.40",
|
||||
"@vercel/build-utils": "2.12.3-canary.42",
|
||||
"@vercel/ruby": "1.2.10-canary.0"
|
||||
},
|
||||
"devDependencies": {
|
||||
|
||||
@@ -1,6 +1,6 @@
|
||||
{
|
||||
"name": "@vercel/python",
|
||||
"version": "2.1.2-canary.1",
|
||||
"version": "2.1.2-canary.2",
|
||||
"main": "./dist/index.js",
|
||||
"license": "MIT",
|
||||
"homepage": "https://vercel.com/docs/runtimes#official-runtimes/python",
|
||||
|
||||
@@ -1,4 +1,3 @@
|
||||
import { relative, basename } from 'path';
|
||||
import execa from 'execa';
|
||||
import { Meta, debug } from '@vercel/build-utils';
|
||||
|
||||
@@ -136,17 +135,10 @@ export async function installRequirementsFile({
|
||||
meta,
|
||||
args = [],
|
||||
}: InstallRequirementsFileArg) {
|
||||
const fileAtRoot = relative(workPath, filePath) === basename(filePath);
|
||||
|
||||
// If the `requirements.txt` file is located in the Root Directory of the project and
|
||||
// the new File System API is used (`avoidTopLevelInstall`), the Install Command
|
||||
// will have already installed its dependencies, so we don't need to do it again.
|
||||
if (meta.avoidTopLevelInstall && fileAtRoot) {
|
||||
debug(
|
||||
`Skipping requirements file installation, already installed by Install Command`
|
||||
);
|
||||
return;
|
||||
}
|
||||
// The Vercel platform already handles `requirements.txt` for frontend projects,
|
||||
// but the installation logic there is different, because it seems to install all
|
||||
// of the dependencies globally, whereas, for this Runtime, we want it to happen only
|
||||
// locally, so we'll run a separate installation.
|
||||
|
||||
if (
|
||||
meta.isDev &&
|
||||
|
||||
Reference in New Issue
Block a user