Skip to content

Commit

Permalink
fix(package-manager): allow custom URL for packageManager version (#…
Browse files Browse the repository at this point in the history
…9624)

Fixes #9611

### Description

<!--
  ✍️ Write a short summary of your work.
  If necessary, include relevant screenshots.
-->

Field `packageManager` of `package.json` can now use a custom URL, e.g.
`npm@https://registry.npmjs.org/some-npm-fork/-/some-npm-fork-1.0.0.tgz`

(I also fixed some minor formatting issues in `CONTRIBUTING.md`)

### Testing Instructions

<!--
  Give a quick description of steps to test your changes.
-->

Added both an integration test and a unit test.

---------

Co-authored-by: Chris Olszewski <[email protected]>
  • Loading branch information
pkerschbaum and chris-olszewski authored Dec 16, 2024
1 parent ffe5be9 commit db580f2
Show file tree
Hide file tree
Showing 4 changed files with 68 additions and 32 deletions.
6 changes: 3 additions & 3 deletions CONTRIBUTING.md
Original file line number Diff line number Diff line change
Expand Up @@ -31,8 +31,8 @@ You will need to have these dependences installed on your machine to work on thi

- For running tests locally, `jq` and `zstd` are also required.
- macOS: `brew install jq zstd`
- Linux: ``sudo apt update && sudo apt install jq zstd`
- Windows: `choco install jq zstandard
- Linux: `sudo apt update && sudo apt install jq zstd`
- Windows: `choco install jq zstandard`
- On Linux, ensure LLD (LLVM Linker) is installed, as it's not installed by default on many Linux distributions (e.g. `apt install lld`).

