diff --git a/lib/fs.js b/lib/fs.js index da67f2742c..b90dcf9e94 100644 --- a/lib/fs.js +++ b/lib/fs.js @@ -16,9 +16,11 @@ const EventEmitter = require('events'); const FSReqWrap = binding.FSReqWrap; const FSEvent = process.binding('fs_event_wrap').FSEvent; const internalFS = require('internal/fs'); +const internalURL = require('internal/url'); const assertEncoding = internalFS.assertEncoding; const stringToFlags = internalFS.stringToFlags; const SyncWriteStream = internalFS.SyncWriteStream; +const getPathFromURL = internalURL.getPathFromURL; Object.defineProperty(exports, 'constants', { configurable: false, @@ -203,6 +205,16 @@ fs.Stats.prototype.isSocket = function() { }); }); +function handleError(val, callback) { + if (val instanceof Error) { + if (typeof callback === 'function') { + process.nextTick(callback, val); + return true; + } else throw val; + } + return false; +} + fs.access = function(path, mode, callback) { if (typeof mode === 'function') { callback = mode; @@ -211,6 +223,9 @@ fs.access = function(path, mode, callback) { throw new TypeError('"callback" argument must be a function'); } + if (handleError((path = getPathFromURL(path)), callback)) + return; + if (!nullCheck(path, callback)) return; @@ -221,6 +236,7 @@ fs.access = function(path, mode, callback) { }; fs.accessSync = function(path, mode) { + handleError((path = getPathFromURL(path))); nullCheck(path); if (mode === undefined) @@ -232,6 +248,8 @@ fs.accessSync = function(path, mode) { }; fs.exists = function(path, callback) { + if (handleError((path = getPathFromURL(path)), cb)) + return; if (!nullCheck(path, cb)) return; var req = new FSReqWrap(); req.oncomplete = cb; @@ -243,6 +261,7 @@ fs.exists = function(path, callback) { fs.existsSync = function(path) { try { + handleError((path = getPathFromURL(path))); nullCheck(path); binding.stat(pathModule._makeLong(path)); return true; @@ -255,6 +274,8 @@ fs.readFile = function(path, options, callback) { callback = maybeCallback(arguments[arguments.length - 1]); options = getOptions(options, { flag: 'r' }); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; @@ -541,6 +562,8 @@ fs.open = function(path, flags, mode, callback_) { var callback = makeCallback(arguments[arguments.length - 1]); mode = modeNum(mode, 0o666); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); @@ -554,6 +577,7 @@ fs.open = function(path, flags, mode, callback_) { fs.openSync = function(path, flags, mode) { mode = modeNum(mode, 0o666); + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.open(pathModule._makeLong(path), stringToFlags(flags), mode); }; @@ -727,6 +751,12 @@ fs.writeSync = function(fd, buffer, offset, length, position) { fs.rename = function(oldPath, newPath, callback) { callback = makeCallback(callback); + if (handleError((oldPath = getPathFromURL(oldPath)), callback)) + return; + + if (handleError((newPath = getPathFromURL(newPath)), callback)) + return; + if (!nullCheck(oldPath, callback)) return; if (!nullCheck(newPath, callback)) return; var req = new FSReqWrap(); @@ -737,6 +767,8 @@ fs.rename = function(oldPath, newPath, callback) { }; fs.renameSync = function(oldPath, newPath) { + handleError((oldPath = getPathFromURL(oldPath))); + handleError((newPath = getPathFromURL(newPath))); nullCheck(oldPath); nullCheck(newPath); return binding.rename(pathModule._makeLong(oldPath), @@ -808,6 +840,8 @@ fs.ftruncateSync = function(fd, len) { fs.rmdir = function(path, callback) { callback = maybeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -815,6 +849,7 @@ fs.rmdir = function(path, callback) { }; fs.rmdirSync = function(path) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.rmdir(pathModule._makeLong(path)); }; @@ -842,6 +877,8 @@ fs.fsyncSync = function(fd) { fs.mkdir = function(path, mode, callback) { if (typeof mode === 'function') callback = mode; callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -851,6 +888,7 @@ fs.mkdir = function(path, mode, callback) { }; fs.mkdirSync = function(path, mode) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.mkdir(pathModule._makeLong(path), modeNum(mode, 0o777)); @@ -859,6 +897,8 @@ fs.mkdirSync = function(path, mode) { fs.readdir = function(path, options, callback) { callback = makeCallback(typeof options === 'function' ? options : callback); options = getOptions(options, {}); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -867,6 +907,7 @@ fs.readdir = function(path, options, callback) { fs.readdirSync = function(path, options) { options = getOptions(options, {}); + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.readdir(pathModule._makeLong(path), options.encoding); }; @@ -879,6 +920,8 @@ fs.fstat = function(fd, callback) { fs.lstat = function(path, callback) { callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -887,6 +930,8 @@ fs.lstat = function(path, callback) { fs.stat = function(path, callback) { callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -898,11 +943,13 @@ fs.fstatSync = function(fd) { }; fs.lstatSync = function(path) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.lstat(pathModule._makeLong(path)); }; fs.statSync = function(path) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.stat(pathModule._makeLong(path)); }; @@ -910,6 +957,8 @@ fs.statSync = function(path) { fs.readlink = function(path, options, callback) { callback = makeCallback(typeof options === 'function' ? options : callback); options = getOptions(options, {}); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -918,6 +967,7 @@ fs.readlink = function(path, options, callback) { fs.readlinkSync = function(path, options) { options = getOptions(options, {}); + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.readlink(pathModule._makeLong(path), options.encoding); }; @@ -941,6 +991,12 @@ fs.symlink = function(target, path, type_, callback_) { var type = (typeof type_ === 'string' ? type_ : null); var callback = makeCallback(arguments[arguments.length - 1]); + if (handleError((target = getPathFromURL(target)), callback)) + return; + + if (handleError((path = getPathFromURL(path)), callback)) + return; + if (!nullCheck(target, callback)) return; if (!nullCheck(path, callback)) return; @@ -955,7 +1011,8 @@ fs.symlink = function(target, path, type_, callback_) { fs.symlinkSync = function(target, path, type) { type = (typeof type === 'string' ? type : null); - + handleError((target = getPathFromURL(target))); + handleError((path = getPathFromURL(path))); nullCheck(target); nullCheck(path); @@ -966,6 +1023,13 @@ fs.symlinkSync = function(target, path, type) { fs.link = function(existingPath, newPath, callback) { callback = makeCallback(callback); + + if (handleError((existingPath = getPathFromURL(existingPath)), callback)) + return; + + if (handleError((newPath = getPathFromURL(newPath)), callback)) + return; + if (!nullCheck(existingPath, callback)) return; if (!nullCheck(newPath, callback)) return; @@ -978,6 +1042,8 @@ fs.link = function(existingPath, newPath, callback) { }; fs.linkSync = function(existingPath, newPath) { + handleError((existingPath = getPathFromURL(existingPath))); + handleError((newPath = getPathFromURL(newPath))); nullCheck(existingPath); nullCheck(newPath); return binding.link(pathModule._makeLong(existingPath), @@ -986,6 +1052,8 @@ fs.linkSync = function(existingPath, newPath) { fs.unlink = function(path, callback) { callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -993,6 +1061,7 @@ fs.unlink = function(path, callback) { }; fs.unlinkSync = function(path) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.unlink(pathModule._makeLong(path)); }; @@ -1049,6 +1118,8 @@ if (constants.hasOwnProperty('O_SYMLINK')) { fs.chmod = function(path, mode, callback) { callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -1058,6 +1129,7 @@ fs.chmod = function(path, mode, callback) { }; fs.chmodSync = function(path, mode) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.chmod(pathModule._makeLong(path), modeNum(mode)); }; @@ -1092,6 +1164,8 @@ fs.fchownSync = function(fd, uid, gid) { fs.chown = function(path, uid, gid, callback) { callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -1099,6 +1173,7 @@ fs.chown = function(path, uid, gid, callback) { }; fs.chownSync = function(path, uid, gid) { + handleError((path = getPathFromURL(path))); nullCheck(path); return binding.chown(pathModule._makeLong(path), uid, gid); }; @@ -1126,6 +1201,8 @@ fs._toUnixTimestamp = toUnixTimestamp; fs.utimes = function(path, atime, mtime, callback) { callback = makeCallback(callback); + if (handleError((path = getPathFromURL(path)), callback)) + return; if (!nullCheck(path, callback)) return; var req = new FSReqWrap(); req.oncomplete = callback; @@ -1136,6 +1213,7 @@ fs.utimes = function(path, atime, mtime, callback) { }; fs.utimesSync = function(path, atime, mtime) { + handleError((path = getPathFromURL(path))); nullCheck(path); atime = toUnixTimestamp(atime); mtime = toUnixTimestamp(mtime); @@ -1296,6 +1374,7 @@ FSWatcher.prototype.start = function(filename, persistent, recursive, encoding) { + handleError((filename = getPathFromURL(filename))); nullCheck(filename); var err = this._handle.start(pathModule._makeLong(filename), persistent, @@ -1314,6 +1393,7 @@ FSWatcher.prototype.close = function() { }; fs.watch = function(filename, options, listener) { + handleError((filename = getPathFromURL(filename))); nullCheck(filename); if (typeof options === 'function') { @@ -1374,6 +1454,7 @@ util.inherits(StatWatcher, EventEmitter); StatWatcher.prototype.start = function(filename, persistent, interval) { + handleError((filename = getPathFromURL(filename))); nullCheck(filename); this._handle.start(pathModule._makeLong(filename), persistent, interval); }; @@ -1387,6 +1468,7 @@ StatWatcher.prototype.stop = function() { const statWatchers = new Map(); fs.watchFile = function(filename, options, listener) { + handleError((filename = getPathFromURL(filename))); nullCheck(filename); filename = pathModule.resolve(filename); var stat; @@ -1423,6 +1505,7 @@ fs.watchFile = function(filename, options, listener) { }; fs.unwatchFile = function(filename, listener) { + handleError((filename = getPathFromURL(filename))); nullCheck(filename); filename = pathModule.resolve(filename); var stat = statWatchers.get(filename); @@ -1466,6 +1549,7 @@ function encodeRealpathResult(result, options) { fs.realpathSync = function realpathSync(p, options) { options = getOptions(options, {}); + handleError((p = getPathFromURL(p))); nullCheck(p); p = p.toString('utf8'); @@ -1569,6 +1653,8 @@ fs.realpathSync = function realpathSync(p, options) { fs.realpath = function realpath(p, options, callback) { callback = maybeCallback(typeof options === 'function' ? options : callback); options = getOptions(options, {}); + if (handleError((p = getPathFromURL(p)), callback)) + return; if (!nullCheck(p, callback)) return; @@ -1727,7 +1813,7 @@ function ReadStream(path, options) { Readable.call(this, options); - this.path = path; + handleError((this.path = getPathFromURL(path))); this.fd = options.fd === undefined ? null : options.fd; this.flags = options.flags === undefined ? 'r' : options.flags; this.mode = options.mode === undefined ? 0o666 : options.mode; @@ -1890,7 +1976,7 @@ function WriteStream(path, options) { Writable.call(this, options); - this.path = path; + handleError((this.path = getPathFromURL(path))); this.fd = options.fd === undefined ? null : options.fd; this.flags = options.flags === undefined ? 'w' : options.flags; this.mode = options.mode === undefined ? 0o666 : options.mode; diff --git a/lib/internal/url.js b/lib/internal/url.js index af90d06617..b79cc23035 100644 --- a/lib/internal/url.js +++ b/lib/internal/url.js @@ -7,6 +7,9 @@ const cannotBeBase = Symbol('cannot-be-base'); const special = Symbol('special'); const searchParams = Symbol('query'); const querystring = require('querystring'); +const os = require('os'); + +const isWindows = process.platform === 'win32'; const kScheme = Symbol('scheme'); const kHost = Symbol('host'); @@ -1070,6 +1073,67 @@ function urlToOptions(url) { return options; } +function getPathFromURLWin32(url) { + var hostname = url.hostname; + var pathname = url.pathname; + for (var n = 0; n < pathname.length; n++) { + if (pathname[n] === '%') { + var third = pathname.codePointAt(n + 2) | 0x20; + if ((pathname[n + 1] === '2' && third === 102) || // 2f 2F / + (pathname[n + 1] === '5' && third === 99)) { // 5c 5C \ + return new TypeError( + 'Path must not include encoded \\ or / characters'); + } + } + } + pathname = decodeURIComponent(pathname); + if (hostname !== '') { + // If hostname is set, then we have a UNC path + // Pass the hostname through domainToUnicode just in case + // it is an IDN using punycode encoding. We do not need to worry + // about percent encoding because the URL parser will have + // already taken care of that for us. Note that this only + // causes IDNs with an appropriate `xn--` prefix to be decoded. + return `//${domainToUnicode(hostname)}${pathname}`; + } else { + // Otherwise, it's a local path that requires a drive letter + var letter = pathname.codePointAt(1) | 0x20; + var sep = pathname[2]; + if (letter < 97 || letter > 122 || // a..z A..Z + (sep !== ':')) { + return new TypeError('File URLs must specify absolute paths'); + } + return pathname.slice(1); + } +} + +function getPathFromURLPosix(url) { + if (url.hostname !== '') { + return new TypeError( + `File URLs on ${os.platform()} must use hostname 'localhost'` + + ' or not specify any hostname'); + } + var pathname = url.pathname; + for (var n = 0; n < pathname.length; n++) { + if (pathname[n] === '%') { + var third = pathname.codePointAt(n + 2) | 0x20; + if (pathname[n + 1] === '2' && third === 102) { + return new TypeError('Path must not include encoded / characters'); + } + } + } + return decodeURIComponent(pathname); +} + +function getPathFromURL(path) { + if (!(path instanceof URL)) + return path; + if (path.protocol !== 'file:') + return new TypeError('Only `file:` URLs are supported'); + return isWindows ? getPathFromURLWin32(path) : getPathFromURLPosix(path); +} + +exports.getPathFromURL = getPathFromURL; exports.URL = URL; exports.URLSearchParams = URLSearchParams; exports.domainToASCII = domainToASCII; diff --git a/test/parallel/test-fs-null-bytes.js b/test/parallel/test-fs-null-bytes.js index 996bf9ae26..c56c8ff7ce 100644 --- a/test/parallel/test-fs-null-bytes.js +++ b/test/parallel/test-fs-null-bytes.js @@ -2,6 +2,7 @@ const common = require('../common'); const assert = require('assert'); const fs = require('fs'); +const URL = require('url').URL; function check(async, sync) { const expected = /Path must be a string without null bytes/; @@ -48,6 +49,65 @@ check(null, fs.watch, 'foo\u0000bar', common.mustNotCall()); check(null, fs.watchFile, 'foo\u0000bar', common.mustNotCall()); check(fs.writeFile, fs.writeFileSync, 'foo\u0000bar'); +const fileUrl = new URL('file:///C:/foo\u0000bar'); +const fileUrl2 = new URL('file:///C:/foo%00bar'); + +check(fs.access, fs.accessSync, fileUrl); +check(fs.access, fs.accessSync, fileUrl, fs.F_OK); +check(fs.appendFile, fs.appendFileSync, fileUrl); +check(fs.chmod, fs.chmodSync, fileUrl, '0644'); +check(fs.chown, fs.chownSync, fileUrl, 12, 34); +check(fs.link, fs.linkSync, fileUrl, 'foobar'); +check(fs.link, fs.linkSync, 'foobar', fileUrl); +check(fs.lstat, fs.lstatSync, fileUrl); +check(fs.mkdir, fs.mkdirSync, fileUrl, '0755'); +check(fs.open, fs.openSync, fileUrl, 'r'); +check(fs.readFile, fs.readFileSync, fileUrl); +check(fs.readdir, fs.readdirSync, fileUrl); +check(fs.readlink, fs.readlinkSync, fileUrl); +check(fs.realpath, fs.realpathSync, fileUrl); +check(fs.rename, fs.renameSync, fileUrl, 'foobar'); +check(fs.rename, fs.renameSync, 'foobar', fileUrl); +check(fs.rmdir, fs.rmdirSync, fileUrl); +check(fs.stat, fs.statSync, fileUrl); +check(fs.symlink, fs.symlinkSync, fileUrl, 'foobar'); +check(fs.symlink, fs.symlinkSync, 'foobar', fileUrl); +check(fs.truncate, fs.truncateSync, fileUrl); +check(fs.unlink, fs.unlinkSync, fileUrl); +check(null, fs.unwatchFile, fileUrl, common.fail); +check(fs.utimes, fs.utimesSync, fileUrl, 0, 0); +check(null, fs.watch, fileUrl, common.fail); +check(null, fs.watchFile, fileUrl, common.fail); +check(fs.writeFile, fs.writeFileSync, fileUrl); + +check(fs.access, fs.accessSync, fileUrl2); +check(fs.access, fs.accessSync, fileUrl2, fs.F_OK); +check(fs.appendFile, fs.appendFileSync, fileUrl2); +check(fs.chmod, fs.chmodSync, fileUrl2, '0644'); +check(fs.chown, fs.chownSync, fileUrl2, 12, 34); +check(fs.link, fs.linkSync, fileUrl2, 'foobar'); +check(fs.link, fs.linkSync, 'foobar', fileUrl2); +check(fs.lstat, fs.lstatSync, fileUrl2); +check(fs.mkdir, fs.mkdirSync, fileUrl2, '0755'); +check(fs.open, fs.openSync, fileUrl2, 'r'); +check(fs.readFile, fs.readFileSync, fileUrl2); +check(fs.readdir, fs.readdirSync, fileUrl2); +check(fs.readlink, fs.readlinkSync, fileUrl2); +check(fs.realpath, fs.realpathSync, fileUrl2); +check(fs.rename, fs.renameSync, fileUrl2, 'foobar'); +check(fs.rename, fs.renameSync, 'foobar', fileUrl2); +check(fs.rmdir, fs.rmdirSync, fileUrl2); +check(fs.stat, fs.statSync, fileUrl2); +check(fs.symlink, fs.symlinkSync, fileUrl2, 'foobar'); +check(fs.symlink, fs.symlinkSync, 'foobar', fileUrl2); +check(fs.truncate, fs.truncateSync, fileUrl2); +check(fs.unlink, fs.unlinkSync, fileUrl2); +check(null, fs.unwatchFile, fileUrl2, common.fail); +check(fs.utimes, fs.utimesSync, fileUrl2, 0, 0); +check(null, fs.watch, fileUrl2, common.fail); +check(null, fs.watchFile, fileUrl2, common.fail); +check(fs.writeFile, fs.writeFileSync, fileUrl2); + // an 'error' for exists means that it doesn't exist. // one of many reasons why this file is the absolute worst. fs.exists('foo\u0000bar', common.mustCall((exists) => { diff --git a/test/parallel/test-fs-whatwg-url.js b/test/parallel/test-fs-whatwg-url.js new file mode 100644 index 0000000000..7a94fd68d7 --- /dev/null +++ b/test/parallel/test-fs-whatwg-url.js @@ -0,0 +1,70 @@ +'use strict'; + +const common = require('../common'); +const assert = require('assert'); +const path = require('path'); +const fs = require('fs'); +const os = require('os'); +const URL = require('url').URL; +const Buffer = require('buffer').Buffer; + +function pathToFileURL(p) { + if (!path.isAbsolute(p)) + throw new Error('Path must be absolute'); + if (common.isWindows && p.startsWith('\\\\')) + p = p.slice(2); + return new URL(`file://${p}`); +} + +const p = path.resolve(common.fixturesDir, 'a.js'); +const url = pathToFileURL(p); + +assert(url instanceof URL); + +// Check that we can pass in a URL object successfully +fs.readFile(url, common.mustCall((err, data) => { + assert.ifError(err); + assert(Buffer.isBuffer(data)); +})); + +// Check that using a non file:// URL reports an error +const httpUrl = new URL('http://example.org'); +fs.readFile(httpUrl, common.mustCall((err) => { + assert(err); + assert.strictEqual(err.message, + 'Only `file:` URLs are supported'); +})); + +// pct-encoded characters in the path will be decoded and checked +fs.readFile(new URL('file:///c:/tmp/%00test'), common.mustCall((err) => { + assert(err); + assert.strictEqual(err.message, + 'Path must be a string without null bytes'); +})); + +if (common.isWindows) { + // encoded back and forward slashes are not permitted on windows + ['%2f', '%2F', '%5c', '%5C'].forEach((i) => { + fs.readFile(new URL(`file:///c:/tmp/${i}`), common.mustCall((err) => { + assert(err); + assert.strictEqual(err.message, + 'Path must not include encoded \\ or / characters'); + })); + }); +} else { + // encoded forward slashes are not permitted on other platforms + ['%2f', '%2F'].forEach((i) => { + fs.readFile(new URL(`file:///c:/tmp/${i}`), common.mustCall((err) => { + assert(err); + assert.strictEqual(err.message, + 'Path must not include encoded / characters'); + })); + }); + + fs.readFile(new URL('file://hostname/a/b/c'), common.mustCall((err) => { + assert(err); + assert.strictEqual(err.message, + `File URLs on ${os.platform()} must use ` + + 'hostname \'localhost\' or not specify any hostname'); + })); +}