diff --git a/doc/api/fs.md b/doc/api/fs.md index 316b1b30111e06..ed3c94ca3258ab 100644 --- a/doc/api/fs.md +++ b/doc/api/fs.md @@ -1463,18 +1463,27 @@ changes: * `target` {string|Buffer|URL} * `path` {string|Buffer|URL} -* `type` {string} **Default:** `'file'` +* `type` {string|null} **Default:** `null` * Returns: {Promise} Fulfills with `undefined` upon success. Creates a symbolic link. The `type` argument is only used on Windows platforms and can be one of `'dir'`, -`'file'`, or `'junction'`. Windows junction points require the destination path -to be absolute. When using `'junction'`, the `target` argument will +`'file'`, or `'junction'`. If the `type` argument is not a string, Node.js will +autodetect `target` type and use `'file'` or `'dir'`. If the `target` does not +exist, `'file'` will be used. Windows junction points require the destination +path to be absolute. When using `'junction'`, the `target` argument will automatically be normalized to absolute path. ### `fsPromises.truncate(path[, len])` diff --git a/lib/internal/fs/promises.js b/lib/internal/fs/promises.js index 34fd0f586766dd..f718f7bff20306 100644 --- a/lib/internal/fs/promises.js +++ b/lib/internal/fs/promises.js @@ -116,6 +116,8 @@ const { const getDirectoryEntriesPromise = promisify(getDirents); const validateRmOptionsPromise = promisify(validateRmOptions); +const isWindows = process.platform === 'win32'; + let cpPromises; function lazyLoadCpPromises() { return cpPromises ??= require('internal/fs/cp/cp').cpFn; @@ -710,7 +712,16 @@ async function readlink(path, options) { } async function symlink(target, path, type_) { - const type = (typeof type_ === 'string' ? type_ : null); + let type = (typeof type_ === 'string' ? type_ : null); + if (isWindows && type === null) { + try { + const absoluteTarget = pathModule.resolve(`${path}`, '..', `${target}`); + type = (await stat(absoluteTarget)).isDirectory() ? 'dir' : 'file'; + } catch { + // Default to 'file' if path is invalid or file does not exist + type = 'file'; + } + } target = getValidatedPath(target, 'target'); path = getValidatedPath(path); return binding.symlink(preprocessSymlinkDestination(target, type, path), diff --git a/test/parallel/test-fs-symlink-dir.js b/test/parallel/test-fs-symlink-dir.js index 012d34c301e835..7ce3039d4aa150 100644 --- a/test/parallel/test-fs-symlink-dir.js +++ b/test/parallel/test-fs-symlink-dir.js @@ -12,6 +12,7 @@ if (!common.canCreateSymLink()) const assert = require('assert'); const path = require('path'); const fs = require('fs'); +const fsPromises = fs.promises; const tmpdir = require('../common/tmpdir'); tmpdir.refresh(); @@ -36,11 +37,18 @@ function testAsync(target, path) { })); } +async function testPromises(target, path) { + await fsPromises.symlink(target, path); + fs.readdirSync(path); +} + for (const linkTarget of linkTargets) { fs.mkdirSync(path.resolve(tmpdir.path, linkTarget)); for (const linkPath of linkPaths) { testSync(linkTarget, `${linkPath}-${path.basename(linkTarget)}-sync`); testAsync(linkTarget, `${linkPath}-${path.basename(linkTarget)}-async`); + testPromises(linkTarget, `${linkPath}-${path.basename(linkTarget)}-promises`) + .then(common.mustCall()); } } @@ -57,10 +65,17 @@ for (const linkTarget of linkTargets) { })); } + async function testPromises(target, path) { + await fsPromises.symlink(target, path); + assert(!fs.existsSync(path)); + } + for (const linkTarget of linkTargets.map((p) => p + '-broken')) { for (const linkPath of linkPaths) { testSync(linkTarget, `${linkPath}-${path.basename(linkTarget)}-sync`); testAsync(linkTarget, `${linkPath}-${path.basename(linkTarget)}-async`); + testPromises(linkTarget, `${linkPath}-${path.basename(linkTarget)}-promises`) + .then(common.mustCall()); } } }