## Structure of the repository
Expand Down Expand Up @@ -64,7 +64,7 @@ out of the box. If you wish to select `native-tls`, you may do so by running `ca
## Running tests

> [!IMPORTANT]
> You will need to have `jq` and `zstd` installed on your system in order to run tests. See [General dDependencies](#general-dependencies) for instructions on how to install these tools.
> You will need to have `jq` and `zstd` installed on your system in order to run tests. See [General dependencies](#general-dependencies) for instructions on how to install these tools.
First, install Turborepo globally with your package manager of choice. For instance, with npm, `npm install -g turbo`. This will install the `turbo` binary in your system's `PATH`, making it globally available.

Expand Down
2 changes: 1 addition & 1 deletion crates/turborepo-repository/src/discovery.rs
Original file line number Diff line number Diff line change
Expand Up @@ -120,7 +120,7 @@ impl PackageDiscoveryBuilder for LocalPackageDiscoveryBuilder {
if self.allow_missing_package_manager {
PackageManager::read_or_detect_package_manager(&package_json, &self.repo_root)?
} else {
PackageManager::get_package_manager(&package_json)?
PackageManager::get_package_manager(&self.repo_root, &package_json)?
}
}
};
Expand Down
90 changes: 63 additions & 27 deletions crates/turborepo-repository/src/package_manager/mod.rs
Original file line number Diff line number Diff line change
Expand Up @@ -199,7 +199,7 @@ impl From<std::convert::Infallible> for Error {
}

static PACKAGE_MANAGER_PATTERN: Lazy<Regex> =
lazy_regex!(r"(?P<manager>bun|npm|pnpm|yarn)@(?P<version>\d+\.\d+\.\d+(-.+)?)");
lazy_regex!(r"(?P<manager>bun|npm|pnpm|yarn)@(?P<version>\d+\.\d+\.\d+(-.+)?|https?://.+)");

impl PackageManager {
pub fn supported_managers() -> &'static [Self] {
Expand Down Expand Up @@ -321,34 +321,59 @@ impl PackageManager {
}

/// Try to extract the package manager from package.json.
pub fn get_package_manager(package_json: &PackageJson) -> Result<Self, Error> {
Self::read_package_manager(package_json)
/// Package Manager will be read from package.json only using the file
/// system if the version is a URL and we need to invoke the binary it
/// points to for version information.
pub fn get_package_manager(
repo_root: &AbsoluteSystemPath,
package_json: &PackageJson,
) -> Result<Self, Error> {
Self::read_package_manager(repo_root, package_json)
}

// Attempts to read the package manager from the package.json
fn read_package_manager(pkg: &PackageJson) -> Result<Self, Error> {
fn read_package_manager(
repo_root: &AbsoluteSystemPath,
pkg: &PackageJson,
) -> Result<Self, Error> {
let Some(package_manager) = &pkg.package_manager else {
return Err(Error::MissingPackageManager);
};

let (manager, version) = Self::parse_package_manager_string(package_manager)?;
let version = version.parse().map_err(|err: SemverError| {
let (span, text) = package_manager.span_and_text("package.json");
Error::InvalidVersion {
explanation: err.to_string(),
span,
text,
// if version is a https attempt to check that instead
if version.starts_with("http") {
match manager {
"npm" => Ok(PackageManager::Npm),
"bun" => Ok(PackageManager::Bun),
"yarn" => Ok(YarnDetector::new(repo_root)
.next()
.ok_or_else(|| Error::MissingPackageManager)??),
"pnpm" => Ok(PnpmDetector::new(repo_root)
.next()
.ok_or_else(|| Error::MissingPackageManager)??),
_ => unreachable!(
"found invalid package manager even though regex should have caught it"
),
}
} else {
let version = version.parse().map_err(|err: SemverError| {
let (span, text) = package_manager.span_and_text("package.json");
Error::InvalidVersion {
explanation: err.to_string(),
span,
text,
}
})?;
match manager {
"npm" => Ok(PackageManager::Npm),
"bun" => Ok(PackageManager::Bun),
"yarn" => Ok(YarnDetector::detect_berry_or_yarn(&version)?),
"pnpm" => Ok(PnpmDetector::detect_pnpm6_or_pnpm(&version)?),
_ => unreachable!(
"found invalid package manager even though regex should have caught it"
),
}
})?;

match manager {
"npm" => Ok(PackageManager::Npm),
"bun" => Ok(PackageManager::Bun),
"yarn" => Ok(YarnDetector::detect_berry_or_yarn(&version)?),
"pnpm" => Ok(PnpmDetector::detect_pnpm6_or_pnpm(&version)?),
_ => unreachable!(
"found invalid package manager even though regex should have caught it"
),
}
}

Expand Down Expand Up @@ -380,7 +405,8 @@ impl PackageManager {
package_json: &PackageJson,
repo_root: &AbsoluteSystemPath,
) -> Result<Self, Error> {
Self::get_package_manager(package_json).or_else(|_| Self::detect_package_manager(repo_root))
Self::get_package_manager(repo_root, package_json)
.or_else(|_| Self::detect_package_manager(repo_root))
}

pub(crate) fn parse_package_manager_string(
Expand Down Expand Up @@ -526,6 +552,7 @@ mod tests {
use std::collections::HashSet;

use pretty_assertions::assert_eq;
use tempfile::TempDir;

use super::*;

Expand Down Expand Up @@ -697,6 +724,13 @@ mod tests {
expected_version: "1.0.1".to_owned(),
expected_error: false,
},
TestCase {
name: "supports custom URL".to_owned(),
package_manager: Spanned::new("npm@https://some-npm-fork".to_owned()),
expected_manager: "npm".to_owned(),
expected_version: "https://some-npm-fork".to_owned(),
expected_error: false,
},
];

for case in tests {
Expand All @@ -713,31 +747,33 @@ mod tests {

#[test]
fn test_read_package_manager() -> Result<(), Error> {
let dir = TempDir::new()?;
let repo_root = AbsoluteSystemPath::from_std_path(dir.path())?;
let mut package_json = PackageJson {
package_manager: Some(Spanned::new("[email protected]".to_string())),
..Default::default()
};
let package_manager = PackageManager::read_package_manager(&package_json)?;
let package_manager = PackageManager::read_package_manager(repo_root, &package_json)?;
assert_eq!(package_manager, PackageManager::Npm);

package_json.package_manager = Some(Spanned::new("[email protected]".to_string()));
let package_manager = PackageManager::read_package_manager(&package_json)?;
let package_manager = PackageManager::read_package_manager(repo_root, &package_json)?;
assert_eq!(package_manager, PackageManager::Berry);

package_json.package_manager = Some(Spanned::new("[email protected]".to_string()));
let package_manager = PackageManager::read_package_manager(&package_json)?;
let package_manager = PackageManager::read_package_manager(repo_root, &package_json)?;
assert_eq!(package_manager, PackageManager::Yarn);

package_json.package_manager = Some(Spanned::new("[email protected]".to_string()));
let package_manager = PackageManager::read_package_manager(&package_json)?;
let package_manager = PackageManager::read_package_manager(repo_root, &package_json)?;
assert_eq!(package_manager, PackageManager::Pnpm6);

package_json.package_manager = Some(Spanned::new("[email protected]".to_string()));
let package_manager = PackageManager::read_package_manager(&package_json)?;
let package_manager = PackageManager::read_package_manager(repo_root, &package_json)?;
assert_eq!(package_manager, PackageManager::Pnpm);

package_json.package_manager = Some(Spanned::new("[email protected]".to_string()));
let package_manager = PackageManager::read_package_manager(&package_json)?;
let package_manager = PackageManager::read_package_manager(repo_root, &package_json)?;
assert_eq!(package_manager, PackageManager::Bun);

Ok(())
Expand Down
2 changes: 1 addition & 1 deletion turborepo-tests/integration/tests/invalid-package-json.t
Original file line number Diff line number Diff line change
Expand Up @@ -25,7 +25,7 @@ Build should fail due to invalid packageManager field (sed removes the square br
x could not resolve workspaces
`-> could not parse the packageManager field in package.json, expected to
match regular expression (?P<manager>bun|npm|pnpm|yarn)@(?P<version>\d+
\.\d+\.\d+(-.+)?)
\.\d+\.\d+(-.+)?|https?://.+)
,-\1
5 | },
6 | "packageManager": "[email protected]",
Expand Down

0 comments on commit db580f2

Please sign in to comment.