Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

node: Add support for Git sources with lockfile v2 processing #351

Open
wants to merge 1 commit into
base: master
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
73 changes: 51 additions & 22 deletions node/flatpak_node_generator/providers/npm.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,11 @@
'application/vnd.npm.install-v1+json; q=1.0, application/json; q=0.8, */*'
)

# TODO Once lockfile v2 syntax support is complete, use _process_packages_v2 for
# both v2 and v3 lockfiles
_LOCKFILE_V1_VERSIONS = {1, 2}
_LOCKFILE_V2_VERSIONS = {3}


class NpmLockfileProvider(LockfileProvider):
_ALIAS_RE = re.compile(r'^npm:(.[^@]*)@(.*)$')
Expand Down Expand Up @@ -106,6 +111,16 @@ def _process_packages_v2(
continue

name = info.get('name')
inferred_name = False

# NOTE We can't reliably determine the package name from the lockfile v2 syntax,
# but we need it for registry queries and special source processing;
# If we didn't get the package name at this point, try determining it from
# the install path as the last resort
if name is None:
inferred_name = True
path_list = install_path.split('/')
name = '/'.join(path_list[-path_list[::-1].index('node_modules') :])

source: PackageSource
package_json_path = lockfile.parent / install_path / 'package.json'
Expand All @@ -114,7 +129,8 @@ def _process_packages_v2(
and package_json_path.exists()
):
source = LocalSource(path=install_path)
if name is None:
if inferred_name:
# Prefer getting the name directly instead of inferring it.
with package_json_path.open('rb') as fp:
name = json.load(fp)['name']
elif 'resolved' in info:
Expand All @@ -130,23 +146,12 @@ def _process_packages_v2(
integrity=integrity, resolved=info['resolved']
)
elif resolved_url.scheme.startswith('git+'):
raise NotImplementedError(
'Git sources in lockfile v2 format are not supported yet'
f' (package {install_path} in {lockfile})'
)
source = self.parse_git_source(info['resolved'], name)
else:
raise NotImplementedError(
f"Don't know how to handle package {install_path} in {lockfile}"
)

# NOTE We can't reliably determine the package name from the lockfile v2 syntax,
# but we need it for registry queries and special source processing;
# If we didn't get the package name at this point, try determining it from
# the install path as the last resort
if name is None:
path_list = install_path.split('/')
name = '/'.join(path_list[-path_list[::-1].index('node_modules') :])

yield Package(
name=name,
version=info.get('version'),
Expand All @@ -158,11 +163,9 @@ def process_lockfile(self, lockfile: Path) -> Iterator[Package]:
with open(lockfile) as fp:
data = json.load(fp)

# TODO Once lockfile v2 syntax support is complete, use _process_packages_v2
# for both v2 and v2 lockfiles
if data['lockfileVersion'] in {1, 2}:
if data['lockfileVersion'] in _LOCKFILE_V1_VERSIONS:
yield from self._process_packages_v1(lockfile, data)
elif data['lockfileVersion'] in {3}:
elif data['lockfileVersion'] in _LOCKFILE_V2_VERSIONS:
yield from self._process_packages_v2(lockfile, data)
else:
raise NotImplementedError(
Expand Down Expand Up @@ -426,14 +429,21 @@ def _finalize(self) -> None:

if self.git_sources:
# Generate jq scripts to patch the package*.json files.

# NOTE: In package.json, we always assign to .value regardless of
# $match_against, because even if we're matching against the keys / package
# names, what we actually want to set is the value / version. With
# package-lock.json, we are in fact matching the values themselves,
# so we re-assign to it directly.
scripts = {
'package.json': r"""
walk(
if type == "object"
then
to_entries | map(
if (.value | type == "string") and $data[.value]
then .value = "git+file:\($buildroot)/\($data[.value])"
.[$match_against] as $match |
if ($match | type == "string") and $data[$match]
then .value = "git+file:\($buildroot)/\($data[$match])"
else .
end
) | from_entries
Expand All @@ -443,16 +453,34 @@ def _finalize(self) -> None:
""",
'package-lock.json': r"""
walk(
if type == "object" and (.version | type == "string") and $data[.version]
then
.version = "git+file:\($buildroot)/\($data[.version])"
if type == "object" then
.[$match_against] as $match |
if ($match | type == "string") and $data[$match]
then .[$match_against] =
"git+file:\($buildroot)/\($data[$match])"
else .
end
else .
end
)
""",
}

for lockfile, sources in self.git_sources.items():
with lockfile.open() as fp:
version = json.load(fp)['lockfileVersion']

if version in _LOCKFILE_V2_VERSIONS:
match_against = {
'package.json': 'key',
'package-lock.json': 'resolved',
}
else:
match_against = {
'package.json': 'value',
'package-lock.json': 'version',
}

prefix = self.relative_lockfile_dir(lockfile)
data: Dict[str, Dict[str, str]] = {
'package.json': {},
Expand Down Expand Up @@ -486,6 +514,7 @@ def _finalize(self) -> None:
patch_commands[lockfile].append(
'jq'
' --arg buildroot "$FLATPAK_BUILDER_BUILDDIR"'
f' --arg match_against {match_against[filename]}'
f' --argjson data {shlex.quote(json_data)}'
f' {shlex.quote(script)} {target}'
f' > {target}.new'
Expand Down
3 changes: 1 addition & 2 deletions node/tests/data/packages/minimal-git/package-lock.v3.json
Original file line number Diff line number Diff line change
Expand Up @@ -13,8 +13,7 @@
},
"node_modules/nop": {
"version": "1.0.0",
"resolved": "git+ssh://git@github.com/supershabam/nop.git#f110e75f62cfe3bf4468ac3b74e3dc72ab9ae4bf",
"integrity": "sha512-xCwdA7C4QIORvTMytKHMlkEN6axJGimR0gv5vgjKpEKRvQrPOwhjJnrZEcd5g0LP+7IY38+TY7MP59HRY6gcwA==",
"resolved": "git+https://git@github.com/supershabam/nop.git#f110e75f62cfe3bf4468ac3b74e3dc72ab9ae4bf",
"license": "MIT"
}
}
Expand Down
3 changes: 0 additions & 3 deletions node/tests/test_providers.py
Original file line number Diff line number Diff line change
Expand Up @@ -13,9 +13,6 @@ async def test_minimal_git(
provider_factory_spec: ProviderFactorySpec,
node_version: int,
) -> None:
if node_version >= 18:
pytest.xfail(reason='Git sources not yet supported for lockfile v2 syntax')

with ManifestGenerator() as gen:
await provider_factory_spec.generate_modules('minimal-git', gen, node_version)

Expand Down