diff options
author | 2022-11-06 21:42:05 -0800 | |
---|---|---|
committer | 2022-11-06 21:42:05 -0800 | |
commit | e45f72e8e422191adeb4fd1bad896dc6a47c76b3 (patch) | |
tree | 3a76da8b343c081dba84e0ac95f3c2cc2423106a /src/resolver | |
parent | 645cf903350a7fe5f5076100b7c4a6bc8cd1b431 (diff) | |
download | bun-e45f72e8e422191adeb4fd1bad896dc6a47c76b3.tar.gz bun-e45f72e8e422191adeb4fd1bad896dc6a47c76b3.tar.zst bun-e45f72e8e422191adeb4fd1bad896dc6a47c76b3.zip |
Automatically install npm packages when running a script in Bun's runtime (#1459)
* Update bundler.zig
* WIP
* Update README.md
* Update README.md
* wip
* Support running scripts without package.json
* Add `--no-auto-install` and `--prefer-offline` flags
* WIP
* wip
* Update headers-handwritten.h
* WIP
* Build fixes
* Fix UAF
* Update install.zig
* Must call .allocate()
* Micro-optimization: only call .timestamp() once per tick when installing packages
* Support progress bar
* Extend the timestamp for package staleness checks to 1 day
* Add `--prefer-latest`, `-i` CLI Flags
* Fix crash
* Support line text manually being set on an Error instance
* Add a few more fields for error messages
* Fix bug when counting 8 character strings in string builder
* Implement error handling for automatic package installs!
* Fix crash
* Make it say module when there's a slash
* Update module_loader.zig
* Ban dependency versions in import specifiers when a package.json is present
* Remove unused field
* Update README.md
* Update README.md
* Update README.md
* Update README.md
Co-authored-by: Jarred Sumner <709451+Jarred-Sumner@users.noreply.github.com>
Diffstat (limited to 'src/resolver')
-rw-r--r-- | src/resolver/dir_info.zig | 27 | ||||
-rw-r--r-- | src/resolver/package_json.zig | 326 | ||||
-rw-r--r-- | src/resolver/resolver.zig | 1100 |
3 files changed, 1213 insertions, 240 deletions
diff --git a/src/resolver/dir_info.zig b/src/resolver/dir_info.zig index 86d165df9..0d1bac6a7 100644 --- a/src/resolver/dir_info.zig +++ b/src/resolver/dir_info.zig @@ -1,4 +1,5 @@ const bun = @import("../global.zig"); +const std = @import("std"); const string = bun.string; const Output = bun.Output; const Global = bun.Global; @@ -35,17 +36,37 @@ enclosing_tsconfig_json: ?*const TSConfigJSON = null, /// https://github.com/oven-sh/bun/issues/229 enclosing_package_json: ?*PackageJSON = null, +package_json_for_dependencies: ?*PackageJSON = null, + abs_path: string = "", entries: Index = undefined, -has_node_modules: bool = false, // Is there a "node_modules" subdirectory? -is_node_modules: bool = false, // Is this a "node_modules" directory? package_json: ?*PackageJSON = null, // Is there a "package.json" file? tsconfig_json: ?*TSConfigJSON = null, // Is there a "tsconfig.json" file in this directory or a parent directory? abs_real_path: string = "", // If non-empty, this is the real absolute path resolving any symlinks +flags: Flags.Set = Flags.Set{}, + +/// Is there a "node_modules" subdirectory? +pub inline fn hasNodeModules(this: *const DirInfo) bool { + return this.flags.contains(.has_node_modules); +} +/// Is this a "node_modules" directory? +pub inline fn isNodeModules(this: *const DirInfo) bool { + return this.flags.contains(.is_node_modules); +} + +pub const Flags = enum { + /// This directory is a node_modules directory + is_node_modules, + /// This directory has a node_modules subdirectory + has_node_modules, + + pub const Set = std.enums.EnumSet(Flags); +}; + pub fn hasParentPackage(this: *const DirInfo) bool { const parent = this.getParent() orelse return false; - return !parent.is_node_modules; + return !parent.isNodeModules(); } pub fn getFileDescriptor(dirinfo: *const DirInfo) StoredFileDescriptorType { diff --git a/src/resolver/package_json.zig b/src/resolver/package_json.zig index e33c5ac60..034debe59 100644 --- a/src/resolver/package_json.zig +++ b/src/resolver/package_json.zig @@ -28,6 +28,26 @@ pub const MacroImportReplacementMap = std.StringArrayHashMap(string); pub const MacroMap = std.StringArrayHashMapUnmanaged(MacroImportReplacementMap); const ScriptsMap = std.StringArrayHashMap(string); +const Semver = @import("../install/semver.zig"); +const Dependency = @import("../install/dependency.zig"); +const String = @import("../install/semver.zig").String; +const Version = Semver.Version; +const Install = @import("../install/install.zig"); +const FolderResolver = @import("../install/resolvers/folder_resolver.zig"); + +const Architecture = @import("../install/npm.zig").Architecture; +const OperatingSystem = @import("../install/npm.zig").OperatingSystem; +pub const DependencyMap = struct { + map: HashMap = .{}, + source_buf: []const u8 = "", + + pub const HashMap = std.ArrayHashMapUnmanaged( + String, + Dependency, + String.ArrayHashContext, + false, + ); +}; pub const PackageJSON = struct { pub const LoadFramework = enum { @@ -85,6 +105,12 @@ pub const PackageJSON = struct { scripts: ?*ScriptsMap = null, + arch: Architecture = Architecture.all, + os: OperatingSystem = OperatingSystem.all, + + package_manager_package_id: Install.PackageID = Install.invalid_package_id, + dependencies: DependencyMap = .{}, + // Present if the "browser" field is present. This field is intended to be // used by bundlers and lets you redirect the paths of certain 3rd-party // modules that don't work in the browser to other modules that shim that @@ -538,12 +564,13 @@ pub const PackageJSON = struct { } pub fn parse( - comptime ResolverType: type, - r: *ResolverType, + r: *resolver.Resolver, input_path: string, dirname_fd: StoredFileDescriptorType, - comptime generate_hash: bool, + package_id: ?Install.PackageID, comptime include_scripts: bool, + comptime include_dependencies: @Type(.EnumLiteral), + comptime generate_hash: bool, ) ?PackageJSON { // TODO: remove this extra copy @@ -566,7 +593,7 @@ pub const PackageJSON = struct { }; if (r.debug_logs) |*debug| { - debug.addNoteFmt("The file \"{s}\" exists", .{package_json_path}) catch unreachable; + debug.addNoteFmt("The file \"{s}\" exists", .{package_json_path}); } const key_path = fs.Path.init(package_json_path); @@ -716,6 +743,160 @@ pub const PackageJSON = struct { } } + if (comptime include_dependencies == .main or include_dependencies == .local) { + update_dependencies: { + if (package_id) |pkg| { + package_json.package_manager_package_id = pkg; + break :update_dependencies; + } + + // // if there is a name & version, check if the lockfile has the package + if (package_json.name.len > 0 and package_json.version.len > 0) { + if (r.package_manager) |pm| { + const tag = Dependency.Version.Tag.infer(package_json.version); + + if (tag == .npm) { + const sliced = Semver.SlicedString.init(package_json.version, package_json.version); + if (Dependency.parseWithTag(r.allocator, package_json.version, .npm, &sliced, r.log)) |dependency_version| { + if (dependency_version.value.npm.isExact()) { + if (pm.lockfile.resolve(package_json.name, dependency_version)) |resolved| { + package_json.package_manager_package_id = resolved; + if (resolved > 0) { + break :update_dependencies; + } + } + } + } + } + } + } + if (json.get("cpu")) |os_field| { + var first = true; + if (os_field.asArray()) |*array| { + while (array.next()) |item| { + if (item.asString(bun.default_allocator)) |str| { + if (first) { + package_json.arch = Architecture.none; + first = false; + } + package_json.arch = package_json.arch.apply(str); + } + } + } + } + + if (json.get("os")) |os_field| { + var first = true; + if (os_field.asArray()) |*array| { + while (array.next()) |item| { + if (item.asString(bun.default_allocator)) |str| { + if (first) { + package_json.os = OperatingSystem.none; + first = false; + } + package_json.os = package_json.os.apply(str); + } + } + } + } + + const DependencyGroup = Install.Lockfile.Package.DependencyGroup; + const features = .{ + .dependencies = true, + .dev_dependencies = include_dependencies == .main, + .optional_dependencies = false, + .peer_dependencies = false, + }; + + const dependency_groups = comptime brk: { + var out_groups: [ + @as(usize, @boolToInt(features.dependencies)) + + @as(usize, @boolToInt(features.dev_dependencies)) + + @as(usize, @boolToInt(features.optional_dependencies)) + + @as(usize, @boolToInt(features.peer_dependencies)) + ]DependencyGroup = undefined; + var out_group_i: usize = 0; + if (features.dependencies) { + out_groups[out_group_i] = DependencyGroup.dependencies; + out_group_i += 1; + } + + if (features.dev_dependencies) { + out_groups[out_group_i] = DependencyGroup.dev; + out_group_i += 1; + } + if (features.optional_dependencies) { + out_groups[out_group_i] = DependencyGroup.optional; + out_group_i += 1; + } + + if (features.peer_dependencies) { + out_groups[out_group_i] = DependencyGroup.peer; + out_group_i += 1; + } + + break :brk out_groups; + }; + + var total_dependency_count: usize = 0; + inline for (dependency_groups) |group| { + if (json.get(group.field)) |group_json| { + if (group_json.data == .e_object) { + total_dependency_count += group_json.data.e_object.properties.len; + } + } + } + + if (total_dependency_count > 0) { + package_json.dependencies.map = DependencyMap.HashMap{}; + package_json.dependencies.source_buf = json_source.contents; + const ctx = String.ArrayHashContext{ + .a_buf = json_source.contents, + .b_buf = json_source.contents, + }; + package_json.dependencies.map.ensureTotalCapacityContext( + r.allocator, + total_dependency_count, + ctx, + ) catch unreachable; + + inline for (dependency_groups) |group| { + if (json.get(group.field)) |group_json| { + if (group_json.data == .e_object) { + var group_obj = group_json.data.e_object; + for (group_obj.properties.slice()) |*prop| { + const name = prop.key orelse continue; + const name_str = name.asString(r.allocator) orelse continue; + const version_value = prop.value orelse continue; + const version_str = version_value.asString(r.allocator) orelse continue; + const sliced_str = Semver.SlicedString.init(version_str, version_str); + + if (Dependency.parse( + r.allocator, + version_str, + &sliced_str, + r.log, + )) |dependency_version| { + const dependency = Dependency{ + .name = String.init(name_str, name_str), + .version = dependency_version, + .name_hash = bun.hash(name_str), + .behavior = group.behavior, + }; + package_json.dependencies.map.putAssumeCapacityContext( + dependency.name, + dependency, + ctx, + ); + } + } + } + } + } + } + } + } + // used by `bun run` if (include_scripts) { read_scripts: { @@ -1043,8 +1224,49 @@ pub const ESModule = struct { pub const Package = struct { name: string, + version: string = "", subpath: string, + pub const External = struct { + name: Semver.String = .{}, + version: Semver.String = .{}, + subpath: Semver.String = .{}, + }; + + pub fn count(this: Package, builder: *Semver.String.Builder) void { + builder.count(this.name); + builder.count(this.version); + builder.count(this.subpath); + } + + pub fn clone(this: Package, builder: *Semver.String.Builder) External { + return .{ + .name = builder.appendUTF8WithoutPool(Semver.String, this.name, 0), + .version = builder.appendUTF8WithoutPool(Semver.String, this.version, 0), + .subpath = builder.appendUTF8WithoutPool(Semver.String, this.subpath, 0), + }; + } + + pub fn toExternal(this: Package, buffer: []const u8) External { + return .{ + .name = Semver.String.init(buffer, this.name), + .version = Semver.String.init(buffer, this.version), + .subpath = Semver.String.init(buffer, this.subpath), + }; + } + + pub fn withAutoVersion(this: Package) Package { + if (this.version.len == 0) { + return .{ + .name = this.name, + .subpath = this.subpath, + .version = ">=0.0.0", + }; + } + + return this; + } + pub fn parseName(specifier: string) ?string { var slash = strings.indexOfCharNeg(specifier, '/'); if (!strings.startsWithChar(specifier, '@')) { @@ -1059,6 +1281,27 @@ pub const ESModule = struct { } } + pub fn parseVersion(specifier_after_name: string) ?string { + if (strings.indexOfChar(specifier_after_name, '/')) |slash| { + // "foo@/bar" is not a valid specifier\ + // "foo@/" is not a valid specifier + // "foo/@/bar" is not a valid specifier + // "foo@1/bar" is a valid specifier + // "foo@^123.2.3+ba-ab/bar" is a valid specifier + // ^^^^^^^^^^^^^^ + // this is the version + + const remainder = specifier_after_name[0..slash]; + if (remainder.len > 0 and remainder[0] == '@') { + return remainder[1..]; + } + + return remainder; + } + + return null; + } + pub fn parse(specifier: string, subpath_buf: []u8) ?Package { if (specifier.len == 0) return null; var package = Package{ .name = parseName(specifier) orelse return null, .subpath = "" }; @@ -1066,11 +1309,30 @@ pub const ESModule = struct { if (strings.startsWith(package.name, ".") or strings.indexAnyComptime(package.name, "\\%") != null) return null; - std.mem.copy(u8, subpath_buf[1..], specifier[package.name.len..]); - subpath_buf[0] = '.'; - package.subpath = subpath_buf[0 .. specifier[package.name.len..].len + 1]; + const offset: usize = if (package.name.len == 0 or package.name[0] != '@') 0 else 1; + if (strings.indexOfChar(specifier[offset..], '@')) |at| { + package.version = parseVersion(specifier[offset..][at..]) orelse ""; + if (package.version.len == 0) { + package.version = specifier[offset..][at..]; + if (package.version.len > 0 and package.version[0] == '@') { + package.version = package.version[1..]; + } + } + package.name = specifier[0 .. at + offset]; + + parseSubpath(&package.subpath, specifier[@minimum(package.name.len + package.version.len + 1, specifier.len)..], subpath_buf); + } else { + parseSubpath(&package.subpath, specifier[package.name.len..], subpath_buf); + } + return package; } + + pub fn parseSubpath(subpath: *[]const u8, specifier: string, subpath_buf: []u8) void { + std.mem.copy(u8, subpath_buf[1..], specifier); + subpath_buf[0] = '.'; + subpath.* = subpath_buf[0 .. specifier.len + 1]; + } }; const ReverseKind = enum { exact, pattern, prefix }; @@ -1170,7 +1432,7 @@ pub const ESModule = struct { ) Resolution { if (exports.data == .invalid) { if (r.debug_logs) |logs| { - logs.addNote("Invalid package configuration") catch unreachable; + logs.addNote("Invalid package configuration"); } return Resolution{ .status = .InvalidPackageConfiguration, .debug = .{ .token = exports.first_token } }; @@ -1210,7 +1472,7 @@ pub const ESModule = struct { } if (r.debug_logs) |logs| { - logs.addNoteFmt("The path \"{s}\" was not exported", .{subpath}) catch unreachable; + logs.addNoteFmt("The path \"{s}\" was not exported", .{subpath}); } return Resolution{ .status = .PackagePathNotExported, .debug = .{ .token = exports.first_token } }; @@ -1224,13 +1486,13 @@ pub const ESModule = struct { package_url: string, ) Resolution { if (r.debug_logs) |logs| { - logs.addNoteFmt("Checking object path map for \"{s}\"", .{match_key}) catch unreachable; + logs.addNoteFmt("Checking object path map for \"{s}\"", .{match_key}); } if (!strings.endsWithChar(match_key, '.')) { if (match_obj.valueForKey(match_key)) |target| { if (r.debug_logs) |log| { - log.addNoteFmt("Found \"{s}\"", .{match_key}) catch unreachable; + log.addNoteFmt("Found \"{s}\"", .{match_key}); } return r.resolveTarget(package_url, target, "", is_imports, false); @@ -1248,7 +1510,7 @@ pub const ESModule = struct { const target = expansion.value; const subpath = match_key[expansion.key.len - 1 ..]; if (r.debug_logs) |log| { - log.addNoteFmt("The key \"{s}\" matched with \"{s}\" left over", .{ expansion.key, subpath }) catch unreachable; + log.addNoteFmt("The key \"{s}\" matched with \"{s}\" left over", .{ expansion.key, subpath }); } return r.resolveTarget(package_url, target, subpath, is_imports, true); @@ -1259,7 +1521,7 @@ pub const ESModule = struct { const target = expansion.value; const subpath = match_key[expansion.key.len..]; if (r.debug_logs) |log| { - log.addNoteFmt("The key \"{s}\" matched with \"{s}\" left over", .{ expansion.key, subpath }) catch unreachable; + log.addNoteFmt("The key \"{s}\" matched with \"{s}\" left over", .{ expansion.key, subpath }); } var result = r.resolveTarget(package_url, target, subpath, is_imports, false); @@ -1273,13 +1535,13 @@ pub const ESModule = struct { } if (r.debug_logs) |log| { - log.addNoteFmt("The key \"{s}\" did not match", .{expansion.key}) catch unreachable; + log.addNoteFmt("The key \"{s}\" did not match", .{expansion.key}); } } } if (r.debug_logs) |log| { - log.addNoteFmt("No keys matched \"{s}\"", .{match_key}) catch unreachable; + log.addNoteFmt("No keys matched \"{s}\"", .{match_key}); } return Resolution{ @@ -1301,12 +1563,12 @@ pub const ESModule = struct { switch (target.data) { .string => |str| { if (r.debug_logs) |log| { - log.addNoteFmt("Checking path \"{s}\" against target \"{s}\"", .{ subpath, str }) catch unreachable; - log.increaseIndent() catch unreachable; + log.addNoteFmt("Checking path \"{s}\" against target \"{s}\"", .{ subpath, str }); + log.increaseIndent(); } defer { if (r.debug_logs) |log| { - log.decreaseIndent() catch unreachable; + log.decreaseIndent(); } } @@ -1315,7 +1577,7 @@ pub const ESModule = struct { if (comptime !pattern) { if (subpath.len > 0 and !strings.endsWithChar(str, '/')) { if (r.debug_logs) |log| { - log.addNoteFmt("The target \"{s}\" is invalid because it doesn't end with a \"/\"", .{str}) catch unreachable; + log.addNoteFmt("The target \"{s}\" is invalid because it doesn't end with a \"/\"", .{str}); } return Resolution{ .path = str, .status = .InvalidModuleSpecifier, .debug = .{ .token = target.first_token } }; @@ -1325,7 +1587,7 @@ pub const ESModule = struct { // If target does not start with "./", then... if (!strings.startsWith(str, "./")) { if (r.debug_logs) |log| { - log.addNoteFmt("The target \"{s}\" is invalid because it doesn't start with a \"./\"", .{str}) catch unreachable; + log.addNoteFmt("The target \"{s}\" is invalid because it doesn't start with a \"./\"", .{str}); } if (internal and !strings.hasPrefixComptime(str, "../") and !strings.hasPrefix(str, "/")) { @@ -1335,7 +1597,7 @@ pub const ESModule = struct { _ = std.mem.replace(u8, str, "*", subpath, &resolve_target_buf2); const result = resolve_target_buf2[0..len]; if (r.debug_logs) |log| { - log.addNoteFmt("Subsituted \"{s}\" for \"*\" in \".{s}\" to get \".{s}\" ", .{ subpath, str, result }) catch unreachable; + log.addNoteFmt("Subsituted \"{s}\" for \"*\" in \".{s}\" to get \".{s}\" ", .{ subpath, str, result }); } return Resolution{ .path = result, .status = .PackageResolve, .debug = .{ .token = target.first_token } }; @@ -1343,7 +1605,7 @@ pub const ESModule = struct { var parts2 = [_]string{ str, subpath }; const result = resolve_path.joinStringBuf(&resolve_target_buf2, parts2, .auto); if (r.debug_logs) |log| { - log.addNoteFmt("Resolved \".{s}\" to \".{s}\"", .{ str, result }) catch unreachable; + log.addNoteFmt("Resolved \".{s}\" to \".{s}\"", .{ str, result }); } return Resolution{ .path = result, .status = .PackageResolve, .debug = .{ .token = target.first_token } }; @@ -1357,7 +1619,7 @@ pub const ESModule = struct { // segments after the first segment, throw an Invalid Package Target error. if (findInvalidSegment(str)) |invalid| { if (r.debug_logs) |log| { - log.addNoteFmt("The target \"{s}\" is invalid because it contains an invalid segment \"{s}\"", .{ str, invalid }) catch unreachable; + log.addNoteFmt("The target \"{s}\" is invalid because it contains an invalid segment \"{s}\"", .{ str, invalid }); } return Resolution{ .path = str, .status = .InvalidPackageTarget, .debug = .{ .token = target.first_token } }; @@ -1371,7 +1633,7 @@ pub const ESModule = struct { // segments after the first segment, throw an Invalid Package Target error. if (findInvalidSegment(resolved_target)) |invalid| { if (r.debug_logs) |log| { - log.addNoteFmt("The target \"{s}\" is invalid because it contains an invalid segment \"{s}\"", .{ str, invalid }) catch unreachable; + log.addNoteFmt("The target \"{s}\" is invalid because it contains an invalid segment \"{s}\"", .{ str, invalid }); } return Resolution{ .path = str, .status = .InvalidModuleSpecifier, .debug = .{ .token = target.first_token } }; @@ -1383,7 +1645,7 @@ pub const ESModule = struct { _ = std.mem.replace(u8, resolved_target, "*", subpath, &resolve_target_buf2); const result = resolve_target_buf2[0..len]; if (r.debug_logs) |log| { - log.addNoteFmt("Subsituted \"{s}\" for \"*\" in \".{s}\" to get \".{s}\" ", .{ subpath, resolved_target, result }) catch unreachable; + log.addNoteFmt("Subsituted \"{s}\" for \"*\" in \".{s}\" to get \".{s}\" ", .{ subpath, resolved_target, result }); } return Resolution{ .path = result, .status = .Exact, .debug = .{ .token = target.first_token } }; @@ -1391,7 +1653,7 @@ pub const ESModule = struct { var parts2 = [_]string{ package_url, str, subpath }; const result = resolve_path.joinStringBuf(&resolve_target_buf2, parts2, .auto); if (r.debug_logs) |log| { - log.addNoteFmt("Substituted \"{s}\" for \"*\" in \".{s}\" to get \".{s}\" ", .{ subpath, resolved_target, result }) catch unreachable; + log.addNoteFmt("Substituted \"{s}\" for \"*\" in \".{s}\" to get \".{s}\" ", .{ subpath, resolved_target, result }); } return Resolution{ .path = result, .status = .Exact, .debug = .{ .token = target.first_token } }; @@ -1406,7 +1668,7 @@ pub const ESModule = struct { for (keys) |key, i| { if (strings.eqlComptime(key, "default") or r.conditions.contains(key)) { if (r.debug_logs) |log| { - log.addNoteFmt("The key \"{s}\" matched", .{key}) catch unreachable; + log.addNoteFmt("The key \"{s}\" matched", .{key}); } var result = r.resolveTarget(package_url, slice.items(.value)[i], subpath, internal, pattern); @@ -1420,12 +1682,12 @@ pub const ESModule = struct { } if (r.debug_logs) |log| { - log.addNoteFmt("The key \"{s}\" did not match", .{key}) catch unreachable; + log.addNoteFmt("The key \"{s}\" did not match", .{key}); } } if (r.debug_logs) |log| { - log.addNoteFmt("No keys matched", .{}) catch unreachable; + log.addNoteFmt("No keys matched", .{}); } var return_target = target; @@ -1489,7 +1751,7 @@ pub const ESModule = struct { .array => |array| { if (array.len == 0) { if (r.debug_logs) |log| { - log.addNoteFmt("The path \"{s}\" is an empty array", .{subpath}) catch unreachable; + log.addNoteFmt("The path \"{s}\" is an empty array", .{subpath}); } return Resolution{ .path = "", .status = .Null, .debug = .{ .token = target.first_token } }; @@ -1517,7 +1779,7 @@ pub const ESModule = struct { }, .@"null" => { if (r.debug_logs) |log| { - log.addNoteFmt("The path \"{s}\" is null", .{subpath}) catch unreachable; + log.addNoteFmt("The path \"{s}\" is null", .{subpath}); } return Resolution{ .path = "", .status = .Null, .debug = .{ .token = target.first_token } }; @@ -1526,7 +1788,7 @@ pub const ESModule = struct { } if (r.debug_logs) |logs| { - logs.addNoteFmt("Invalid package target for path \"{s}\"", .{subpath}) catch unreachable; + logs.addNoteFmt("Invalid package target for path \"{s}\"", .{subpath}); } return Resolution{ .status = .InvalidPackageTarget, .debug = .{ .token = target.first_token } }; diff --git a/src/resolver/resolver.zig b/src/resolver/resolver.zig index 2c14089ee..a6e6f9b94 100644 --- a/src/resolver/resolver.zig +++ b/src/resolver/resolver.zig @@ -38,6 +38,14 @@ const allocators = @import("../allocators.zig"); const Msg = logger.Msg; const Path = Fs.Path; const NodeModuleBundle = @import("../node_module_bundle.zig").NodeModuleBundle; +const PackageManager = @import("../install/install.zig").PackageManager; +const Dependency = @import("../install/dependency.zig"); +const Install = @import("../install/install.zig"); +const Lockfile = @import("../install/lockfile.zig").Lockfile; +const Package = @import("../install/lockfile.zig").Package; +const Resolution = @import("../install/resolution.zig").Resolution; +const Semver = @import("../install/semver.zig"); +const DotEnv = @import("../env_loader.zig"); pub fn isPackagePath(path: string) bool { // this could probably be flattened into something more optimized @@ -126,6 +134,13 @@ pub const Result = struct { file_fd: StoredFileDescriptorType = 0, import_kind: ast.ImportKind = undefined, + pub const Union = union(enum) { + success: Result, + failure: anyerror, + pending: PendingResolution, + not_found: void, + }; + pub fn path(this: *Result) ?*Path { if (!this.path_pair.primary.is_disabled) return &this.path_pair.primary; @@ -235,6 +250,7 @@ threadlocal var remap_path_buf: [bun.MAX_PATH_BYTES]u8 = undefined; threadlocal var load_as_file_buf: [bun.MAX_PATH_BYTES]u8 = undefined; threadlocal var remap_path_trailing_slash: [bun.MAX_PATH_BYTES]u8 = undefined; threadlocal var tsconfig_paths_buf: [bun.MAX_PATH_BYTES]u8 = undefined; +threadlocal var path_in_global_disk_cache_buf: [bun.MAX_PATH_BYTES]u8 = undefined; pub const DebugLogs = struct { what: string = "", @@ -256,33 +272,33 @@ pub const DebugLogs = struct { // d.indent.deinit(); } - pub fn increaseIndent(d: *DebugLogs) !void { + pub fn increaseIndent(d: *DebugLogs) void { @setCold(true); - try d.indent.append(" "); + d.indent.append(" ") catch unreachable; } - pub fn decreaseIndent(d: *DebugLogs) !void { + pub fn decreaseIndent(d: *DebugLogs) void { @setCold(true); d.indent.list.shrinkRetainingCapacity(d.indent.list.items.len - 1); } - pub fn addNote(d: *DebugLogs, _text: string) !void { + pub fn addNote(d: *DebugLogs, _text: string) void { @setCold(true); var text = _text; const len = d.indent.len(); if (len > 0) { - var __text = try d.notes.allocator.alloc(u8, text.len + len); + var __text = d.notes.allocator.alloc(u8, text.len + len) catch unreachable; std.mem.copy(u8, __text, d.indent.list.items); std.mem.copy(u8, __text[len..__text.len], _text); d.notes.allocator.free(_text); } - try d.notes.append(logger.rangeData(null, logger.Range.None, text)); + d.notes.append(logger.rangeData(null, logger.Range.None, text)) catch unreachable; } - pub fn addNoteFmt(d: *DebugLogs, comptime fmt: string, args: anytype) !void { + pub fn addNoteFmt(d: *DebugLogs, comptime fmt: string, args: anytype) void { @setCold(true); - return try d.addNote(try std.fmt.allocPrint(d.notes.allocator, fmt, args)); + return d.addNote(std.fmt.allocPrint(d.notes.allocator, fmt, args) catch unreachable); } }; @@ -294,6 +310,62 @@ pub const MatchResult = struct { package_json: ?*PackageJSON = null, diff_case: ?Fs.FileSystem.Entry.Lookup.DifferentCase = null, dir_info: ?*DirInfo = null, + + pub const Union = union(enum) { + not_found: void, + success: MatchResult, + pending: PendingResolution, + failure: anyerror, + }; +}; + +pub const PendingResolution = struct { + esm: ESModule.Package.External = .{}, + dependency: Dependency.Version = .{}, + resolution_id: Install.PackageID = Install.invalid_package_id, + root_dependency_id: Install.PackageID = Install.invalid_package_id, + import_record_id: u32 = std.math.maxInt(u32), + string_buf: []u8 = "", + tag: Tag, + + pub const List = std.MultiArrayList(PendingResolution); + + pub fn deinitListItems(list_: List, allocator: std.mem.Allocator) void { + var list = list_; + var dependencies = list.items(.dependency); + var string_bufs = list.items(.string_buf); + for (dependencies) |*dependency| { + dependency.deinit(); + } + + for (string_bufs) |string_buf| { + allocator.free(string_buf); + } + } + + pub fn deinit(this: *PendingResolution, allocator: std.mem.Allocator) void { + this.dependency.deinit(); + allocator.free(this.string_buf); + } + + pub const Tag = enum { + download, + resolve, + done, + }; + + pub fn init( + allocator: std.mem.Allocator, + esm: ESModule.Package, + dependency: Dependency.Version, + resolution_id: Install.PackageID, + ) !PendingResolution { + return PendingResolution{ + .esm = try esm.copy(allocator), + .dependency = dependency, + .resolution_id = resolution_id, + }; + } }; pub const LoadResult = struct { @@ -358,6 +430,11 @@ pub const Resolver = struct { caches: CacheSet, + package_manager: ?*PackageManager = null, + onWakePackageManager: PackageManager.WakeHandler = .{}, + main_file_for_package_manager: []const u8 = "", + env_loader: ?*DotEnv.Loader = null, + // These are sets that represent various conditions for the "exports" field // in package.json. // esm_conditions_default: std.StringHashMap(bool), @@ -402,6 +479,27 @@ pub const Resolver = struct { // all parent directories dir_cache: *DirInfo.HashMap, + pub fn getPackageManager(this: *Resolver) *PackageManager { + if (this.package_manager != null) { + return this.package_manager.?; + } + bun.HTTPThead.init() catch unreachable; + this.package_manager = PackageManager.initWithRuntime( + this.log, + this.opts.install, + this.allocator, + .{}, + this.env_loader.?, + ) catch @panic("Failed to initialize package manager"); + this.package_manager.?.onWake = this.onWakePackageManager; + + return this.package_manager.?; + } + + pub inline fn usePackageManager(self: *const ThisResolver) bool { + return self.opts.global_cache.isEnabled(); + } + pub fn init1( allocator: std.mem.Allocator, log: *logger.Log, @@ -590,7 +688,13 @@ pub const Resolver = struct { } } - pub fn resolve(r: *ThisResolver, source_dir: string, import_path: string, kind: ast.ImportKind) !Result { + pub fn resolveAndAutoInstall( + r: *ThisResolver, + source_dir: string, + import_path: string, + kind: ast.ImportKind, + global_cache: GlobalCache, + ) Result.Union { const original_order = r.extension_order; defer r.extension_order = original_order; r.extension_order = switch (kind) { @@ -613,10 +717,10 @@ pub const Resolver = struct { r.debug_logs.?.deinit(); } - r.debug_logs = try DebugLogs.init(r.allocator); + r.debug_logs = DebugLogs.init(r.allocator) catch unreachable; } - if (import_path.len == 0) return error.ModuleNotFound; + if (import_path.len == 0) return .{ .not_found = {} }; // Certain types of URLs default to being external for convenience if (r.isExternalPattern(import_path) or @@ -633,17 +737,19 @@ pub const Resolver = struct { strings.startsWith(import_path, "//")) { if (r.debug_logs) |*debug| { - try debug.addNote("Marking this path as implicitly external"); + debug.addNote("Marking this path as implicitly external"); r.flushDebugLogs(.success) catch {}; } - return Result{ - .import_kind = kind, - .path_pair = PathPair{ - .primary = Path.init(import_path), + return .{ + .success = Result{ + .import_kind = kind, + .path_pair = PathPair{ + .primary = Path.init(import_path), + }, + .is_external = true, + .module_type = .esm, }, - .is_external = true, - .module_type = .esm, }; } @@ -653,22 +759,26 @@ pub const Resolver = struct { // "@import 'data:text/css,body{background:white}';" if (data_url.decode_mime_type() != .Unsupported) { if (r.debug_logs) |*debug| { - debug.addNote("Putting this path in the \"dataurl\" namespace") catch {}; + debug.addNote("Putting this path in the \"dataurl\" namespace"); r.flushDebugLogs(.success) catch {}; } - return Result{ .path_pair = PathPair{ .primary = Path.initWithNamespace(import_path, "dataurl") } }; + return .{ + .success = Result{ .path_pair = PathPair{ .primary = Path.initWithNamespace(import_path, "dataurl") } }, + }; } // "background: url();" if (r.debug_logs) |*debug| { - debug.addNote("Marking this \"dataurl\" as external") catch {}; + debug.addNote("Marking this \"dataurl\" as external"); r.flushDebugLogs(.success) catch {}; } - return Result{ - .path_pair = PathPair{ .primary = Path.initWithNamespace(import_path, "dataurl") }, - .is_external = true, + return .{ + .success = Result{ + .path_pair = PathPair{ .primary = Path.initWithNamespace(import_path, "dataurl") }, + .is_external = true, + }, }; } @@ -676,27 +786,48 @@ pub const Resolver = struct { // virtual modules (e.g. stdin) if a resolve directory is not specified. if (source_dir.len == 0) { if (r.debug_logs) |*debug| { - debug.addNote("Cannot resolve this path without a directory") catch {}; + debug.addNote("Cannot resolve this path without a directory"); r.flushDebugLogs(.fail) catch {}; } - return error.MissingResolveDir; + return .{ .failure = error.MissingResolveDir }; } // r.mutex.lock(); // defer r.mutex.unlock(); errdefer (r.flushDebugLogs(.fail) catch {}); - var result = (try r.resolveWithoutSymlinks(source_dir, import_path, kind)) orelse { - r.flushDebugLogs(.fail) catch {}; - return error.ModuleNotFound; - }; - if (!strings.eqlComptime(result.path_pair.primary.namespace, "node")) - try r.finalizeResult(&result, kind); + switch (r.resolveWithoutSymlinks(source_dir, import_path, kind, global_cache)) { + .success => |*result| { + if (!strings.eqlComptime(result.path_pair.primary.namespace, "node")) + r.finalizeResult(result, kind) catch |err| return .{ .failure = err }; - r.flushDebugLogs(.success) catch {}; - result.import_kind = kind; - return result; + r.flushDebugLogs(.success) catch {}; + result.import_kind = kind; + return .{ .success = result.* }; + }, + .failure => |e| { + r.flushDebugLogs(.fail) catch {}; + return .{ .failure = e }; + }, + .pending => |pending| { + r.flushDebugLogs(.fail) catch {}; + return .{ .pending = pending }; + }, + .not_found => { + r.flushDebugLogs(.fail) catch {}; + return .{ .not_found = {} }; + }, + } + } + + pub fn resolve(r: *ThisResolver, source_dir: string, import_path: string, kind: ast.ImportKind) !Result { + switch (r.resolveAndAutoInstall(source_dir, import_path, kind, GlobalCache.disable)) { + .success => |result| return result, + .pending, .not_found => return error.ModuleNotFound, + + .failure => |e| return e, + } } const ModuleTypeMap = bun.ComptimeStringMap(options.ModuleType, .{ @@ -738,7 +869,7 @@ pub const Resolver = struct { if (result.file_fd == 0) result.file_fd = query.entry.cache.fd; if (r.debug_logs) |*debug| { - debug.addNoteFmt("Resolved symlink \"{s}\" to \"{s}\"", .{ path.text, symlink_path }) catch {}; + debug.addNoteFmt("Resolved symlink \"{s}\" to \"{s}\"", .{ path.text, symlink_path }); } } else if (dir.abs_real_path.len > 0) { var parts = [_]string{ dir.abs_real_path, query.entry.base() }; @@ -776,7 +907,7 @@ pub const Resolver = struct { const symlink = try Fs.FileSystem.FilenameStore.instance.append(@TypeOf(out), out); if (r.debug_logs) |*debug| { - debug.addNoteFmt("Resolved symlink \"{s}\" to \"{s}\"", .{ symlink, path.text }) catch {}; + debug.addNoteFmt("Resolved symlink \"{s}\" to \"{s}\"", .{ symlink, path.text }); } query.entry.cache.symlink = PathString.init(symlink); if (result.file_fd == 0) result.file_fd = query.entry.cache.fd; @@ -796,7 +927,13 @@ pub const Resolver = struct { result.module_type = module_type; } - pub fn resolveWithoutSymlinks(r: *ThisResolver, source_dir: string, import_path_: string, kind: ast.ImportKind) !?Result { + pub fn resolveWithoutSymlinks( + r: *ThisResolver, + source_dir: string, + import_path_: string, + kind: ast.ImportKind, + global_cache: GlobalCache, + ) Result.Union { var import_path = import_path_; // This implements the module resolution algorithm from node.js, which is @@ -819,7 +956,7 @@ pub const Resolver = struct { // users will not be able to accidentally make use of these paths. if (strings.startsWith(import_path, "/") or std.fs.path.isAbsolutePosix(import_path)) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("The import \"{s}\" is being treated as an absolute path", .{import_path}) catch {}; + debug.addNoteFmt("The import \"{s}\" is being treated as an absolute path", .{import_path}); } // First, check path overrides from the nearest enclosing TypeScript "tsconfig.json" file @@ -830,13 +967,15 @@ pub const Resolver = struct { if (r.matchTSConfigPaths(tsconfig, import_path, kind)) |res| { // We don't set the directory fd here because it might remap an entirely different directory - return Result{ - .path_pair = res.path_pair, - .diff_case = res.diff_case, - .package_json = res.package_json, - .dirname_fd = res.dirname_fd, - .file_fd = res.file_fd, - .jsx = tsconfig.mergeJSX(result.jsx), + return .{ + .success = Result{ + .path_pair = res.path_pair, + .diff_case = res.diff_case, + .package_json = res.package_json, + .dirname_fd = res.dirname_fd, + .file_fd = res.file_fd, + .jsx = tsconfig.mergeJSX(result.jsx), + }, }; } } @@ -849,28 +988,32 @@ pub const Resolver = struct { // That way we preserve the literal text in the output and don't generate // a relative path from the output directory to that path. if (r.debug_logs) |*debug| { - debug.addNoteFmt("The path \"{s}\" is marked as external by the user", .{import_path}) catch {}; + debug.addNoteFmt("The path \"{s}\" is marked as external by the user", .{import_path}); } - return Result{ - .path_pair = .{ .primary = Path.init(import_path) }, - .is_external = true, + return .{ + .success = Result{ + .path_pair = .{ .primary = Path.init(import_path) }, + .is_external = true, + }, }; } // Run node's resolution rules (e.g. adding ".js") if (r.loadAsFileOrDirectory(import_path, kind)) |entry| { - return Result{ - .dirname_fd = entry.dirname_fd, - .path_pair = entry.path_pair, - .diff_case = entry.diff_case, - .package_json = entry.package_json, - .file_fd = entry.file_fd, - .jsx = r.opts.jsx, + return .{ + .success = Result{ + .dirname_fd = entry.dirname_fd, + .path_pair = entry.path_pair, + .diff_case = entry.diff_case, + .package_json = entry.package_json, + .file_fd = entry.file_fd, + .jsx = r.opts.jsx, + }, }; } - return null; + return .{ .not_found = {} }; } // Check both relative and package paths for CSS URL tokens, with relative @@ -889,12 +1032,14 @@ pub const Resolver = struct { // That way we preserve the literal text in the output and don't generate // a relative path from the output directory to that path. if (r.debug_logs) |*debug| { - debug.addNoteFmt("The path \"{s}\" is marked as external by the user", .{abs_path}) catch {}; + debug.addNoteFmt("The path \"{s}\" is marked as external by the user", .{abs_path}); } - return Result{ - .path_pair = .{ .primary = Path.init(r.fs.dirname_store.append(@TypeOf(abs_path), abs_path) catch unreachable) }, - .is_external = true, + return .{ + .success = Result{ + .path_pair = .{ .primary = Path.init(r.fs.dirname_store.append(@TypeOf(abs_path), abs_path) catch unreachable) }, + .is_external = true, + }, }; } @@ -912,23 +1057,28 @@ pub const Resolver = struct { if (remap.len == 0) { var _path = Path.init(r.fs.dirname_store.append(string, abs_path) catch unreachable); _path.is_disabled = true; - return Result{ - .path_pair = PathPair{ - .primary = _path, + return .{ + .success = Result{ + .path_pair = PathPair{ + .primary = _path, + }, }, }; } - if (r.resolveWithoutRemapping(import_dir_info, remap, kind)) |_result| { - result = Result{ - .path_pair = _result.path_pair, - .diff_case = _result.diff_case, - .dirname_fd = _result.dirname_fd, - .package_json = pkg, - .jsx = r.opts.jsx, - }; - check_relative = false; - check_package = false; + switch (r.resolveWithoutRemapping(import_dir_info, remap, kind, global_cache)) { + .success => |_result| { + result = Result{ + .path_pair = _result.path_pair, + .diff_case = _result.diff_case, + .dirname_fd = _result.dirname_fd, + .package_json = pkg, + .jsx = r.opts.jsx, + }; + check_relative = false; + check_package = false; + }, + else => {}, } } } @@ -945,7 +1095,7 @@ pub const Resolver = struct { .jsx = r.opts.jsx, }; } else if (!check_package) { - return null; + return .{ .not_found = {} }; } } } @@ -966,7 +1116,7 @@ pub const Resolver = struct { result.module_type = .cjs; result.package_json = @intToPtr(*PackageJSON, @ptrToInt(fallback_module.package_json)); result.is_from_node_modules = true; - return result; + return .{ .success = result }; // "node:* // "fs" // "fs/*" @@ -982,7 +1132,7 @@ pub const Resolver = struct { result.module_type = .cjs; result.path_pair.primary.is_disabled = true; result.is_from_node_modules = true; - return result; + return .{ .success = result }; } } @@ -992,11 +1142,13 @@ pub const Resolver = struct { while (true) { if (r.opts.external.node_modules.contains(query)) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("The path \"{s}\" was marked as external by the user", .{query}) catch {}; + debug.addNoteFmt("The path \"{s}\" was marked as external by the user", .{query}); } - return Result{ - .path_pair = .{ .primary = Path.init(query) }, - .is_external = true, + return .{ + .success = Result{ + .path_pair = .{ .primary = Path.init(query) }, + .is_external = true, + }, }; } @@ -1007,7 +1159,7 @@ pub const Resolver = struct { } } - var source_dir_info = (r.dirInfoCached(source_dir) catch null) orelse return null; + var source_dir_info = (r.dirInfoCached(source_dir) catch null) orelse return .{ .not_found = {} }; // Support remapping one package path to another via the "browser" field if (source_dir_info.getEnclosingBrowserScope()) |browser_scope| { @@ -1020,30 +1172,37 @@ pub const Resolver = struct { if (remapped.len == 0) { // "browser": {"module": false} // does the module exist in the filesystem? - if (r.loadNodeModules(import_path, kind, source_dir_info, false)) |node_module| { - var pair = node_module.path_pair; - pair.primary.is_disabled = true; - if (pair.secondary != null) { - pair.secondary.?.is_disabled = true; - } - return Result{ - .path_pair = pair, - .dirname_fd = node_module.dirname_fd, - .diff_case = node_module.diff_case, - .package_json = package_json, - .jsx = r.opts.jsx, - }; - } else { - // "browser": {"module": false} - // the module doesn't exist and it's disabled - // so we should just not try to load it - var primary = Path.init(import_path); - primary.is_disabled = true; - return Result{ - .path_pair = PathPair{ .primary = primary }, - .diff_case = null, - .jsx = r.opts.jsx, - }; + switch (r.loadNodeModules(import_path, kind, source_dir_info, global_cache, false)) { + .success => |node_module| { + var pair = node_module.path_pair; + pair.primary.is_disabled = true; + if (pair.secondary != null) { + pair.secondary.?.is_disabled = true; + } + return .{ + .success = Result{ + .path_pair = pair, + .dirname_fd = node_module.dirname_fd, + .diff_case = node_module.diff_case, + .package_json = package_json, + .jsx = r.opts.jsx, + }, + }; + }, + else => { + // "browser": {"module": false} + // the module doesn't exist and it's disabled + // so we should just not try to load it + var primary = Path.init(import_path); + primary.is_disabled = true; + return .{ + .success = Result{ + .path_pair = PathPair{ .primary = primary }, + .diff_case = null, + .jsx = r.opts.jsx, + }, + }; + }, } } @@ -1053,54 +1212,59 @@ pub const Resolver = struct { } } - if (r.resolveWithoutRemapping(source_dir_info, import_path, kind)) |res| { - result.path_pair = res.path_pair; - result.dirname_fd = res.dirname_fd; - result.file_fd = res.file_fd; - result.package_json = res.package_json; - result.diff_case = res.diff_case; - result.is_from_node_modules = result.is_from_node_modules or res.is_node_module; - result.jsx = r.opts.jsx; + switch (r.resolveWithoutRemapping(source_dir_info, import_path, kind, global_cache)) { + .success => |res| { + result.path_pair = res.path_pair; + result.dirname_fd = res.dirname_fd; + result.file_fd = res.file_fd; + result.package_json = res.package_json; + result.diff_case = res.diff_case; + result.is_from_node_modules = result.is_from_node_modules or res.is_node_module; + result.jsx = r.opts.jsx; - if (res.path_pair.primary.is_disabled and res.path_pair.secondary == null) { - return result; - } + if (res.path_pair.primary.is_disabled and res.path_pair.secondary == null) { + return .{ .success = result }; + } - if (res.package_json != null) { - var base_dir_info = res.dir_info orelse (r.readDirInfo(res.path_pair.primary.name.dir) catch null) orelse return result; - if (base_dir_info.getEnclosingBrowserScope()) |browser_scope| { - if (r.checkBrowserMap( - browser_scope, - res.path_pair.primary.text, - .AbsolutePath, - )) |remap| { - if (remap.len == 0) { - result.path_pair.primary.is_disabled = true; - result.path_pair.primary = Fs.Path.initWithNamespace(remap, "file"); - } else { - if (r.resolveWithoutRemapping(browser_scope, remap, kind)) |remapped| { - result.path_pair = remapped.path_pair; - result.dirname_fd = remapped.dirname_fd; - result.file_fd = remapped.file_fd; - result.package_json = remapped.package_json; - result.diff_case = remapped.diff_case; - - result.is_from_node_modules = result.is_from_node_modules or remapped.is_node_module; - return result; + if (res.package_json != null) { + var base_dir_info = res.dir_info orelse (r.readDirInfo(res.path_pair.primary.name.dir) catch null) orelse return .{ .success = result }; + if (base_dir_info.getEnclosingBrowserScope()) |browser_scope| { + if (r.checkBrowserMap( + browser_scope, + res.path_pair.primary.text, + .AbsolutePath, + )) |remap| { + if (remap.len == 0) { + result.path_pair.primary.is_disabled = true; + result.path_pair.primary = Fs.Path.initWithNamespace(remap, "file"); + } else { + switch (r.resolveWithoutRemapping(browser_scope, remap, kind, global_cache)) { + .success => |remapped| { + result.path_pair = remapped.path_pair; + result.dirname_fd = remapped.dirname_fd; + result.file_fd = remapped.file_fd; + result.package_json = remapped.package_json; + result.diff_case = remapped.diff_case; + + result.is_from_node_modules = result.is_from_node_modules or remapped.is_node_module; + return .{ .success = result }; + }, + else => {}, + } } } } } - } - return result; - } else { - // Note: node's "self references" are not currently supported - return null; + return .{ .success = result }; + }, + .pending => |p| return .{ .pending = p }, + .failure => |p| return .{ .failure = p }, + else => return .{ .not_found = {} }, } } - return result; + return .{ .success = result }; } pub fn packageJSONForResolvedNodeModule( @@ -1201,17 +1365,18 @@ pub const Resolver = struct { import_path: string, kind: ast.ImportKind, _dir_info: *DirInfo, + global_cache: GlobalCache, forbid_imports: bool, - ) ?MatchResult { + ) MatchResult.Union { var dir_info = _dir_info; if (r.debug_logs) |*debug| { - debug.addNoteFmt("Searching for {s} in \"node_modules\" directories starting from \"{s}\"", .{ import_path, dir_info.abs_path }) catch {}; - debug.increaseIndent() catch {}; + debug.addNoteFmt("Searching for {s} in \"node_modules\" directories starting from \"{s}\"", .{ import_path, dir_info.abs_path }); + debug.increaseIndent(); } defer { if (r.debug_logs) |*debug| { - debug.decreaseIndent() catch {}; + debug.decreaseIndent(); } } @@ -1221,7 +1386,7 @@ pub const Resolver = struct { // Try path substitutions first if (tsconfig.paths.count() > 0) { if (r.matchTSConfigPaths(tsconfig, import_path, kind)) |res| { - return res; + return .{ .success = res }; } } @@ -1232,7 +1397,7 @@ pub const Resolver = struct { const abs = r.fs.absBuf(&paths, &load_as_file_or_directory_via_tsconfig_base_path); if (r.loadAsFileOrDirectory(abs, kind)) |res| { - return res; + return .{ .success = res }; } // r.allocator.free(abs); } @@ -1248,9 +1413,9 @@ pub const Resolver = struct { if (import_path.len == 1 or strings.hasPrefix(import_path, "#/")) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("The path \"{s}\" must not equal \"#\" and must not start with \"#/\"", .{import_path}) catch {}; + debug.addNoteFmt("The path \"{s}\" must not equal \"#\" and must not start with \"#/\"", .{import_path}); } - return null; + return .{ .not_found = {} }; } const esmodule = ESModule{ @@ -1269,24 +1434,34 @@ pub const Resolver = struct { esm_resolution.path, kind, dir_info, + global_cache, true, ); - return r.handleESMResolution(esm_resolution, package_json.source.path.name.dir, kind, package_json); + if (r.handleESMResolution(esm_resolution, package_json.source.path.name.dir, kind, package_json)) |result| { + return .{ .success = result }; + } + + return .{ .not_found = {} }; } } } } + var source_dir_info = dir_info; + var any_node_modules_folder = false; + const use_node_module_resolver = global_cache != .force; + // Then check for the package in any enclosing "node_modules" directories - while (true) { + while (use_node_module_resolver) { // Skip directories that are themselves called "node_modules", since we // don't ever want to search for "node_modules/node_modules" - if (dir_info.has_node_modules) { + if (dir_info.hasNodeModules()) { + any_node_modules_folder = true; var _paths = [_]string{ dir_info.abs_path, "node_modules", import_path }; const abs_path = r.fs.absBuf(&_paths, &node_modules_check_buf); if (r.debug_logs) |*debug| { - debug.addNoteFmt("Checking for a package in the directory \"{s}\"", .{abs_path}) catch {}; + debug.addNoteFmt("Checking for a package in the directory \"{s}\"", .{abs_path}); } if (esm_) |esm| { @@ -1318,25 +1493,448 @@ pub const Resolver = struct { // directory path accidentally being interpreted as URL escapes. const esm_resolution = esmodule.resolve("/", esm.subpath, exports_map.root); - return r.handleESMResolution(esm_resolution, abs_package_path, kind, package_json); + if (r.handleESMResolution(esm_resolution, abs_package_path, kind, package_json)) |result| { + return .{ .success = result }; + } + + return .{ .not_found = {} }; } } } } if (r.loadAsFileOrDirectory(abs_path, kind)) |res| { - return res; + return .{ .success = res }; } - // r.allocator.free(abs_path); } dir_info = dir_info.getParent() orelse break; } + dir_info = source_dir_info; + + // this is the magic! + if (global_cache.canUse(any_node_modules_folder) and r.usePackageManager() and esm_ != null) { + const esm = esm_.?.withAutoVersion(); + load_module_from_cache: { + + // If the source directory doesn't have a node_modules directory, we can + // check the global cache directory for a package.json file. + var manager = r.getPackageManager(); + var dependency_version: Dependency.Version = .{}; + var dependency_behavior = @intToEnum(Dependency.Behavior, Dependency.Behavior.normal); + // const initial_pending_tasks = manager.pending_tasks; + var resolved_package_id: Install.PackageID = brk: { + // check if the package.json in the source directory was already added to the lockfile + // and try to look up the dependency from there + if (dir_info.package_json_for_dependencies) |package_json| { + var dependencies_list: []const Dependency = &[_]Dependency{}; + var string_buf: []const u8 = ""; + const resolve_from_lockfile = package_json.package_manager_package_id != Install.invalid_package_id; + + if (resolve_from_lockfile) { + const dependencies = &manager.lockfile.packages.items(.dependencies)[package_json.package_manager_package_id]; + + // try to find this package name in the dependencies of the enclosing package + dependencies_list = dependencies.get(manager.lockfile.buffers.dependencies.items); + string_buf = manager.lockfile.buffers.string_bytes.items; + } else if (esm_.?.version.len == 0) { + // If you don't specify a version, default to the one chosen in your package.json + dependencies_list = package_json.dependencies.map.values(); + string_buf = package_json.dependencies.source_buf; + } + + var hash: u64 = std.math.maxInt(u64); + + for (dependencies_list) |dependency, dependency_id| { + const dep_name_ = &dependency.name; + const dep_name = dep_name_.slice(string_buf); + if (dep_name.len == esm.name.len) { + if (hash == std.math.maxInt(u64)) { + hash = bun.hash(dep_name); + } + + if (hash != dependency.name_hash) { + continue; + } + + std.debug.assert(strings.eql(dep_name, esm.name)); + + dependency_version = dependency.version; + dependency_behavior = dependency.behavior; + + if (resolve_from_lockfile) { + const resolutions = &manager.lockfile.packages.items(.resolutions)[package_json.package_manager_package_id]; + + // found it! + break :brk resolutions.get(manager.lockfile.buffers.resolutions.items)[dependency_id]; + } + + break; + } + } + } + + // check if the lockfile already resolved this package somewhere + { + if (dependency_version.tag == .uninitialized) { + const sliced_string = Semver.SlicedString.init(esm.version, esm.version); + if (esm_.?.version.len > 0 and dir_info.enclosing_package_json != null and global_cache.allowVersionSpecifier()) { + return .{ .failure = error.VersionSpecifierNotAllowedHere }; + } + dependency_version = Dependency.parse( + r.allocator, + esm.version, + &sliced_string, + r.log, + ) orelse break :load_module_from_cache; + } + + // first we check if the lockfile already has a version of this package somewhere at all + if (manager.lockfile.resolve(esm.name, dependency_version)) |id| { + break :brk id; + } + } + + // If we get here, it means that the lockfile doesn't have this package at all. + // we know nothing + break :brk Install.invalid_package_id; + }; + + // Now, there are two possible states: + // 1) We have resolved the package ID, either from the + // lockfile globally OR from the particular package.json + // dependencies list + // + // 2) We parsed the Dependency.Version but there is no + // existing resolved package ID + + // If its an exact version, we can just immediately look it up in the global cache and resolve from there + // If the resolved package ID is _not_ invalid, we can just check + + // If this returns null, then it means we need to *resolve* the package + // Even after resolution, we might still need to download the package + // There are two steps here! Two steps! + const resolution: Resolution = brk: { + if (resolved_package_id != Install.invalid_package_id) { + break :brk manager.lockfile.packages.items(.resolution)[resolved_package_id]; + } + + // unsupported or not found dependency, we might need to install it to the cache + switch (r.enqueueDependencyToResolve( + dir_info.package_json_for_dependencies orelse dir_info.package_json, + esm, + dependency_behavior, + &resolved_package_id, + dependency_version, + )) { + .resolution => |res| break :brk res, + .pending => |pending| return .{ .pending = pending }, + .failure => |err| return .{ .failure = err }, + // this means we looked it up in the registry and the package doesn't exist or the version doesn't exist + .not_found => return .{ .not_found = {} }, + } + }; + + const dir_path_for_resolution = manager.pathForResolution(resolved_package_id, resolution, &path_in_global_disk_cache_buf) catch |err| { + // if it's missing, we need to install it + if (err == error.FileNotFound) { + switch (manager.getPreinstallState(resolved_package_id, manager.lockfile)) { + .done => { + var path = Fs.Path.init(import_path); + path.is_disabled = true; + // this might mean the package is disabled + return .{ + .success = .{ + .path_pair = .{ + .primary = path, + }, + }, + }; + }, + .extract, .extracting => |st| { + if (!global_cache.canInstall()) { + return .{ .not_found = {} }; + } + var builder = Semver.String.Builder{}; + esm.count(&builder); + builder.allocate(manager.allocator) catch unreachable; + const cloned = esm.clone(&builder); + + if (st == .extract) + manager.enqueuePackageForDownload( + esm.name, + resolved_package_id, + resolution.value.npm.version, + manager.lockfile.str(resolution.value.npm.url), + .{ + .root_request_id = 0, + }, + ); + + return .{ + .pending = .{ + .esm = cloned, + .dependency = dependency_version, + .resolution_id = resolved_package_id, + + .string_buf = builder.allocatedSlice(), + .tag = .download, + }, + }; + }, + else => {}, + } + } + + return .{ .failure = err }; + }; + + if (r.dirInfoForResolution(dir_path_for_resolution, resolved_package_id)) |dir_info_to_use_| { + if (dir_info_to_use_) |pkg_dir_info| { + const abs_package_path = pkg_dir_info.abs_path; + + if (pkg_dir_info.package_json) |package_json| { + if (package_json.exports) |exports_map| { + // The condition set is determined by the kind of import + const esmodule = ESModule{ + .conditions = switch (kind) { + ast.ImportKind.require, + ast.ImportKind.require_resolve, + => r.opts.conditions.require, + else => r.opts.conditions.import, + }, + .allocator = r.allocator, + .debug_logs = if (r.debug_logs) |*debug| + debug + else + null, + }; + + // Resolve against the path "/", then join it with the absolute + // directory path. This is done because ESM package resolution uses + // URLs while our path resolution uses file system paths. We don't + // want problems due to Windows paths, which are very unlike URL + // paths. We also want to avoid any "%" characters in the absolute + // directory path accidentally being interpreted as URL escapes. + const esm_resolution = esmodule.resolve("/", esm.subpath, exports_map.root); + + if (r.handleESMResolution(esm_resolution, abs_package_path, kind, package_json)) |*result| { + result.is_node_module = true; + return .{ .success = result.* }; + } + + return .{ .not_found = {} }; + } + } + + var _paths = [_]string{ pkg_dir_info.abs_path, esm.subpath }; + const abs_path = r.fs.absBuf(&_paths, &node_modules_check_buf); + if (r.debug_logs) |*debug| { + debug.addNoteFmt("Checking for a package in the directory \"{s}\"", .{abs_path}); + } + + if (r.loadAsFileOrDirectory(abs_path, kind)) |*res| { + res.is_node_module = true; + return .{ .success = res.* }; + } + } + } else |err| { + return .{ .failure = err }; + } + } + } + // Mostly to cut scope, we don't resolve `NODE_PATH` environment variable. // But also: https://github.com/nodejs/node/issues/38128#issuecomment-814969356 + return .{ .not_found = {} }; + } + fn dirInfoForResolution( + r: *ThisResolver, + dir_path: []const u8, + package_id: Install.PackageID, + ) !?*DirInfo { + std.debug.assert(r.package_manager != null); - return null; + var dir_cache_info_result = r.dir_cache.getOrPut(dir_path) catch unreachable; + if (dir_cache_info_result.status == .exists) { + // we've already looked up this package before + return r.dir_cache.atIndex(dir_cache_info_result.index).?; + } + var rfs = &r.fs.fs; + var cached_dir_entry_result = rfs.entries.getOrPut(dir_path) catch unreachable; + + var dir_entries_option: *Fs.FileSystem.RealFS.EntriesOption = undefined; + var needs_iter: bool = true; + var open_dir = std.fs.openDirAbsolute(dir_path, .{ .iterate = true }) catch |err| { + switch (err) { + error.FileNotFound => unreachable, + else => { + // TODO: handle this error better + r.log.addErrorFmt(null, logger.Loc.Empty, r.allocator, "Unable to open directory: {s}", .{std.mem.span(@errorName(err))}) catch unreachable; + return err; + }, + } + }; + + if (rfs.entries.atIndex(cached_dir_entry_result.index)) |cached_entry| { + if (cached_entry.* == .entries) { + dir_entries_option = cached_entry; + needs_iter = false; + } + } + + if (needs_iter) { + const allocator = r.fs.allocator; + dir_entries_option = rfs.entries.put(&cached_dir_entry_result, .{ + .entries = Fs.FileSystem.DirEntry.init(dir_path), + }) catch unreachable; + + if (FeatureFlags.store_file_descriptors) { + Fs.FileSystem.setMaxFd(open_dir.fd); + dir_entries_option.entries.fd = open_dir.fd; + } + var dir_iterator = open_dir.iterate(); + while (dir_iterator.next() catch null) |_value| { + dir_entries_option.entries.addEntry(_value, allocator, void, void{}) catch unreachable; + } + } + + // We must initialize it as empty so that the result index is correct. + // This is important so that browser_scope has a valid index. + var dir_info_ptr = r.dir_cache.put(&dir_cache_info_result, DirInfo{}) catch unreachable; + + try r.dirInfoUncached( + dir_info_ptr, + dir_path, + dir_entries_option, + dir_cache_info_result, + cached_dir_entry_result.index, + // Packages in the global disk cache are top-level, we shouldn't try + // to check for a parent package.json + null, + allocators.NotFound, + open_dir.fd, + package_id, + ); + return dir_info_ptr; + } + + const DependencyToResolve = union(enum) { + not_found: void, + pending: PendingResolution, + failure: anyerror, + resolution: Resolution, + }; + + fn enqueueDependencyToResolve( + r: *ThisResolver, + package_json_: ?*PackageJSON, + esm: ESModule.Package, + behavior: Dependency.Behavior, + input_package_id_: *Install.PackageID, + version: Dependency.Version, + ) DependencyToResolve { + if (r.debug_logs) |*debug| { + debug.addNoteFmt("Enqueueing pending dependency \"{s}@{s}\"", .{ esm.name, esm.version }); + } + + const input_package_id = input_package_id_.*; + var pm = r.getPackageManager(); + if (comptime Environment.allow_assert) { + // we should never be trying to resolve a dependency that is already resolved + std.debug.assert(pm.lockfile.resolve(esm.name, version) == null); + } + + // Add the containing package to the lockfile + + var package: Package = .{}; + + if (pm.lockfile.packages.len == 0 and input_package_id == Install.invalid_package_id) { + if (package_json_) |package_json| { + package = Package.fromPackageJSON( + pm.allocator, + pm.lockfile, + r.log, + package_json, + Install.Features{ + .dev_dependencies = true, + .is_main = true, + .dependencies = true, + .optional_dependencies = true, + }, + ) catch |err| { + return .{ .failure = err }; + }; + + package.resolution = .{ + .tag = .root, + .value = .{ .root = {} }, + }; + + package = pm.lockfile.appendPackage(package) catch |err| { + return .{ .failure = err }; + }; + package_json.package_manager_package_id = package.meta.id; + } else { + // we're resolving an unknown package + // the unknown package is the root package + package = Package{ + .name = Semver.String.init("", ""), + }; + package.resolution = .{ + .tag = .root, + .value = .{ .root = {} }, + }; + package = pm.lockfile.appendPackage(package) catch |err| { + return .{ .failure = err }; + }; + } + } + + if (r.opts.prefer_offline_install) { + if (pm.resolveFromDiskCache(esm.name, version)) |package_id| { + input_package_id_.* = package_id; + return .{ .resolution = pm.lockfile.packages.items(.resolution)[package_id] }; + } + } + + if (input_package_id == Install.invalid_package_id or input_package_id == 0) { + + // All packages are enqueued to the root + // because we download all the npm package dependencies + switch (pm.enqueueDependencyToRoot(esm.name, esm.version, version, behavior)) { + .resolution => |result| { + input_package_id_.* = result.package_id; + return .{ .resolution = result.resolution }; + }, + .pending => |id| { + var builder = Semver.String.Builder{}; + esm.count(&builder); + builder.allocate(pm.allocator) catch unreachable; + const cloned = esm.clone(&builder); + + return .{ + .pending = .{ + .esm = cloned, + .dependency = version, + .resolution_id = Install.invalid_package_id, + .root_dependency_id = id, + .string_buf = builder.allocatedSlice(), + .tag = .resolve, + }, + }; + }, + .not_found => { + return .{ .not_found = {} }; + }, + .failure => |err| { + return .{ .failure = err }; + }, + } + } + + bun.unreachablePanic("TODO: implement enqueueDependencyToResolve for non-root packages", .{}); } fn handleESMResolution(r: *ThisResolver, esm_resolution_: ESModule.Resolution, abs_package_path: string, kind: ast.ImportKind, package_json: *PackageJSON) ?MatchResult { @@ -1409,13 +2007,22 @@ pub const Resolver = struct { } } - pub fn resolveWithoutRemapping(r: *ThisResolver, source_dir_info: *DirInfo, import_path: string, kind: ast.ImportKind) ?MatchResult { + pub fn resolveWithoutRemapping( + r: *ThisResolver, + source_dir_info: *DirInfo, + import_path: string, + kind: ast.ImportKind, + global_cache: GlobalCache, + ) MatchResult.Union { if (isPackagePath(import_path)) { - return r.loadNodeModules(import_path, kind, source_dir_info, false); + return r.loadNodeModules(import_path, kind, source_dir_info, global_cache, false); } else { const paths = [_]string{ source_dir_info.abs_path, import_path }; var resolved = r.fs.absBuf(&paths, &resolve_without_remapping_buf); - return r.loadAsFileOrDirectory(resolved, kind); + if (r.loadAsFileOrDirectory(resolved, kind)) |result| { + return .{ .success = result }; + } + return .{ .not_found = {} }; } } @@ -1469,12 +2076,34 @@ pub const Resolver = struct { return bin_folders.constSlice(); } - pub fn parsePackageJSON(r: *ThisResolver, file: string, dirname_fd: StoredFileDescriptorType) !?*PackageJSON { + pub fn parsePackageJSON( + r: *ThisResolver, + file: string, + dirname_fd: StoredFileDescriptorType, + package_id: ?Install.PackageID, + comptime allow_dependencies: bool, + ) !?*PackageJSON { var pkg: PackageJSON = undefined; if (!r.care_about_scripts) { - pkg = PackageJSON.parse(ThisResolver, r, file, dirname_fd, true, false) orelse return null; + pkg = PackageJSON.parse( + r, + file, + dirname_fd, + package_id, + true, + if (allow_dependencies) .local else .none, + false, + ) orelse return null; } else { - pkg = PackageJSON.parse(ThisResolver, r, file, dirname_fd, true, true) orelse return null; + pkg = PackageJSON.parse( + r, + file, + dirname_fd, + package_id, + true, + if (allow_dependencies) .local else .none, + true, + ) orelse return null; } var _pkg = try r.allocator.create(PackageJSON); @@ -1755,6 +2384,7 @@ pub const Resolver = struct { r.dir_cache.atIndex(top_parent.index), top_parent.index, open_dir.fd, + null, ); if (queue_slice.len == 0) { @@ -1779,7 +2409,7 @@ pub const Resolver = struct { // official TypeScript compiler pub fn matchTSConfigPaths(r: *ThisResolver, tsconfig: *const TSConfigJSON, path: string, kind: ast.ImportKind) ?MatchResult { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Matching \"{s}\" against \"paths\" in \"{s}\"", .{ path, tsconfig.abs_path }) catch unreachable; + debug.addNoteFmt("Matching \"{s}\" against \"paths\" in \"{s}\"", .{ path, tsconfig.abs_path }); } var abs_base_url = tsconfig.base_url_for_paths; @@ -1792,7 +2422,7 @@ pub const Resolver = struct { } if (r.debug_logs) |*debug| { - debug.addNoteFmt("Using \"{s}\" as \"baseURL\"", .{abs_base_url}) catch unreachable; + debug.addNoteFmt("Using \"{s}\" as \"baseURL\"", .{abs_base_url}); } // Check for exact matches first @@ -1857,7 +2487,7 @@ pub const Resolver = struct { // prefix. This matches the behavior of the TypeScript compiler. if (longest_match_prefix_length > -1) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Found a fuzzy match for \"{s}*{s}\" in \"paths\"", .{ longest_match.prefix, longest_match.suffix }) catch unreachable; + debug.addNoteFmt("Found a fuzzy match for \"{s}*{s}\" in \"paths\"", .{ longest_match.prefix, longest_match.suffix }); } for (longest_match.original_paths) |original_path| { @@ -1920,7 +2550,7 @@ pub const Resolver = struct { std.mem.copy(u8, TemporaryBuffer.ExtensionPathBuf[cleaned.len .. cleaned.len + ext.len], ext); const new_path = TemporaryBuffer.ExtensionPathBuf[0 .. cleaned.len + ext.len]; // if (r.debug_logs) |*debug| { - // debug.addNoteFmt("Checking for \"{s}\" ", .{new_path}) catch {}; + // debug.addNoteFmt("Checking for \"{s}\" ", .{new_path}); // } if (map.get(new_path)) |_remapped| { this.remapped = _remapped; @@ -1950,7 +2580,7 @@ pub const Resolver = struct { std.mem.copy(u8, TemporaryBuffer.ExtensionPathBuf[index_path.len .. index_path.len + ext.len], ext); const new_path = TemporaryBuffer.ExtensionPathBuf[0 .. index_path.len + ext.len]; // if (r.debug_logs) |*debug| { - // debug.addNoteFmt("Checking for \"{s}\" ", .{new_path}) catch {}; + // debug.addNoteFmt("Checking for \"{s}\" ", .{new_path}); // } if (map.get(new_path)) |_remapped| { this.remapped = _remapped; @@ -2032,7 +2662,7 @@ pub const Resolver = struct { // package and the parent package. const isInSamePackage = brk: { const parent = dir_info.getParent() orelse break :brk true; - break :brk !parent.is_node_modules; + break :brk !parent.isNodeModules(); }; if (isInSamePackage) { @@ -2054,13 +2684,13 @@ pub const Resolver = struct { var field_rel_path = _field_rel_path; // Is this a directory? if (r.debug_logs) |*debug| { - debug.addNoteFmt("Found main field \"{s}\" with path \"{s}\"", .{ field, field_rel_path }) catch {}; - debug.increaseIndent() catch {}; + debug.addNoteFmt("Found main field \"{s}\" with path \"{s}\"", .{ field, field_rel_path }); + debug.increaseIndent(); } defer { if (r.debug_logs) |*debug| { - debug.decreaseIndent() catch {}; + debug.decreaseIndent(); } } @@ -2142,7 +2772,7 @@ pub const Resolver = struct { }; if (r.debug_logs) |*debug| { - debug.addNoteFmt("Found file: \"{s}\"", .{out_buf}) catch unreachable; + debug.addNoteFmt("Found file: \"{s}\"", .{out_buf}); } if (dir_info.package_json) |package_json| { @@ -2165,7 +2795,7 @@ pub const Resolver = struct { } if (r.debug_logs) |*debug| { - debug.addNoteFmt("Failed to find file: \"{s}/{s}\"", .{ path, base }) catch unreachable; + debug.addNoteFmt("Failed to find file: \"{s}/{s}\"", .{ path, base }); } } @@ -2264,13 +2894,13 @@ pub const Resolver = struct { // Is this a directory? if (r.debug_logs) |*debug| { - debug.addNoteFmt("Attempting to load \"{s}\" as a directory", .{path}) catch {}; - debug.increaseIndent() catch {}; + debug.addNoteFmt("Attempting to load \"{s}\" as a directory", .{path}); + debug.increaseIndent(); } defer { if (r.debug_logs) |*debug| { - debug.decreaseIndent() catch {}; + debug.decreaseIndent(); } } @@ -2290,13 +2920,13 @@ pub const Resolver = struct { const auto_main = r.opts.main_fields.ptr == options.Platform.DefaultMainFields.get(r.opts.platform).ptr; if (r.debug_logs) |*debug| { - debug.addNoteFmt("Searching for main fields in \"{s}\"", .{pkg_json.source.path.text}) catch {}; + debug.addNoteFmt("Searching for main fields in \"{s}\"", .{pkg_json.source.path.text}); } for (main_field_keys) |key| { const field_rel_path = (main_field_values.get(key)) orelse { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Did not find main field \"{s}\"", .{key}) catch {}; + debug.addNoteFmt("Did not find main field \"{s}\"", .{key}); } continue; }; @@ -2331,9 +2961,9 @@ pub const Resolver = struct { // same time. if (kind != ast.ImportKind.require) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Resolved to \"{s}\" using the \"module\" field in \"{s}\"", .{ auto_main_result.path_pair.primary.text, pkg_json.source.key_path.text }) catch {}; + debug.addNoteFmt("Resolved to \"{s}\" using the \"module\" field in \"{s}\"", .{ auto_main_result.path_pair.primary.text, pkg_json.source.key_path.text }); - debug.addNoteFmt("The fallback path in case of \"require\" is {s}", .{auto_main_result.path_pair.primary.text}) catch {}; + debug.addNoteFmt("The fallback path in case of \"require\" is {s}", .{auto_main_result.path_pair.primary.text}); } return MatchResult{ @@ -2352,7 +2982,7 @@ pub const Resolver = struct { auto_main_result.path_pair.primary.text, key, pkg_json.source.key_path.text, - }) catch {}; + }); } var _auto_main_result = auto_main_result; _auto_main_result.package_json = package_json; @@ -2380,12 +3010,12 @@ pub const Resolver = struct { var rfs: *Fs.FileSystem.RealFS = &r.fs.fs; if (r.debug_logs) |*debug| { - debug.addNoteFmt("Attempting to load \"{s}\" as a file", .{path}) catch {}; - debug.increaseIndent() catch {}; + debug.addNoteFmt("Attempting to load \"{s}\" as a file", .{path}); + debug.increaseIndent(); } defer { if (r.debug_logs) |*debug| { - debug.decreaseIndent() catch {}; + debug.decreaseIndent(); } } @@ -2420,13 +3050,13 @@ pub const Resolver = struct { // Try the plain path without any extensions if (r.debug_logs) |*debug| { - debug.addNoteFmt("Checking for file \"{s}\" ", .{base}) catch {}; + debug.addNoteFmt("Checking for file \"{s}\" ", .{base}); } if (entries.get(base)) |query| { if (query.entry.kind(rfs) == .file) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Found file \"{s}\" ", .{base}) catch {}; + debug.addNoteFmt("Found file \"{s}\" ", .{base}); } const abs_path = brk: { @@ -2455,13 +3085,13 @@ pub const Resolver = struct { const file_name = buffer[path.len - base.len .. buffer.len]; if (r.debug_logs) |*debug| { - debug.addNoteFmt("Checking for file \"{s}\" ", .{buffer}) catch {}; + debug.addNoteFmt("Checking for file \"{s}\" ", .{buffer}); } if (entries.get(file_name)) |query| { if (query.entry.kind(rfs) == .file) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Found file \"{s}\" ", .{buffer}) catch {}; + debug.addNoteFmt("Found file \"{s}\" ", .{buffer}); } // now that we've found it, we allocate it. @@ -2513,7 +3143,7 @@ pub const Resolver = struct { if (entries.get(buffer)) |query| { if (query.entry.kind(rfs) == .file) { if (r.debug_logs) |*debug| { - debug.addNoteFmt("Rewrote to \"{s}\" ", .{buffer}) catch {}; + debug.addNoteFmt("Rewrote to \"{s}\" ", .{buffer}); } return LoadResult{ @@ -2538,14 +3168,14 @@ pub const Resolver = struct { } } if (r.debug_logs) |*debug| { - debug.addNoteFmt("Failed to rewrite \"{s}\" ", .{base}) catch {}; + debug.addNoteFmt("Failed to rewrite \"{s}\" ", .{base}); } } } } if (r.debug_logs) |*debug| { - debug.addNoteFmt("Failed to find \"{s}\" ", .{path}) catch {}; + debug.addNoteFmt("Failed to find \"{s}\" ", .{path}); } if (comptime FeatureFlags.watch_directories) { @@ -2568,6 +3198,7 @@ pub const Resolver = struct { parent: ?*DirInfo, parent_index: allocators.IndexType, fd: FileDescriptorType, + package_id: ?Install.PackageID, ) anyerror!void { var result = _result; @@ -2587,18 +3218,18 @@ pub const Resolver = struct { // base must if (base.len > 1 and base[base.len - 1] == std.fs.path.sep) base = base[0 .. base.len - 1]; - info.is_node_modules = strings.eqlComptime(base, "node_modules"); + info.flags.setPresent(.is_node_modules, strings.eqlComptime(base, "node_modules")); // if (entries != null) { - if (!info.is_node_modules) { + if (!info.isNodeModules()) { if (entries.getComptimeQuery("node_modules")) |entry| { - info.has_node_modules = (entry.entry.kind(rfs)) == .dir; + info.flags.setPresent(.has_node_modules, (entry.entry.kind(rfs)) == .dir); } } if (r.care_about_bin_folder) { append_bin_dir: { - if (info.has_node_modules) { + if (info.hasNodeModules()) { if (entries.hasComptimeQuery("node_modules")) { if (!bin_folders_loaded) { bin_folders_loaded = true; @@ -2622,7 +3253,7 @@ pub const Resolver = struct { } } - if (info.is_node_modules) { + if (info.isNodeModules()) { if (entries.getComptimeQuery(".bin")) |q| { if (q.entry.kind(rfs) == .dir) { if (!bin_folders_loaded) { @@ -2663,9 +3294,14 @@ pub const Resolver = struct { if (parent_package_json.name.len > 0 or r.care_about_bin_folder) { info.enclosing_package_json = parent_package_json; } + + if (parent_package_json.dependencies.map.count() > 0 or parent_package_json.package_manager_package_id != Install.invalid_package_id) { + info.package_json_for_dependencies = parent_package_json; + } } info.enclosing_package_json = info.enclosing_package_json orelse parent.?.enclosing_package_json; + info.package_json_for_dependencies = info.package_json_for_dependencies orelse parent.?.package_json_for_dependencies; // Make sure "absRealPath" is the real path of the directory (resolving any symlinks) if (!r.opts.preserve_symlinks) { @@ -2677,7 +3313,7 @@ pub const Resolver = struct { var symlink = entry.symlink(rfs); if (symlink.len > 0) { if (r.debug_logs) |*logs| { - try logs.addNote(std.fmt.allocPrint(r.allocator, "Resolved symlink \"{s}\" to \"{s}\"", .{ path, symlink }) catch unreachable); + logs.addNote(std.fmt.allocPrint(r.allocator, "Resolved symlink \"{s}\" to \"{s}\"", .{ path, symlink }) catch unreachable); } info.abs_real_path = symlink; } else if (parent.?.abs_real_path.len > 0) { @@ -2686,7 +3322,7 @@ pub const Resolver = struct { symlink = r.fs.dirname_store.append(string, r.fs.absBuf(&parts, &dir_info_uncached_filename_buf)) catch unreachable; if (r.debug_logs) |*logs| { - try logs.addNote(std.fmt.allocPrint(r.allocator, "Resolved symlink \"{s}\" to \"{s}\"", .{ path, symlink }) catch unreachable); + logs.addNote(std.fmt.allocPrint(r.allocator, "Resolved symlink \"{s}\" to \"{s}\"", .{ path, symlink }) catch unreachable); } lookup.entry.cache.symlink = PathString.init(symlink); info.abs_real_path = symlink; @@ -2700,7 +3336,10 @@ pub const Resolver = struct { if (entries.getComptimeQuery("package.json")) |lookup| { const entry = lookup.entry; if (entry.kind(rfs) == .file) { - info.package_json = r.parsePackageJSON(path, if (FeatureFlags.store_file_descriptors) fd else 0) catch null; + info.package_json = if (r.usePackageManager() and !info.hasNodeModules() and !info.isNodeModules()) + r.parsePackageJSON(path, if (FeatureFlags.store_file_descriptors) fd else 0, package_id, true) catch null + else + r.parsePackageJSON(path, if (FeatureFlags.store_file_descriptors) fd else 0, null, false) catch null; if (info.package_json) |pkg| { if (pkg.browser_map.count() > 0) { @@ -2711,10 +3350,13 @@ pub const Resolver = struct { if (pkg.name.len > 0 or r.care_about_bin_folder) info.enclosing_package_json = pkg; + if (pkg.dependencies.map.count() > 0 or pkg.package_manager_package_id != Install.invalid_package_id) + info.package_json_for_dependencies = pkg; + if (r.debug_logs) |*logs| { logs.addNoteFmt("Resolved package.json in \"{s}\"", .{ path, - }) catch unreachable; + }); } } } @@ -2836,3 +3478,51 @@ pub const RootPathPair = struct { base_path: string, package_json: *const PackageJSON, }; + +pub const GlobalCache = enum { + allow_install, + read_only, + auto, + force, + fallback, + disable, + + pub const Map = bun.ComptimeStringMap(GlobalCache, .{ + .{ "auto", GlobalCache.auto }, + .{ "force", GlobalCache.force }, + .{ "disable", GlobalCache.disable }, + .{ "fallback", GlobalCache.fallback }, + }); + + pub fn allowVersionSpecifier(this: GlobalCache) bool { + return this == .force; + } + + pub fn canUse(this: GlobalCache, has_a_node_modules_folder: bool) bool { + // When there is a node_modules folder, we default to false + // When there is NOT a node_modules folder, we default to true + // That is the difference between these two branches. + if (has_a_node_modules_folder) { + return switch (this) { + .fallback, .allow_install, .force => true, + .read_only, .disable, .auto => false, + }; + } else { + return switch (this) { + .fallback, .allow_install, .auto, .force => true, + .read_only, .disable => false, + }; + } + } + + pub fn isEnabled(this: GlobalCache) bool { + return this != .disable; + } + + pub fn canInstall(this: GlobalCache) bool { + return switch (this) { + .auto, .allow_install, .force, .fallback => true, + else => false, + }; + } +}; |