Skip to content

Commit

Permalink
simplify
Browse files Browse the repository at this point in the history
  • Loading branch information
wolfv committed Sep 25, 2024
1 parent 228761a commit 6b670c7
Show file tree
Hide file tree
Showing 4 changed files with 464 additions and 1 deletion.
5 changes: 4 additions & 1 deletion conda_forge_tick/migrators/migration_yaml.py
Original file line number Diff line number Diff line change
Expand Up @@ -280,7 +280,10 @@ def migrate(
yaml_safe_dump(cfg, fp)

with pushd(recipe_dir):
self.set_build_number("meta.yaml")
if os.path.exists("recipe.yaml"):
self.set_build_number("recipe.yaml")
else:
self.set_build_number("meta.yaml")

return super().migrate(recipe_dir, attrs)

Expand Down
346 changes: 346 additions & 0 deletions tests/test_migrators_v1.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,346 @@
from __future__ import annotations

import os
import re
import subprocess
from pathlib import Path

import pytest
from test_recipe_yaml_parsing import TEST_RECIPE_YAML_PATH

from conda_forge_tick.contexts import ClonedFeedstockContext
from conda_forge_tick.feedstock_parser import (
parse_recipe_yaml,
populate_feedstock_attributes,
)
from conda_forge_tick.migrators import (
MigrationYaml,
Migrator,
MiniMigrator,
Replacement,
Version,
)
from conda_forge_tick.os_utils import pushd
from conda_forge_tick.utils import frozen_to_json_friendly


class NoFilter:
def filter(self, attrs, not_bad_str_start=""):
return False


class _MigrationYaml(NoFilter, MigrationYaml):
pass


yaml_rebuild = _MigrationYaml(yaml_contents="hello world", name="hi")
yaml_rebuild.cycles = []
yaml_rebuild_no_build_number = _MigrationYaml(
yaml_contents="hello world",
name="hi",
bump_number=0,
)
yaml_rebuild_no_build_number.cycles = []


def requirements_from_yaml(reqs: list) -> set[str]:
res = set()
for req in reqs:
if isinstance(req, dict):
if "pin_compatible" in req:
res.add(req["pin_compatible"]["name"])
elif "pin_subpackage" in req:
res.add(req["pin_subpackage"]["name"])

Check warning on line 53 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L52-L53

Added lines #L52 - L53 were not covered by tests
else:
# add if and else branch
res |= set(req["then"])
res |= set(req.get("else", []))

Check warning on line 57 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L56-L57

Added lines #L56 - L57 were not covered by tests
else:
res.add(req)

return res


def run_test_yaml_migration(
m, *, inp, output, kwargs, prb, mr_out, tmpdir, should_filter=False, is_v1=False
):
os.makedirs(os.path.join(tmpdir, "recipe"), exist_ok=True)

with open(os.path.join(tmpdir, "recipe", "recipe.yaml"), "w") as f:
f.write(inp)

with pushd(tmpdir):
subprocess.run(["git", "init"])
# Load the recipe.yaml (this is done in the graph)
try:
pmy = parse_recipe_yaml(inp)
except Exception:
pmy = {}

Check warning on line 78 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L77-L78

Added lines #L77 - L78 were not covered by tests
if pmy:
pmy["version"] = pmy["package"]["version"]
pmy["req"] = set()
for k in ["build", "host", "run"]:
reqs = requirements_from_yaml(pmy.get("requirements", {}).get(k, set()))
pmy["req"] |= reqs
try:
pmy["recipe_yaml"] = parse_recipe_yaml(inp)
except Exception:
pmy["recipe_yaml"] = {}

Check warning on line 88 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L87-L88

Added lines #L87 - L88 were not covered by tests
pmy["raw_meta_yaml"] = inp
pmy.update(kwargs)

assert m.filter(pmy) is should_filter
if should_filter:
return

Check warning on line 94 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L94

Added line #L94 was not covered by tests

mr = m.migrate(os.path.join(tmpdir, "recipe"), pmy)
assert mr_out == mr
pmy["pr_info"] = {}
pmy["pr_info"].update(PRed=[frozen_to_json_friendly(mr)])
with open(os.path.join(tmpdir, "recipe/recipe.yaml")) as f:
actual_output = f.read()
assert actual_output == output
assert os.path.exists(os.path.join(tmpdir, ".ci_support/migrations/hi.yaml"))
with open(os.path.join(tmpdir, ".ci_support/migrations/hi.yaml")) as f:
saved_migration = f.read()
assert saved_migration == m.yaml_contents


def sample_yaml_rebuild() -> str:
yaml = TEST_RECIPE_YAML_PATH / "scipy_migrate.yaml"
sample_yaml_rebuild = yaml.read_text()
return sample_yaml_rebuild


def test_yaml_migration_rebuild(tmpdir):
"""Test that the build number is bumped"""
sample = sample_yaml_rebuild()
updated_yaml_rebuild = sample.replace("number: 0", "number: 1")

run_test_yaml_migration(
m=yaml_rebuild,
inp=sample,
output=updated_yaml_rebuild,
kwargs={"feedstock_name": "scipy"},
prb="This PR has been triggered in an effort to update **hi**.",
mr_out={
"migrator_name": yaml_rebuild.__class__.__name__,
"migrator_version": yaml_rebuild.migrator_version,
"name": "hi",
"bot_rerun": False,
},
tmpdir=tmpdir,
is_v1=True,
)


def test_yaml_migration_rebuild_no_buildno(tmpdir):
sample = sample_yaml_rebuild()

run_test_yaml_migration(
m=yaml_rebuild_no_build_number,
inp=sample,
output=sample,
kwargs={"feedstock_name": "scipy"},
prb="This PR has been triggered in an effort to update **hi**.",
mr_out={
"migrator_name": yaml_rebuild.__class__.__name__,
"migrator_version": yaml_rebuild.migrator_version,
"name": "hi",
"bot_rerun": False,
},
tmpdir=tmpdir,
)


##################################################################
# Run Matplotlib mini-migrator ###
##################################################################

version = Version(set())

matplotlib = Replacement(
old_pkg="matplotlib",
new_pkg="matplotlib-base",
rationale=(
"Unless you need `pyqt`, recipes should depend only on " "`matplotlib-base`."
),
pr_limit=5,
)


class MockLazyJson:
def __init__(self, data):
self.data = data

Check warning on line 174 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L174

Added line #L174 was not covered by tests

def __enter__(self):
return self

Check warning on line 177 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L177

Added line #L177 was not covered by tests

def __exit__(self, *args):
pass

Check warning on line 180 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L180

Added line #L180 was not covered by tests


os.environ["RUN_URL"] = "hi world"


def run_test_migration(
m: Migrator,
inp: str,
output: str,
kwargs: dict,
prb: str,
mr_out: dict,
tmpdir: str,
should_filter: bool = False,
make_body: bool = False,
):
if mr_out:
mr_out.update(bot_rerun=False)

Path(tmpdir).joinpath("recipe.yaml").write_text(inp)

# read the conda-forge.yml
cf_yml_path = Path(tmpdir).parent / "conda-forge.yml"
cf_yml = cf_yml_path.read_text() if cf_yml_path.exists() else "{}"

# Load the recipe.yaml (this is done in the graph)
try:
name = parse_recipe_yaml(inp)["package"]["name"]
except Exception:
name = "blah"

Check warning on line 210 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L209-L210

Added lines #L209 - L210 were not covered by tests

pmy = populate_feedstock_attributes(
name, sub_graph={}, recipe_yaml=inp, conda_forge_yaml=cf_yml
)

# these are here for legacy migrators
pmy["version"] = pmy["recipe_yaml"]["package"]["version"]
pmy["req"] = set()
for k in ["build", "host", "run"]:
reqs = requirements_from_yaml(pmy.get("requirements", {}).get(k, set()))
pmy["req"] |= reqs
pmy["raw_meta_yaml"] = inp
pmy.update(kwargs)

Check warning on line 223 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L217-L223

Added lines #L217 - L223 were not covered by tests

try:
if "new_version" in kwargs:
pmy["version_pr_info"] = {"new_version": kwargs["new_version"]}
assert m.filter(pmy) == should_filter

Check warning on line 228 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L225-L228

Added lines #L225 - L228 were not covered by tests
finally:
pmy.pop("version_pr_info", None)
if should_filter:
return pmy

Check warning on line 232 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L230-L232

Added lines #L230 - L232 were not covered by tests

m.run_pre_piggyback_migrations(

Check warning on line 234 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L234

Added line #L234 was not covered by tests
tmpdir,
pmy,
hash_type=pmy.get("hash_type", "sha256"),
)
mr = m.migrate(tmpdir, pmy, hash_type=pmy.get("hash_type", "sha256"))
m.run_post_piggyback_migrations(

Check warning on line 240 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L239-L240

Added lines #L239 - L240 were not covered by tests
tmpdir,
pmy,
hash_type=pmy.get("hash_type", "sha256"),
)

if make_body:
fctx = ClonedFeedstockContext(

Check warning on line 247 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L246-L247

Added lines #L246 - L247 were not covered by tests
feedstock_name=name,
attrs=pmy,
local_clone_dir=Path(tmpdir),
)
m.effective_graph.add_node(name)
m.effective_graph.nodes[name]["payload"] = MockLazyJson({})
m.pr_body(fctx)

Check warning on line 254 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L252-L254

Added lines #L252 - L254 were not covered by tests

assert mr_out == mr
if not mr:
return pmy

Check warning on line 258 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L256-L258

Added lines #L256 - L258 were not covered by tests

pmy["pr_info"] = {}
pmy["pr_info"].update(PRed=[frozen_to_json_friendly(mr)])
with open(os.path.join(tmpdir, "recipe.yaml")) as f:
actual_output = f.read()

Check warning on line 263 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L260-L263

Added lines #L260 - L263 were not covered by tests

# strip jinja comments
pat = re.compile(r"{#.*#}")
actual_output = pat.sub("", actual_output)
output = pat.sub("", output)
assert actual_output == output

Check warning on line 269 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L266-L269

Added lines #L266 - L269 were not covered by tests
# TODO: fix subgraph here (need this to be xsh file)
if isinstance(m, Version):
pass

Check warning on line 272 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L271-L272

Added lines #L271 - L272 were not covered by tests
else:
assert prb in m.pr_body(None)
try:
if "new_version" in kwargs:
pmy["version_pr_info"] = {"new_version": kwargs["new_version"]}
assert m.filter(pmy) is True

Check warning on line 278 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L274-L278

Added lines #L274 - L278 were not covered by tests
finally:
pmy.pop("version_pr_info", None)

Check warning on line 280 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L280

Added line #L280 was not covered by tests

return pmy

Check warning on line 282 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L282

Added line #L282 was not covered by tests


def run_minimigrator(
migrator: MiniMigrator,
inp: str,
output: str,
mr_out: dict,
tmpdir: str,
should_filter: bool = False,
):
if mr_out:
mr_out.update(bot_rerun=False)
with open(os.path.join(tmpdir, "recipe.yaml"), "w") as f:
f.write(inp)

Check warning on line 296 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L293-L296

Added lines #L293 - L296 were not covered by tests

# read the conda-forge.yml
if os.path.exists(os.path.join(tmpdir, "..", "conda-forge.yml")):
with open(os.path.join(tmpdir, "..", "conda-forge.yml")) as fp:
cf_yml = fp.read()

Check warning on line 301 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L299-L301

Added lines #L299 - L301 were not covered by tests
else:
cf_yml = "{}"

Check warning on line 303 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L303

Added line #L303 was not covered by tests

# Load the recipe.yaml (this is done in the graph)
try:
name = parse_recipe_yaml(inp)["package"]["name"]
except Exception:
name = "blah"

Check warning on line 309 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L306-L309

Added lines #L306 - L309 were not covered by tests

pmy = populate_feedstock_attributes(name, {}, inp, None, cf_yml)
filtered = migrator.filter(pmy)
if should_filter and filtered:
return migrator
assert filtered == should_filter

Check warning on line 315 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L311-L315

Added lines #L311 - L315 were not covered by tests

with open(os.path.join(tmpdir, "recipe.yaml")) as f:
actual_output = f.read()

Check warning on line 318 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L317-L318

Added lines #L317 - L318 were not covered by tests
# strip jinja comments
pat = re.compile(r"{#.*#}")
actual_output = pat.sub("", actual_output)
output = pat.sub("", output)
assert actual_output == output

Check warning on line 323 in tests/test_migrators_v1.py

View check run for this annotation

Codecov / codecov/patch

tests/test_migrators_v1.py#L320-L323

Added lines #L320 - L323 were not covered by tests


def test_generic_replacement(tmpdir):
sample_matplotlib = TEST_RECIPE_YAML_PATH / "sample_matplotlib.yaml"
sample_matplotlib = sample_matplotlib.read_text()
sample_matplotlib_correct = sample_matplotlib.replace(
" - matplotlib", " - matplotlib-base"
)
# "recipe_yaml generic parsing not implemented yet" is raised here!
with pytest.raises(NotImplementedError):
run_test_migration(
m=matplotlib,
inp=sample_matplotlib,
output=sample_matplotlib_correct,
kwargs={},
prb="I noticed that this recipe depends on `matplotlib` instead of ",
mr_out={
"migrator_name": "Replacement",
"migrator_version": matplotlib.migrator_version,
"name": "matplotlib-to-matplotlib-base",
},
tmpdir=tmpdir,
)
Loading

0 comments on commit 6b670c7

Please sign in to comment.