Skip to content

Commit

Permalink
add examples smoke tests
Browse files Browse the repository at this point in the history
  • Loading branch information
mattseddon committed Jul 31, 2024
1 parent 915e5c8 commit bdede34
Show file tree
Hide file tree
Showing 10 changed files with 175 additions and 30 deletions.
2 changes: 1 addition & 1 deletion .github/workflows/benchmarks.yml
Original file line number Diff line number Diff line change
Expand Up @@ -11,7 +11,7 @@ env:
FORCE_COLOR: "1"

jobs:
build:
run:
if: ${{ github.event_name != 'pull_request' || contains(github.event.pull_request.labels.*.name, 'run-benchmarks') }}
runs-on: ubuntu-latest

Expand Down
37 changes: 37 additions & 0 deletions .github/workflows/examples.yml
Original file line number Diff line number Diff line change
@@ -0,0 +1,37 @@
name: Examples

on:
workflow_dispatch:
schedule:
- cron: '0 3 * * *'
push: # to remove

env:
FORCE_COLOR: "1"

jobs:
run:
runs-on: ${{ matrix.os }}
timeout-minutes: 60
strategy:
fail-fast: false
matrix:
os: [ubuntu-latest-16-cores, macos-latest, windows-latest-8-cores]
pyv: ['3.9', '3.12']
steps:
- uses: actions/checkout@v4

- name: Set up Python ${{ matrix.pyv }}
uses: actions/setup-python@v5
with:
python-version: ${{ matrix.pyv }}
cache: 'pip'

- name: Upgrade nox and uv
run: |
python -m pip install --upgrade 'nox[uv]'
nox --version
uv --version
- name: Run examples
run: nox -s examples -p ${{ matrix.pyv }}
3 changes: 2 additions & 1 deletion examples/computer_vision/iptc_exif_xmp_lib.py
Original file line number Diff line number Diff line change
@@ -1,3 +1,4 @@
# pip install defusedxml
import json

from PIL import (
Expand Down Expand Up @@ -63,7 +64,7 @@ def image_description(file):
DataChain.from_storage(source, type="image")
.settings(parallel=-1)
.filter(C("file.name").glob("*.jpg"))
.limit(10000)
.limit(5000)
.map(
image_description,
params=["file"],
Expand Down
2 changes: 1 addition & 1 deletion examples/computer_vision/openimage-detect.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ def openimage_detect(args):
yield fstream, bbox


source = "gs://datachain-demo/openimages-v6-test-jsonpairs"
source = "gs://datachain-demo/openimages-v6-test-jsonpairs/"

(
DataChain.from_storage(source)
Expand Down
18 changes: 4 additions & 14 deletions examples/get_started/json-csv-reader.py
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ def main():
print("========================================================================")
uri = "gs://datachain-demo/jsonl/object.jsonl"
jsonl_ds = DataChain.from_json(uri, meta_type="jsonl", print_schema=True)
print(jsonl_ds.to_pandas())
jsonl_ds.show()

print()
print("========================================================================")
Expand All @@ -49,8 +49,7 @@ def main():
json_pairs_ds = DataChain.from_json(
uri, schema_from=schema_uri, jmespath="@", model_name="OpenImage"
)
print(json_pairs_ds.to_pandas())
# print(list(json_pairs_ds.collect())[0])
json_pairs_ds.show()

uri = "gs://datachain-demo/coco2017/annotations_captions/"

Expand All @@ -72,7 +71,7 @@ def main():
static_json_ds = DataChain.from_json(
uri, jmespath="licenses", spec=LicenseFeature, nrows=3
)
print(static_json_ds.to_pandas())
static_json_ds.show()

print()
print("========================================================================")
Expand All @@ -88,16 +87,7 @@ def main():
print("========================================================================")
static_csv_ds = DataChain.from_csv(uri, output=ChatDialog, object_name="chat")
static_csv_ds.print_schema()
print(static_csv_ds.to_pandas())

uri = "gs://datachain-demo/laion-aesthetics-csv"
print()
print("========================================================================")
print("dynamic CSV with header schema test parsing 3/3M objects")
print("========================================================================")
dynamic_csv_ds = DataChain.from_csv(uri, object_name="laion", nrows=3)
dynamic_csv_ds.print_schema()
print(dynamic_csv_ds.to_pandas())
static_csv_ds.show()


if __name__ == "__main__":
Expand Down
2 changes: 1 addition & 1 deletion examples/get_started/torch-loader.py
Original file line number Diff line number Diff line change
Expand Up @@ -64,7 +64,7 @@ def forward(self, x):
optimizer = optim.Adam(model.parameters(), lr=0.001)

# Train the model
num_epochs = 10
num_epochs = 1
for epoch in range(num_epochs):
for i, data in enumerate(train_loader):
inputs, labels = data
Expand Down
31 changes: 20 additions & 11 deletions examples/multimodal/wds.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,3 @@
import pandas as pd

from datachain import C, DataChain
from datachain.lib.webdataset import process_webdataset
from datachain.lib.webdataset_laion import WDSLaion, process_laion_meta
Expand All @@ -9,25 +7,36 @@
.filter(C("file.name").glob("00000000.tar"))
.settings(cache=True)
.gen(laion=process_webdataset(spec=WDSLaion), params="file")
.save() # materialize chain to avoid downloading data multiple times
)

meta_pq = (
DataChain.from_parquet("gs://datachain-demo/datacomp-small/metadata/0020f*.parquet")
.filter(
C("uid").in_(values[0] for values in wds.select("laion.json.uid").collect())
)
.map(stem=lambda file: file.get_file_stem(), params=["source.file"], output=str)
.save()
)

meta_emd = (
DataChain.from_storage("gs://datachain-demo/datacomp-small/metadata")
.filter(C("file.name").glob("0020f*.npz"))
DataChain.from_storage("gs://datachain-demo/datacomp-small/metadata/0020f*.npz")
.gen(emd=process_laion_meta)
.filter(
C("emd.index").in_(
values[0] for values in meta_pq.select("source.index").collect()
)
)
.map(stem=lambda file: file.get_file_stem(), params=["emd.file"], output=str)
)

meta_pq = DataChain.from_parquet(
"gs://datachain-demo/datacomp-small/metadata/0020f*.parquet"
).map(stem=lambda file: file.get_file_stem(), params=["source.file"], output=str)

meta = meta_emd.merge(
meta_pq, on=["stem", "emd.index"], right_on=["stem", "source.index"]
meta_pq,
on=["stem", "emd.index"],
right_on=["stem", "source.index"],
)

res = wds.merge(meta, on="laion.json.uid", right_on="uid")

df = res.limit(10).to_pandas()
with pd.option_context("display.max_columns", None):
print(df)
res.show(3)
16 changes: 16 additions & 0 deletions noxfile.py
Original file line number Diff line number Diff line change
Expand Up @@ -74,3 +74,19 @@ def dev(session: nox.Session) -> None:

python = os.path.join(venv_dir, "bin/python")
session.run(python, "-m", "pip", "install", "-e", ".[dev]", external=True)


@nox.session(python=["3.9", "3.10", "3.11", "3.12", "pypy3.9", "pypy3.10"])
def examples(session: nox.Session) -> None:
session.install(".[examples]")
try:
session.install("unstructured[all-docs]")
except: # noqa: S110, E722
pass
session.run(
"pytest",
"-m",
"examples",
"-vvv",
*session.posargs,
)
7 changes: 6 additions & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -93,6 +93,11 @@ dev = [
"types-PyYAML",
"types-requests"
]
examples = [
"datachain[tests]",
"defusedxml",
"accelerate",
]

[project.urls]
Documentation = "https://datachain.dvc.ai"
Expand All @@ -110,7 +115,7 @@ namespaces = false
[tool.setuptools_scm]

[tool.pytest.ini_options]
addopts = "-rfEs -m 'not benchmark'"
addopts = "-rfEs -m 'not benchmark and not examples'"
markers = [
"benchmark: benchmarks.",
"e2e: End-to-end tests"
Expand Down
87 changes: 87 additions & 0 deletions tests/examples/test_examples.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,87 @@
import glob
import os
import subprocess
import sys

import pytest

NO_EXAMPLES = "no examples found"


def can_import_unstructured():
try:
import unstructured # noqa: F401

return True
except ImportError:
return False


get_started_examples = [
filename
for filename in glob.glob("examples/get_started/**/*.py", recursive=True)
if "torch" not in filename or os.environ.get("RUNNER_OS") != "Linux"
]

llm_and_nlp_examples = [
filename
for filename in glob.glob("examples/llm_and_nlp/**/*.py", recursive=True)
# no anthropic token
if "claude" not in filename
and ("unstructured" not in filename or can_import_unstructured())
] or [NO_EXAMPLES]

multimodal_examples = [
filename
for filename in glob.glob("examples/multimodal/**/*.py", recursive=True)
# no OpenAI token
# and hf download painfully slow
if "openai" not in filename and "hf" not in filename
]

computer_vision_examples = [
filename
for filename in glob.glob("examples/multimodal/**/*.py", recursive=True)
# fashion product images tutorial out of scope
# and hf download painfully slow
if "image_desc" not in filename and "fashion_product_images" not in filename
]


def smoke_test(example: str):
if example == NO_EXAMPLES:
return

completed_process = subprocess.run( # noqa: S603
[sys.executable, example],
capture_output=True,
cwd=os.path.abspath(os.path.join(__file__, "..", "..", "..")),
check=True,
)

assert completed_process.stdout
assert completed_process.stderr


@pytest.mark.examples
@pytest.mark.parametrize("example", get_started_examples)
def test_get_started_examples(example):
smoke_test(example)


@pytest.mark.examples
@pytest.mark.parametrize("example", llm_and_nlp_examples)
def test_llm_and_nlp_examples(example):
smoke_test(example)


@pytest.mark.examples
@pytest.mark.parametrize("example", multimodal_examples)
def test_multimodal(example):
smoke_test(example)


@pytest.mark.examples
@pytest.mark.parametrize("example", computer_vision_examples)
def test_computer_vision_examples(example):
smoke_test(example)

0 comments on commit bdede34

Please sign in to comment.