From 564ab243cc3fa7c0239d71a6dacb4a6c7765e9f5 Mon Sep 17 00:00:00 2001
From: jaseg <git@jaseg.de>
Date: Thu, 3 Jun 2021 23:45:11 +0200
Subject: Add svg-flatten SVG feature tests

---
 svg-flatten/Makefile                               |  15 +-
 svg-flatten/src/nopencv_test.cpp                   | 405 ---------------------
 svg-flatten/src/test/nopencv_test.cpp              | 405 +++++++++++++++++++++
 svg-flatten/src/test/svg_tests.py                  |  75 ++++
 svg-flatten/testdata/svg/circles.svg               |  78 ++++
 svg-flatten/testdata/svg/compound_xform.svg        | 102 ++++++
 svg-flatten/testdata/svg/empty.svg                 |  11 +
 svg-flatten/testdata/svg/empty_inkscape.svg        |  54 +++
 svg-flatten/testdata/svg/groups.svg                |  96 +++++
 svg-flatten/testdata/svg/pattern_fill.svg          | 135 +++++++
 svg-flatten/testdata/svg/pattern_stroke.svg        |  89 +++++
 svg-flatten/testdata/svg/pattern_stroke_dashed.svg |  89 +++++
 svg-flatten/testdata/svg/rect.svg                  |  65 ++++
 svg-flatten/testdata/svg/rect_occlusion.svg        |  79 ++++
 svg-flatten/testdata/svg/rotation.svg              |  66 ++++
 svg-flatten/testdata/svg/rotation_90.svg           |  66 ++++
 svg-flatten/testdata/svg/scale.svg                 |  91 +++++
 svg-flatten/testdata/svg/shear.svg                 |  59 +++
 svg-flatten/testdata/svg/stroke.svg                |  67 ++++
 svg-flatten/testdata/svg/stroke_caps.svg           |  68 ++++
 svg-flatten/testdata/svg/stroke_dashes.svg         |  75 ++++
 svg-flatten/testdata/svg/stroke_joins.svg          |  68 ++++
 svg-flatten/testdata/svg/text.svg                  | 108 ++++++
 23 files changed, 1952 insertions(+), 414 deletions(-)
 delete mode 100644 svg-flatten/src/nopencv_test.cpp
 create mode 100644 svg-flatten/src/test/nopencv_test.cpp
 create mode 100644 svg-flatten/src/test/svg_tests.py
 create mode 100644 svg-flatten/testdata/svg/circles.svg
 create mode 100644 svg-flatten/testdata/svg/compound_xform.svg
 create mode 100644 svg-flatten/testdata/svg/empty.svg
 create mode 100644 svg-flatten/testdata/svg/empty_inkscape.svg
 create mode 100644 svg-flatten/testdata/svg/groups.svg
 create mode 100644 svg-flatten/testdata/svg/pattern_fill.svg
 create mode 100644 svg-flatten/testdata/svg/pattern_stroke.svg
 create mode 100644 svg-flatten/testdata/svg/pattern_stroke_dashed.svg
 create mode 100644 svg-flatten/testdata/svg/rect.svg
 create mode 100644 svg-flatten/testdata/svg/rect_occlusion.svg
 create mode 100644 svg-flatten/testdata/svg/rotation.svg
 create mode 100644 svg-flatten/testdata/svg/rotation_90.svg
 create mode 100644 svg-flatten/testdata/svg/scale.svg
 create mode 100644 svg-flatten/testdata/svg/shear.svg
 create mode 100644 svg-flatten/testdata/svg/stroke.svg
 create mode 100644 svg-flatten/testdata/svg/stroke_caps.svg
 create mode 100644 svg-flatten/testdata/svg/stroke_dashes.svg
 create mode 100644 svg-flatten/testdata/svg/stroke_joins.svg
 create mode 100644 svg-flatten/testdata/svg/text.svg

(limited to 'svg-flatten')

diff --git a/svg-flatten/Makefile b/svg-flatten/Makefile
index a980c05..2c9ff6c 100644
--- a/svg-flatten/Makefile
+++ b/svg-flatten/Makefile
@@ -1,5 +1,5 @@
 
-CXX := clang++
+CXX ?= clang++
 LD ?= ld
 INSTALL := install
 PKG_CONFIG ?= pkg-config
@@ -45,7 +45,7 @@ SOURCES += $(CLIPPER_SOURCES)
 INCLUDES := -Iinclude -Isrc $(CLIPPER_INCLUDES) $(VORONOI_INCLUDES) $(POISSON_INCLUDES) $(BASE64_INCLUDES) $(ARGAGG_INCLUDES) $(CAVC_INCLUDES) $(SUBPROCESS_INCLUDES) $(MINUNIT_INCLUDES) $(STB_INCLUDES)
 
 CXXFLAGS := -std=c++2a -g -Wall -Wextra -O0
-LDFLAGS := -lm -lc -lstdc++
+LDFLAGS := -lm
 
 PKG_CONFIG_DEPS := 
 ifdef USE_SYSTEM_PUGIXML
@@ -71,19 +71,16 @@ $(BUILDDIR)/%.o: %.cpp
 
 $(BUILDDIR)/$(TARGET): $(SOURCES:%.cpp=$(BUILDDIR)/%.o)
 	@mkdir -p $(dir $@) 
-	if [ $$(uname -s) = "Darwin" ]; then \
-		$(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $^; \
-	else \
-		$(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ -Wl,--start-group $^ -lstdc++fs -Wl,--end-group; \
-	fi
+	$(CXX) $(CXXFLAGS) $(LDFLAGS) -o $@ $^
 
-$(BUILDDIR)/nopencv-tests: src/nopencv_test.cpp src/nopencv.cpp
+$(BUILDDIR)/nopencv-test: src/test/nopencv_test.cpp src/nopencv.cpp
 	@mkdir -p $(dir $@) 
 	$(CXX) $(CXXFLAGS) $(INCLUDES) $(LDFLAGS) -o $@ $^
 
 
 .PHONY: tests
-tests: $(BUILDDIR)/nopencv-tests
+tests: $(BUILDDIR)/nopencv-test
+	$(BUILDDIR)/nopencv-test
 
 .PHONY: install
 install:
diff --git a/svg-flatten/src/nopencv_test.cpp b/svg-flatten/src/nopencv_test.cpp
deleted file mode 100644
index 17ba71e..0000000
--- a/svg-flatten/src/nopencv_test.cpp
+++ /dev/null
@@ -1,405 +0,0 @@
-
-#include <iostream>
-#include <fstream>
-#include <iomanip>
-#include <cmath>
-#include <filesystem>
-
-#include "nopencv.hpp"
-
-#include <subprocess.h>
-#include <minunit.h>
-
-#include "stb_image.h"
-
-using namespace gerbolyze;
-using namespace gerbolyze::nopencv;
-
-char msg[1024];
-
-class TempfileHack {
-public:
-    TempfileHack(const string ext) : m_path { filesystem::temp_directory_path() / (std::tmpnam(nullptr) + ext) } {}
-    ~TempfileHack() { remove(m_path); }
-
-    const char *c_str() { return m_path.c_str(); }
-
-private:
-    filesystem::path m_path;
-};
-
-class SVGPolyRenderer {
-public:
-    SVGPolyRenderer(const char *fn, int width_px, int height_px)
-        : m_svg(fn) {
-        m_svg << "<svg width=\"" << width_px << "px\" height=\"" << height_px << "px\" viewBox=\"0 0 "
-            << width_px << " " << height_px << "\" "
-            << "xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\">" << endl;
-        m_svg << "<rect width=\"100%\" height=\"100%\" fill=\"black\"/>" << endl;
-    }
-
-    ContourCallback callback() {
-        return [this](Polygon_i &poly, ContourPolarity pol) {
-            mu_assert(poly.size() > 0, "Empty contour returned");
-            mu_assert(poly.size() > 2, "Contour has less than three points, no area");
-            mu_assert(pol == CP_CONTOUR || pol == CP_HOLE, "Contour has invalid polarity");
-
-            m_svg << "<path fill=\"" << ((pol == CP_HOLE) ? "black" : "white") << "\" d=\"";
-            m_svg << "M " << poly[0][0] << " " << poly[0][1];
-            for (size_t i=1; i<poly.size(); i++) {
-                m_svg << " L " << poly[i][0] << " " << poly[i][1];
-            }
-            m_svg << " Z\"/>" << endl;
-        };
-    }
-
-    void close() {
-        m_svg << "</svg>" << endl;
-        m_svg.close();
-    }
-
-private:
-    ofstream m_svg;
-};
-
-MU_TEST(test_complex_example_from_paper) {
-    int32_t img_data[6*9] = {
-        0, 0, 0, 0, 0, 0, 0, 0, 0,
-        0, 1, 1, 1, 1, 1, 1, 1, 0,
-        0, 1, 0, 0, 1, 0, 0, 1, 0,
-        0, 1, 0, 0, 1, 0, 0, 1, 0,
-        0, 1, 1, 1, 1, 1, 1, 1, 0,
-        0, 0, 0, 0, 0, 0, 0, 0, 0,
-    };
-    Image32 test_img(9, 6, static_cast<int*>(img_data));
-
-    const Polygon_i expected_polys[3] = {
-        {
-            {1,1}, {1,2}, {1,3}, {1,4}, {1,5},
-            {2,5}, {3,5}, {4,5}, {5,5}, {6,5}, {7,5}, {8,5},
-            {8,4}, {8,3}, {8,2}, {8,1},
-            {7,1}, {6,1}, {5,1}, {4,1}, {3,1}, {2,1}
-        },
-        {
-            {2,2}, {2,3}, {2,4},
-            {3,4}, {4,4},
-            {4,3}, {4,2},
-            {3,2}
-        },
-        {
-            {5,2}, {5,3}, {5,4},
-            {6,4}, {7,4},
-            {7,3}, {7,2},
-            {6,2}
-        }
-    };
-
-    const ContourPolarity expected_polarities[3] = {CP_CONTOUR, CP_HOLE, CP_HOLE};
-    
-    int invocation_count = 0;
-    gerbolyze::nopencv::find_contours(test_img, [&invocation_count, &expected_polarities, &expected_polys](Polygon_i &poly, ContourPolarity pol) {
-            invocation_count += 1;
-            mu_assert((invocation_count <= 3), "Too many contours returned"); 
-
-            mu_assert(poly.size() > 0, "Empty contour returned");
-            mu_assert_int_eq(pol, expected_polarities[invocation_count-1]);
-
-            i2p last;
-            bool first = true;
-            Polygon_i exp = expected_polys[invocation_count-1];
-            for (auto &p : poly) {
-                if (!first) {
-                    mu_assert((fabs(p[0] - last[0]) + fabs(p[1] - last[1]) == 1), "Subsequent contour points have distance other than one");
-                    mu_assert(find(exp.begin(), exp.end(), p) != exp.end(), "Got unexpected contour point");
-                }
-                last = p;
-            }
-        });
-    mu_assert_int_eq(3, invocation_count);
-
-    int32_t tpl[6*9] = {
-        0, 0, 0, 0, 0, 0, 0, 0, 0,
-        0, 2, 2, 2, 2, 2, 2,-2, 0,
-        0,-3, 0, 0,-4, 0, 0,-2, 0,
-        0,-3, 0, 0,-4, 0, 0,-2, 0,
-        0, 2, 2, 2, 2, 2, 2,-2, 0,
-        0, 0, 0, 0, 0, 0, 0, 0, 0,
-    };
-
-
-    for (int y=0; y<6; y++) {
-        for (int x=0; x<9; x++) {
-            int a = test_img.at(x, y), b = tpl[y*9+x];
-            if (a != b) {
-                cout << "Result:" << endl;
-                cout << "    ";
-                for (int x=0; x<9; x++) {
-                    cout << x << "  ";
-                }
-                cout << endl;
-                cout << "    ";
-                for (int x=0; x<9; x++) {
-                    cout << "---";
-                }
-                cout << endl;
-                for (int y=0; y<6; y++) {
-                    cout << y << " | ";
-                    for (int x=0; x<9; x++) {
-                        cout << setfill(' ') << setw(2) << test_img.at(x, y) << " ";
-                    }
-                    cout << endl;
-                }
-
-                snprintf(msg, sizeof(msg), "Result does not match template @(%d, %d): %d != %d\n", x, y, a, b);
-                mu_fail(msg);
-            }
-        }
-    }
-}
-
-int render_svg(const char *in_svg, const char *out_png) {
-    const char *command_line[] = {"resvg", in_svg, out_png, nullptr};
-    struct subprocess_s subprocess;
-    int rc = subprocess_create(command_line, subprocess_option_inherit_environment, &subprocess);
-    if (rc)
-        return rc;
-
-    int resvg_rc = -1;
-    rc = subprocess_join(&subprocess, &resvg_rc);
-    if (rc)
-        return rc;
-    if (resvg_rc)
-        return -resvg_rc;
-
-    rc = subprocess_destroy(&subprocess);
-    if (rc)
-        return rc;
-
-    return 0;
-}
-
-static void testdata_roundtrip(const char *fn) {
-    Image32 ref_img;
-    mu_assert(ref_img.load(fn), "Input image failed to load");
-    ref_img.binarize();
-    Image32 ref_img_copy(ref_img);
-
-    TempfileHack tmp_svg(".svg");
-    TempfileHack tmp_png(".png");
-
-    SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows());
-    gerbolyze::nopencv::find_contours(ref_img, ctx.callback());
-    ctx.close();
-
-    mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str()));
-
-    Image32 out_img;
-    mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load");
-    out_img.binarize();
-
-    mu_assert_int_eq(ref_img.cols(), out_img.cols());
-    mu_assert_int_eq(ref_img.rows(), out_img.rows());
-
-    for (int y=0; y<out_img.rows(); y++) {
-        for (int x=0; x<out_img.cols(); x++) {
-            if (out_img.at(x, y) != ref_img_copy.at(x, y)) {
-                snprintf(msg, sizeof(msg), "%s: Result does not match input @(%d, %d): %d != %d\n", fn, x, y, out_img.at(x, y), ref_img_copy.at(x, y));
-                mu_fail(msg);
-            }
-        }
-    }
-}
-
-MU_TEST(test_round_trip_blank)              { testdata_roundtrip("testdata/blank.png"); }
-MU_TEST(test_round_trip_white)              { testdata_roundtrip("testdata/white.png"); }
-MU_TEST(test_round_trip_blob_border_w)      { testdata_roundtrip("testdata/blob-border-w.png"); }
-MU_TEST(test_round_trip_blobs_borders)      { testdata_roundtrip("testdata/blobs-borders.png"); }
-MU_TEST(test_round_trip_blobs_corners)      { testdata_roundtrip("testdata/blobs-corners.png"); }
-MU_TEST(test_round_trip_blobs_crossing)     { testdata_roundtrip("testdata/blobs-crossing.png"); }
-MU_TEST(test_round_trip_cross)              { testdata_roundtrip("testdata/cross.png"); }
-MU_TEST(test_round_trip_letter_e)           { testdata_roundtrip("testdata/letter-e.png"); }
-MU_TEST(test_round_trip_paper_example)      { testdata_roundtrip("testdata/paper-example.png"); }
-MU_TEST(test_round_trip_paper_example_inv)  { testdata_roundtrip("testdata/paper-example-inv.png"); }
-MU_TEST(test_round_trip_single_px)          { testdata_roundtrip("testdata/single-px.png"); }
-MU_TEST(test_round_trip_single_px_inv)      { testdata_roundtrip("testdata/single-px-inv.png"); }
-MU_TEST(test_round_trip_two_blobs)          { testdata_roundtrip("testdata/two-blobs.png"); }
-MU_TEST(test_round_trip_two_px)             { testdata_roundtrip("testdata/two-px.png"); }
-MU_TEST(test_round_trip_two_px_inv)         { testdata_roundtrip("testdata/two-px-inv.png"); }
-
-static void test_polygon_area(const char *fn) {
-    //cerr << endl << "poly area test " << fn << endl;
-    Image32 ref_img;
-    mu_assert(ref_img.load(fn), "Input image failed to load");
-    ref_img.binarize();
-
-    int white_px_count = 0;
-    int black_px_count = 0;
-    for (int y=0; y<ref_img.rows(); y++) {
-        for (int x=0; x<ref_img.cols(); x++) {
-            if (ref_img.at(x, y)) {
-                white_px_count += 1;
-            } else {
-                black_px_count += 1;
-            }
-        }
-    }
-
-    double pos_sum = 0.0;
-    double neg_sum = ref_img.size();
-    gerbolyze::nopencv::find_contours(ref_img, [&pos_sum, &neg_sum](Polygon_i& poly, ContourPolarity pol) {
-            double area = polygon_area(poly);
-            //cerr << endl << fn << ": " << area << " " << pos_sum << " / " << neg_sum << " -- " << white_px_count << " / " << black_px_count << " GOT: " << poly.size() << " w/ " << pol << endl;
-            mu_assert(fabs(area) > 0.99, "Polygon smaller than a single pixel");
-            mu_assert((pol == CP_CONTOUR) == (area >= 0), "Polygon area has incorrect sign");
-
-            pos_sum += area;
-            neg_sum -= area;
-        });
-
-    mu_assert(pos_sum - white_px_count < 0.01, "Calculated area outside tolerance");
-    mu_assert(neg_sum - black_px_count < 0.01, "Calculated area outside tolerance");
-    //cerr << endl << "poly area test " << fn << " done" << endl;
-}
-
-MU_TEST(test_polygon_area_blank)              { test_polygon_area("testdata/blank.png"); }
-MU_TEST(test_polygon_area_white)              { test_polygon_area("testdata/white.png"); }
-MU_TEST(test_polygon_area_blob_border_w)      { test_polygon_area("testdata/blob-border-w.png"); }
-MU_TEST(test_polygon_area_blobs_borders)      { test_polygon_area("testdata/blobs-borders.png"); }
-MU_TEST(test_polygon_area_blobs_corners)      { test_polygon_area("testdata/blobs-corners.png"); }
-MU_TEST(test_polygon_area_blobs_crossing)     { test_polygon_area("testdata/blobs-crossing.png"); }
-MU_TEST(test_polygon_area_cross)              { test_polygon_area("testdata/cross.png"); }
-MU_TEST(test_polygon_area_letter_e)           { test_polygon_area("testdata/letter-e.png"); }
-MU_TEST(test_polygon_area_paper_example)      { test_polygon_area("testdata/paper-example.png"); }
-MU_TEST(test_polygon_area_paper_example_inv)  { test_polygon_area("testdata/paper-example-inv.png"); }
-MU_TEST(test_polygon_area_single_px)          { test_polygon_area("testdata/single-px.png"); }
-MU_TEST(test_polygon_area_single_px_inv)      { test_polygon_area("testdata/single-px-inv.png"); }
-MU_TEST(test_polygon_area_two_blobs)          { test_polygon_area("testdata/two-blobs.png"); }
-MU_TEST(test_polygon_area_two_px)             { test_polygon_area("testdata/two-px.png"); }
-MU_TEST(test_polygon_area_two_px_inv)         { test_polygon_area("testdata/two-px-inv.png"); }
-
-static void chain_approx_test(const char *fn) {
-    //cout << endl << "Testing \"" << fn << "\"" << endl;
-    Image32 ref_img;
-    mu_assert(ref_img.load(fn), "Input image failed to load");
-    ref_img.binarize();
-    Image32 ref_img_copy(ref_img);
-
-    TempfileHack tmp_svg(".svg");
-    TempfileHack tmp_png(".png");
-
-    SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows());
-    gerbolyze::nopencv::find_contours(ref_img, simplify_contours_teh_chin(ctx.callback()));
-    ctx.close();
-
-    mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str()));
-
-    Image32 out_img;
-    mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load");
-    mu_assert_int_eq(ref_img.rows(), out_img.rows());
-    mu_assert_int_eq(ref_img.cols(), out_img.cols());
-
-    double max_abs_deviation = 0;
-    double rms_sum = 0; 
-    double mean_sum = 0; 
-    for (int y=0; y<out_img.rows(); y++) {
-        for (int x=0; x<out_img.cols(); x++) {
-            double delta = fabs((double)out_img.at(x, y)/255.0 - (double)ref_img_copy.at(x, y));
-            max_abs_deviation = fmax(max_abs_deviation, delta);
-            rms_sum += delta*delta;
-            mean_sum += delta;
-        }
-    }
-
-    rms_sum = sqrt(rms_sum / out_img.size());
-    mean_sum /= out_img.size();
-    if (rms_sum > 0.5) {
-        snprintf(msg, sizeof(msg), "%s: Chain approximation RMS error is above threshold: %.3f > 0.5\n", fn, rms_sum);
-        mu_fail(msg);
-    }
-    if (mean_sum > 0.1) {
-        snprintf(msg, sizeof(msg), "%s: Chain approximation mean error is above threshold: %.3f > 0.1\n", fn, mean_sum);
-        mu_fail(msg);
-    }
-    //mu_assert(max_abs_deviation < 0.5, "Maximum chain approximation error is above threshold");
-}
-
-
-MU_TEST(chain_approx_test_chromosome)         { chain_approx_test("testdata/chain-approx-teh-chin-chromosome.png"); }
-MU_TEST(chain_approx_test_blank)              { chain_approx_test("testdata/blank.png"); }
-MU_TEST(chain_approx_test_white)              { chain_approx_test("testdata/white.png"); }
-MU_TEST(chain_approx_test_blob_border_w)      { chain_approx_test("testdata/blob-border-w.png"); }
-MU_TEST(chain_approx_test_blobs_borders)      { chain_approx_test("testdata/blobs-borders.png"); }
-MU_TEST(chain_approx_test_blobs_corners)      { chain_approx_test("testdata/blobs-corners.png"); }
-MU_TEST(chain_approx_test_blobs_crossing)     { chain_approx_test("testdata/blobs-crossing.png"); }
-MU_TEST(chain_approx_test_cross)              { chain_approx_test("testdata/cross.png"); }
-MU_TEST(chain_approx_test_letter_e)           { chain_approx_test("testdata/letter-e.png"); }
-MU_TEST(chain_approx_test_paper_example)      { chain_approx_test("testdata/paper-example.png"); }
-MU_TEST(chain_approx_test_paper_example_inv)  { chain_approx_test("testdata/paper-example-inv.png"); }
-MU_TEST(chain_approx_test_single_px)          { chain_approx_test("testdata/single-px.png"); }
-MU_TEST(chain_approx_test_single_px_inv)      { chain_approx_test("testdata/single-px-inv.png"); }
-MU_TEST(chain_approx_test_two_blobs)          { chain_approx_test("testdata/two-blobs.png"); }
-MU_TEST(chain_approx_test_two_px)             { chain_approx_test("testdata/two-px.png"); }
-MU_TEST(chain_approx_test_two_px_inv)         { chain_approx_test("testdata/two-px-inv.png"); }
-
-
-MU_TEST_SUITE(nopencv_contours_suite) {
-    MU_RUN_TEST(test_complex_example_from_paper);
-
-    MU_RUN_TEST(test_round_trip_blank);
-    MU_RUN_TEST(test_round_trip_white);
-    MU_RUN_TEST(test_round_trip_blob_border_w);
-    MU_RUN_TEST(test_round_trip_blobs_borders);
-    MU_RUN_TEST(test_round_trip_blobs_corners);
-    MU_RUN_TEST(test_round_trip_blobs_crossing);
-    MU_RUN_TEST(test_round_trip_cross);
-    MU_RUN_TEST(test_round_trip_letter_e);
-    MU_RUN_TEST(test_round_trip_paper_example);
-    MU_RUN_TEST(test_round_trip_paper_example_inv);
-    MU_RUN_TEST(test_round_trip_single_px);
-    MU_RUN_TEST(test_round_trip_single_px_inv);
-    MU_RUN_TEST(test_round_trip_two_blobs);
-    MU_RUN_TEST(test_round_trip_two_px);
-    MU_RUN_TEST(test_round_trip_two_px_inv);
-
-    MU_RUN_TEST(chain_approx_test_chromosome);
-    MU_RUN_TEST(chain_approx_test_blank);
-    MU_RUN_TEST(chain_approx_test_white);
-    MU_RUN_TEST(chain_approx_test_blob_border_w);
-    MU_RUN_TEST(chain_approx_test_blobs_borders);
-    MU_RUN_TEST(chain_approx_test_blobs_corners);
-    MU_RUN_TEST(chain_approx_test_blobs_crossing);
-    MU_RUN_TEST(chain_approx_test_cross);
-    MU_RUN_TEST(chain_approx_test_letter_e);
-    MU_RUN_TEST(chain_approx_test_paper_example);
-    MU_RUN_TEST(chain_approx_test_paper_example_inv);
-    MU_RUN_TEST(chain_approx_test_single_px);
-    MU_RUN_TEST(chain_approx_test_single_px_inv);
-    MU_RUN_TEST(chain_approx_test_two_blobs);
-    MU_RUN_TEST(chain_approx_test_two_px);
-    MU_RUN_TEST(chain_approx_test_two_px_inv);
-
-    MU_RUN_TEST(test_polygon_area_blank);
-    MU_RUN_TEST(test_polygon_area_white);
-    MU_RUN_TEST(test_polygon_area_blob_border_w);
-    MU_RUN_TEST(test_polygon_area_blobs_borders);
-    MU_RUN_TEST(test_polygon_area_blobs_corners);
-    MU_RUN_TEST(test_polygon_area_blobs_crossing);
-    MU_RUN_TEST(test_polygon_area_cross);
-    MU_RUN_TEST(test_polygon_area_letter_e);
-    MU_RUN_TEST(test_polygon_area_paper_example);
-    MU_RUN_TEST(test_polygon_area_paper_example_inv);
-    MU_RUN_TEST(test_polygon_area_single_px);
-    MU_RUN_TEST(test_polygon_area_single_px_inv);
-    MU_RUN_TEST(test_polygon_area_two_blobs);
-    MU_RUN_TEST(test_polygon_area_two_px);
-    MU_RUN_TEST(test_polygon_area_two_px_inv);
-};
-
-int main(int argc, char **argv) {
-    (void)argc;
-    (void)argv;
-
-    MU_RUN_SUITE(nopencv_contours_suite);
-    MU_REPORT();
-    return MU_EXIT_CODE;
-}
diff --git a/svg-flatten/src/test/nopencv_test.cpp b/svg-flatten/src/test/nopencv_test.cpp
new file mode 100644
index 0000000..2f358b2
--- /dev/null
+++ b/svg-flatten/src/test/nopencv_test.cpp
@@ -0,0 +1,405 @@
+
+#include <iostream>
+#include <fstream>
+#include <iomanip>
+#include <cmath>
+#include <filesystem>
+
+#include "nopencv.hpp"
+
+#include <subprocess.h>
+#include <minunit.h>
+
+#include "stb_image.h"
+
+using namespace gerbolyze;
+using namespace gerbolyze::nopencv;
+
+char msg[512];
+
+class TempfileHack {
+public:
+    TempfileHack(const string ext) : m_path { filesystem::temp_directory_path() / (std::tmpnam(nullptr) + ext) } {}
+    ~TempfileHack() { remove(m_path); }
+
+    const char *c_str() { return m_path.c_str(); }
+
+private:
+    filesystem::path m_path;
+};
+
+class SVGPolyRenderer {
+public:
+    SVGPolyRenderer(const char *fn, int width_px, int height_px)
+        : m_svg(fn) {
+        m_svg << "<svg width=\"" << width_px << "px\" height=\"" << height_px << "px\" viewBox=\"0 0 "
+            << width_px << " " << height_px << "\" "
+            << "xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\">" << endl;
+        m_svg << "<rect width=\"100%\" height=\"100%\" fill=\"black\"/>" << endl;
+    }
+
+    ContourCallback callback() {
+        return [this](Polygon_i &poly, ContourPolarity pol) {
+            mu_assert(poly.size() > 0, "Empty contour returned");
+            mu_assert(poly.size() > 2, "Contour has less than three points, no area");
+            mu_assert(pol == CP_CONTOUR || pol == CP_HOLE, "Contour has invalid polarity");
+
+            m_svg << "<path fill=\"" << ((pol == CP_HOLE) ? "black" : "white") << "\" d=\"";
+            m_svg << "M " << poly[0][0] << " " << poly[0][1];
+            for (size_t i=1; i<poly.size(); i++) {
+                m_svg << " L " << poly[i][0] << " " << poly[i][1];
+            }
+            m_svg << " Z\"/>" << endl;
+        };
+    }
+
+    void close() {
+        m_svg << "</svg>" << endl;
+        m_svg.close();
+    }
+
+private:
+    ofstream m_svg;
+};
+
+MU_TEST(test_complex_example_from_paper) {
+    int32_t img_data[6*9] = {
+        0, 0, 0, 0, 0, 0, 0, 0, 0,
+        0, 1, 1, 1, 1, 1, 1, 1, 0,
+        0, 1, 0, 0, 1, 0, 0, 1, 0,
+        0, 1, 0, 0, 1, 0, 0, 1, 0,
+        0, 1, 1, 1, 1, 1, 1, 1, 0,
+        0, 0, 0, 0, 0, 0, 0, 0, 0,
+    };
+    Image32 test_img(9, 6, static_cast<int*>(img_data));
+
+    const Polygon_i expected_polys[3] = {
+        {
+            {1,1}, {1,2}, {1,3}, {1,4}, {1,5},
+            {2,5}, {3,5}, {4,5}, {5,5}, {6,5}, {7,5}, {8,5},
+            {8,4}, {8,3}, {8,2}, {8,1},
+            {7,1}, {6,1}, {5,1}, {4,1}, {3,1}, {2,1}
+        },
+        {
+            {2,2}, {2,3}, {2,4},
+            {3,4}, {4,4},
+            {4,3}, {4,2},
+            {3,2}
+        },
+        {
+            {5,2}, {5,3}, {5,4},
+            {6,4}, {7,4},
+            {7,3}, {7,2},
+            {6,2}
+        }
+    };
+
+    const ContourPolarity expected_polarities[3] = {CP_CONTOUR, CP_HOLE, CP_HOLE};
+    
+    int invocation_count = 0;
+    gerbolyze::nopencv::find_contours(test_img, [&invocation_count, &expected_polarities, &expected_polys](Polygon_i &poly, ContourPolarity pol) {
+            invocation_count += 1;
+            mu_assert((invocation_count <= 3), "Too many contours returned"); 
+
+            mu_assert(poly.size() > 0, "Empty contour returned");
+            mu_assert_int_eq(pol, expected_polarities[invocation_count-1]);
+
+            i2p last;
+            bool first = true;
+            Polygon_i exp = expected_polys[invocation_count-1];
+            for (auto &p : poly) {
+                if (!first) {
+                    mu_assert((fabs(p[0] - last[0]) + fabs(p[1] - last[1]) == 1), "Subsequent contour points have distance other than one");
+                    mu_assert(find(exp.begin(), exp.end(), p) != exp.end(), "Got unexpected contour point");
+                }
+                last = p;
+            }
+        });
+    mu_assert_int_eq(3, invocation_count);
+
+    int32_t tpl[6*9] = {
+        0, 0, 0, 0, 0, 0, 0, 0, 0,
+        0, 2, 2, 2, 2, 2, 2,-2, 0,
+        0,-3, 0, 0,-4, 0, 0,-2, 0,
+        0,-3, 0, 0,-4, 0, 0,-2, 0,
+        0, 2, 2, 2, 2, 2, 2,-2, 0,
+        0, 0, 0, 0, 0, 0, 0, 0, 0,
+    };
+
+
+    for (int y=0; y<6; y++) {
+        for (int x=0; x<9; x++) {
+            int a = test_img.at(x, y), b = tpl[y*9+x];
+            if (a != b) {
+                cout << "Result:" << endl;
+                cout << "    ";
+                for (int x=0; x<9; x++) {
+                    cout << x << "  ";
+                }
+                cout << endl;
+                cout << "    ";
+                for (int x=0; x<9; x++) {
+                    cout << "---";
+                }
+                cout << endl;
+                for (int y=0; y<6; y++) {
+                    cout << y << " | ";
+                    for (int x=0; x<9; x++) {
+                        cout << setfill(' ') << setw(2) << test_img.at(x, y) << " ";
+                    }
+                    cout << endl;
+                }
+
+                snprintf(msg, sizeof(msg), "Result does not match template @(%d, %d): %d != %d\n", x, y, a, b);
+                mu_fail(msg);
+            }
+        }
+    }
+}
+
+int render_svg(const char *in_svg, const char *out_png) {
+    const char *command_line[] = {"resvg", in_svg, out_png, nullptr};
+    struct subprocess_s subprocess;
+    int rc = subprocess_create(command_line, subprocess_option_inherit_environment, &subprocess);
+    if (rc)
+        return rc;
+
+    int resvg_rc = -1;
+    rc = subprocess_join(&subprocess, &resvg_rc);
+    if (rc)
+        return rc;
+    if (resvg_rc)
+        return -resvg_rc;
+
+    rc = subprocess_destroy(&subprocess);
+    if (rc)
+        return rc;
+
+    return 0;
+}
+
+static void testdata_roundtrip(const char *fn) {
+    Image32 ref_img;
+    mu_assert(ref_img.load(fn), "Input image failed to load");
+    ref_img.binarize();
+    Image32 ref_img_copy(ref_img);
+
+    TempfileHack tmp_svg(".svg");
+    TempfileHack tmp_png(".png");
+
+    SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows());
+    gerbolyze::nopencv::find_contours(ref_img, ctx.callback());
+    ctx.close();
+
+    mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str()));
+
+    Image32 out_img;
+    mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load");
+    out_img.binarize();
+
+    mu_assert_int_eq(ref_img.cols(), out_img.cols());
+    mu_assert_int_eq(ref_img.rows(), out_img.rows());
+
+    for (int y=0; y<out_img.rows(); y++) {
+        for (int x=0; x<out_img.cols(); x++) {
+            if (out_img.at(x, y) != ref_img_copy.at(x, y)) {
+                snprintf(msg, sizeof(msg), "%s: Result does not match input @(%d, %d): %d != %d\n", fn, x, y, out_img.at(x, y), ref_img_copy.at(x, y));
+                mu_fail(msg);
+            }
+        }
+    }
+}
+
+MU_TEST(test_round_trip_blank)              { testdata_roundtrip("testdata/blank.png"); }
+MU_TEST(test_round_trip_white)              { testdata_roundtrip("testdata/white.png"); }
+MU_TEST(test_round_trip_blob_border_w)      { testdata_roundtrip("testdata/blob-border-w.png"); }
+MU_TEST(test_round_trip_blobs_borders)      { testdata_roundtrip("testdata/blobs-borders.png"); }
+MU_TEST(test_round_trip_blobs_corners)      { testdata_roundtrip("testdata/blobs-corners.png"); }
+MU_TEST(test_round_trip_blobs_crossing)     { testdata_roundtrip("testdata/blobs-crossing.png"); }
+MU_TEST(test_round_trip_cross)              { testdata_roundtrip("testdata/cross.png"); }
+MU_TEST(test_round_trip_letter_e)           { testdata_roundtrip("testdata/letter-e.png"); }
+MU_TEST(test_round_trip_paper_example)      { testdata_roundtrip("testdata/paper-example.png"); }
+MU_TEST(test_round_trip_paper_example_inv)  { testdata_roundtrip("testdata/paper-example-inv.png"); }
+MU_TEST(test_round_trip_single_px)          { testdata_roundtrip("testdata/single-px.png"); }
+MU_TEST(test_round_trip_single_px_inv)      { testdata_roundtrip("testdata/single-px-inv.png"); }
+MU_TEST(test_round_trip_two_blobs)          { testdata_roundtrip("testdata/two-blobs.png"); }
+MU_TEST(test_round_trip_two_px)             { testdata_roundtrip("testdata/two-px.png"); }
+MU_TEST(test_round_trip_two_px_inv)         { testdata_roundtrip("testdata/two-px-inv.png"); }
+
+static void test_polygon_area(const char *fn) {
+    //cerr << endl << "poly area test " << fn << endl;
+    Image32 ref_img;
+    mu_assert(ref_img.load(fn), "Input image failed to load");
+    ref_img.binarize();
+
+    int white_px_count = 0;
+    int black_px_count = 0;
+    for (int y=0; y<ref_img.rows(); y++) {
+        for (int x=0; x<ref_img.cols(); x++) {
+            if (ref_img.at(x, y)) {
+                white_px_count += 1;
+            } else {
+                black_px_count += 1;
+            }
+        }
+    }
+
+    double pos_sum = 0.0;
+    double neg_sum = ref_img.size();
+    gerbolyze::nopencv::find_contours(ref_img, [&pos_sum, &neg_sum](Polygon_i& poly, ContourPolarity pol) {
+            double area = polygon_area(poly);
+            //cerr << endl << fn << ": " << area << " " << pos_sum << " / " << neg_sum << " -- " << white_px_count << " / " << black_px_count << " GOT: " << poly.size() << " w/ " << pol << endl;
+            mu_assert(fabs(area) > 0.99, "Polygon smaller than a single pixel");
+            mu_assert((pol == CP_CONTOUR) == (area >= 0), "Polygon area has incorrect sign");
+
+            pos_sum += area;
+            neg_sum -= area;
+        });
+
+    mu_assert(pos_sum - white_px_count < 0.01, "Calculated area outside tolerance");
+    mu_assert(neg_sum - black_px_count < 0.01, "Calculated area outside tolerance");
+    //cerr << endl << "poly area test " << fn << " done" << endl;
+}
+
+MU_TEST(test_polygon_area_blank)              { test_polygon_area("testdata/blank.png"); }
+MU_TEST(test_polygon_area_white)              { test_polygon_area("testdata/white.png"); }
+MU_TEST(test_polygon_area_blob_border_w)      { test_polygon_area("testdata/blob-border-w.png"); }
+MU_TEST(test_polygon_area_blobs_borders)      { test_polygon_area("testdata/blobs-borders.png"); }
+MU_TEST(test_polygon_area_blobs_corners)      { test_polygon_area("testdata/blobs-corners.png"); }
+MU_TEST(test_polygon_area_blobs_crossing)     { test_polygon_area("testdata/blobs-crossing.png"); }
+MU_TEST(test_polygon_area_cross)              { test_polygon_area("testdata/cross.png"); }
+MU_TEST(test_polygon_area_letter_e)           { test_polygon_area("testdata/letter-e.png"); }
+MU_TEST(test_polygon_area_paper_example)      { test_polygon_area("testdata/paper-example.png"); }
+MU_TEST(test_polygon_area_paper_example_inv)  { test_polygon_area("testdata/paper-example-inv.png"); }
+MU_TEST(test_polygon_area_single_px)          { test_polygon_area("testdata/single-px.png"); }
+MU_TEST(test_polygon_area_single_px_inv)      { test_polygon_area("testdata/single-px-inv.png"); }
+MU_TEST(test_polygon_area_two_blobs)          { test_polygon_area("testdata/two-blobs.png"); }
+MU_TEST(test_polygon_area_two_px)             { test_polygon_area("testdata/two-px.png"); }
+MU_TEST(test_polygon_area_two_px_inv)         { test_polygon_area("testdata/two-px-inv.png"); }
+
+static void chain_approx_test(const char *fn) {
+    //cout << endl << "Testing \"" << fn << "\"" << endl;
+    Image32 ref_img;
+    mu_assert(ref_img.load(fn), "Input image failed to load");
+    ref_img.binarize();
+    Image32 ref_img_copy(ref_img);
+
+    TempfileHack tmp_svg(".svg");
+    TempfileHack tmp_png(".png");
+
+    SVGPolyRenderer ctx(tmp_svg.c_str(), ref_img.cols(), ref_img.rows());
+    gerbolyze::nopencv::find_contours(ref_img, simplify_contours_teh_chin(ctx.callback()));
+    ctx.close();
+
+    mu_assert_int_eq(0, render_svg(tmp_svg.c_str(), tmp_png.c_str()));
+
+    Image32 out_img;
+    mu_assert(out_img.load(tmp_png.c_str()), "Output image failed to load");
+    mu_assert_int_eq(ref_img.rows(), out_img.rows());
+    mu_assert_int_eq(ref_img.cols(), out_img.cols());
+
+    double max_abs_deviation = 0;
+    double rms_sum = 0; 
+    double mean_sum = 0; 
+    for (int y=0; y<out_img.rows(); y++) {
+        for (int x=0; x<out_img.cols(); x++) {
+            double delta = fabs((double)out_img.at(x, y)/255.0 - (double)ref_img_copy.at(x, y));
+            max_abs_deviation = fmax(max_abs_deviation, delta);
+            rms_sum += delta*delta;
+            mean_sum += delta;
+        }
+    }
+
+    rms_sum = sqrt(rms_sum / out_img.size());
+    mean_sum /= out_img.size();
+    if (rms_sum > 0.5) {
+        snprintf(msg, sizeof(msg), "%s: Chain approximation RMS error is above threshold: %.3f > 0.5\n", fn, rms_sum);
+        mu_fail(msg);
+    }
+    if (mean_sum > 0.1) {
+        snprintf(msg, sizeof(msg), "%s: Chain approximation mean error is above threshold: %.3f > 0.1\n", fn, mean_sum);
+        mu_fail(msg);
+    }
+    //mu_assert(max_abs_deviation < 0.5, "Maximum chain approximation error is above threshold");
+}
+
+
+MU_TEST(chain_approx_test_chromosome)         { chain_approx_test("testdata/chain-approx-teh-chin-chromosome.png"); }
+MU_TEST(chain_approx_test_blank)              { chain_approx_test("testdata/blank.png"); }
+MU_TEST(chain_approx_test_white)              { chain_approx_test("testdata/white.png"); }
+MU_TEST(chain_approx_test_blob_border_w)      { chain_approx_test("testdata/blob-border-w.png"); }
+MU_TEST(chain_approx_test_blobs_borders)      { chain_approx_test("testdata/blobs-borders.png"); }
+MU_TEST(chain_approx_test_blobs_corners)      { chain_approx_test("testdata/blobs-corners.png"); }
+MU_TEST(chain_approx_test_blobs_crossing)     { chain_approx_test("testdata/blobs-crossing.png"); }
+MU_TEST(chain_approx_test_cross)              { chain_approx_test("testdata/cross.png"); }
+MU_TEST(chain_approx_test_letter_e)           { chain_approx_test("testdata/letter-e.png"); }
+MU_TEST(chain_approx_test_paper_example)      { chain_approx_test("testdata/paper-example.png"); }
+MU_TEST(chain_approx_test_paper_example_inv)  { chain_approx_test("testdata/paper-example-inv.png"); }
+MU_TEST(chain_approx_test_single_px)          { chain_approx_test("testdata/single-px.png"); }
+MU_TEST(chain_approx_test_single_px_inv)      { chain_approx_test("testdata/single-px-inv.png"); }
+MU_TEST(chain_approx_test_two_blobs)          { chain_approx_test("testdata/two-blobs.png"); }
+MU_TEST(chain_approx_test_two_px)             { chain_approx_test("testdata/two-px.png"); }
+MU_TEST(chain_approx_test_two_px_inv)         { chain_approx_test("testdata/two-px-inv.png"); }
+
+
+MU_TEST_SUITE(nopencv_contours_suite) {
+    MU_RUN_TEST(test_complex_example_from_paper);
+
+    MU_RUN_TEST(test_round_trip_blank);
+    MU_RUN_TEST(test_round_trip_white);
+    MU_RUN_TEST(test_round_trip_blob_border_w);
+    MU_RUN_TEST(test_round_trip_blobs_borders);
+    MU_RUN_TEST(test_round_trip_blobs_corners);
+    MU_RUN_TEST(test_round_trip_blobs_crossing);
+    MU_RUN_TEST(test_round_trip_cross);
+    MU_RUN_TEST(test_round_trip_letter_e);
+    MU_RUN_TEST(test_round_trip_paper_example);
+    MU_RUN_TEST(test_round_trip_paper_example_inv);
+    MU_RUN_TEST(test_round_trip_single_px);
+    MU_RUN_TEST(test_round_trip_single_px_inv);
+    MU_RUN_TEST(test_round_trip_two_blobs);
+    MU_RUN_TEST(test_round_trip_two_px);
+    MU_RUN_TEST(test_round_trip_two_px_inv);
+
+    MU_RUN_TEST(chain_approx_test_chromosome);
+    MU_RUN_TEST(chain_approx_test_blank);
+    MU_RUN_TEST(chain_approx_test_white);
+    MU_RUN_TEST(chain_approx_test_blob_border_w);
+    MU_RUN_TEST(chain_approx_test_blobs_borders);
+    MU_RUN_TEST(chain_approx_test_blobs_corners);
+    MU_RUN_TEST(chain_approx_test_blobs_crossing);
+    MU_RUN_TEST(chain_approx_test_cross);
+    MU_RUN_TEST(chain_approx_test_letter_e);
+    MU_RUN_TEST(chain_approx_test_paper_example);
+    MU_RUN_TEST(chain_approx_test_paper_example_inv);
+    MU_RUN_TEST(chain_approx_test_single_px);
+    MU_RUN_TEST(chain_approx_test_single_px_inv);
+    MU_RUN_TEST(chain_approx_test_two_blobs);
+    MU_RUN_TEST(chain_approx_test_two_px);
+    MU_RUN_TEST(chain_approx_test_two_px_inv);
+
+    MU_RUN_TEST(test_polygon_area_blank);
+    MU_RUN_TEST(test_polygon_area_white);
+    MU_RUN_TEST(test_polygon_area_blob_border_w);
+    MU_RUN_TEST(test_polygon_area_blobs_borders);
+    MU_RUN_TEST(test_polygon_area_blobs_corners);
+    MU_RUN_TEST(test_polygon_area_blobs_crossing);
+    MU_RUN_TEST(test_polygon_area_cross);
+    MU_RUN_TEST(test_polygon_area_letter_e);
+    MU_RUN_TEST(test_polygon_area_paper_example);
+    MU_RUN_TEST(test_polygon_area_paper_example_inv);
+    MU_RUN_TEST(test_polygon_area_single_px);
+    MU_RUN_TEST(test_polygon_area_single_px_inv);
+    MU_RUN_TEST(test_polygon_area_two_blobs);
+    MU_RUN_TEST(test_polygon_area_two_px);
+    MU_RUN_TEST(test_polygon_area_two_px_inv);
+};
+
+int main(int argc, char **argv) {
+    (void)argc;
+    (void)argv;
+
+    MU_RUN_SUITE(nopencv_contours_suite);
+    MU_REPORT();
+    return MU_EXIT_CODE;
+}
diff --git a/svg-flatten/src/test/svg_tests.py b/svg-flatten/src/test/svg_tests.py
new file mode 100644
index 0000000..a5139ec
--- /dev/null
+++ b/svg-flatten/src/test/svg_tests.py
@@ -0,0 +1,75 @@
+#!/usr/bin/env python3
+
+import tempfile
+import unittest
+from pathlib import Path
+import subprocess
+import os
+
+from PIL import Image
+import numpy as np
+
+def run_svg_flatten(input_file, output_file, **kwargs):
+    if 'SVG_FLATTEN' in os.environ:
+        svg_flatten = os.environ.get('SVG_FLATTEN')
+    elif (Path(__file__) / '../../build/svg-flatten').is_file():
+        svg_flatten = '../../build/svg-flatten'
+    elif Path('./build/svg-flatten').is_file():
+        svg_flatten = './build/svg-flatten'
+    else:
+        svg_flatten = 'svg-flatten'
+
+    args = [ svg_flatten,
+            *(arg for (key, value) in kwargs.items() for arg in (f'--{key.replace("_", "-")}', value)),
+            str(input_file), str(output_file) ]
+
+    try:
+        proc = subprocess.run(args, capture_output=True, check=True)
+    except:
+        print('Subprocess stdout:')
+        print(proc.stdout)
+        print('Subprocess stderr:')
+        print(proc.stderr)
+        raise
+
+class SVGRoundTripTests(unittest.TestCase):
+
+    def compare_images(self, reference, output, test_name, mean=0.01):
+        ref = np.array(Image.open(reference))
+        out = np.array(Image.open(output))
+        delta = np.abs(out - ref).astype(float) / 255
+
+        #print(f'{test_name}: mean={delta.mean():.5g}')
+
+        self.assertTrue(delta.mean() < mean,
+                f'Expected mean pixel difference between images to be <{mean}, was {delta.mean():.5g}')
+
+    def run_svg_round_trip_test(self, test_in_svg):
+        with tempfile.NamedTemporaryFile(suffix='.svg') as tmp_out_svg,\
+            tempfile.NamedTemporaryFile(suffix='.png') as tmp_out_png,\
+            tempfile.NamedTemporaryFile(suffix='.png') as tmp_in_png:
+
+            run_svg_flatten(test_in_svg, tmp_out_svg.name, format='svg')
+
+            subprocess.run(['resvg', tmp_out_svg.name, tmp_out_png.name], check=True, stdout=subprocess.DEVNULL)
+            subprocess.run(['resvg', test_in_svg, tmp_in_png.name], check=True, stdout=subprocess.DEVNULL)
+
+            try:
+                self.compare_images(tmp_in_png, tmp_out_png, test_in_svg.stem)
+            except AssertionError as e:
+                import shutil
+                shutil.copyfile(tmp_in_png.name, f'/tmp/gerbolyze-fail-{test_in_svg.stem}-in.png')
+                shutil.copyfile(tmp_out_png.name, f'/tmp/gerbolyze-fail-{test_in_svg.stem}-out.png')
+                foo = list(e.args)
+                foo[0] += '\nFailing test renderings copied to:\n'
+                foo[0] += f'  /tmp/gerbolyze-fail-{test_in_svg.stem}-{{in|out}}.png\n'
+                e.args = tuple(foo)
+                raise e
+
+for test_in_svg in Path('testdata/svg').glob('*.svg'):
+    # We need to make sure we capture the loop variable's current value here.
+    gen = lambda testcase: lambda self: self.run_svg_round_trip_test(testcase)
+    setattr(SVGRoundTripTests, f'test_{test_in_svg.stem}', gen(test_in_svg))
+
+if __name__ == '__main__':
+    unittest.main()
diff --git a/svg-flatten/testdata/svg/circles.svg b/svg-flatten/testdata/svg/circles.svg
new file mode 100644
index 0000000..9be474e
--- /dev/null
+++ b/svg-flatten/testdata/svg/circles.svg
@@ -0,0 +1,78 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="circles.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.4553"
+     inkscape:cy="97.954218"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <circle
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="path950"
+     cx="16.593897"
+     cy="20.344759"
+     r="8.5389299" />
+  <ellipse
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="path952"
+     cx="30.070639"
+     cy="28.833015"
+     rx="4.5358982"
+     ry="8.5728388" />
+  <ellipse
+     style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="path954"
+     cx="26.473625"
+     cy="24.711111"
+     rx="11.056757"
+     ry="5.5283785" />
+  <ellipse
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="path952-5"
+     cx="33.704639"
+     cy="15.709757"
+     rx="4.5358982"
+     ry="8.5728388" />
+</svg>
diff --git a/svg-flatten/testdata/svg/compound_xform.svg b/svg-flatten/testdata/svg/compound_xform.svg
new file mode 100644
index 0000000..229e05d
--- /dev/null
+++ b/svg-flatten/testdata/svg/compound_xform.svg
@@ -0,0 +1,102 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="compound_xform.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="83.428533"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <g
+     id="g1200">
+    <g
+       id="g1190"
+       transform="rotate(-6.212576,25,25)">
+      <rect
+         style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+         id="rect1166"
+         width="36.952793"
+         height="36.952793"
+         x="6.5236034"
+         y="6.5236034" />
+      <rect
+         style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+         id="rect1166-9"
+         width="32.644421"
+         height="32.644421"
+         x="8.6777897"
+         y="8.6777897" />
+    </g>
+    <g
+       id="g1164">
+      <g
+         id="g1154"
+         transform="matrix(-1,0,0,1,50,0)">
+        <ellipse
+           style="fill:#000000;stroke:none;stroke-width:0.0307622;stroke-linejoin:round;stop-color:#000000"
+           id="path1130"
+           cx="47.739796"
+           cy="33.794945"
+           rx="14.123164"
+           ry="19.091663"
+           transform="matrix(1,0,-0.67287566,0.7397556,0,0)" />
+        <rect
+           style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+           id="rect1150"
+           width="22.3258"
+           height="20.708179"
+           x="13.8371"
+           y="14.64591" />
+      </g>
+      <rect
+         style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+         id="rect1158"
+         width="7.5826759"
+         height="7.5826759"
+         x="31.564001"
+         y="-3.791338"
+         transform="rotate(45)" />
+    </g>
+  </g>
+</svg>
diff --git a/svg-flatten/testdata/svg/empty.svg b/svg-flatten/testdata/svg/empty.svg
new file mode 100644
index 0000000..d2cdc52
--- /dev/null
+++ b/svg-flatten/testdata/svg/empty.svg
@@ -0,0 +1,11 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8">
+</svg>
diff --git a/svg-flatten/testdata/svg/empty_inkscape.svg b/svg-flatten/testdata/svg/empty_inkscape.svg
new file mode 100644
index 0000000..716379f
--- /dev/null
+++ b/svg-flatten/testdata/svg/empty_inkscape.svg
@@ -0,0 +1,54 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)"
+   sodipodi:docname="empty_inkscape.svg">
+  <defs
+     id="defs2" />
+  <sodipodi:namedview
+     id="base"
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1.0"
+     inkscape:pageopacity="0.0"
+     inkscape:pageshadow="2"
+     inkscape:zoom="1.7472574"
+     inkscape:cx="76.820344"
+     inkscape:cy="114.80779"
+     inkscape:document-units="mm"
+     inkscape:current-layer="layer1"
+     inkscape:document-rotation="0"
+     showgrid="false"
+     inkscape:window-width="1530"
+     inkscape:window-height="669"
+     inkscape:window-x="390"
+     inkscape:window-y="0"
+     inkscape:window-maximized="0" />
+  <metadata
+     id="metadata5">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <g
+     inkscape:label="Layer 1"
+     inkscape:groupmode="layer"
+     id="layer1" />
+</svg>
diff --git a/svg-flatten/testdata/svg/groups.svg b/svg-flatten/testdata/svg/groups.svg
new file mode 100644
index 0000000..d94dad0
--- /dev/null
+++ b/svg-flatten/testdata/svg/groups.svg
@@ -0,0 +1,96 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="groups.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="97.954218"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect1002"
+     width="5.4755607"
+     height="36.937885"
+     x="22.069475"
+     y="5.679245" />
+  <g
+     id="g998"
+     transform="translate(0.10591836,3.2209219)">
+    <circle
+       style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+       id="path950"
+       cx="16.593897"
+       cy="20.344759"
+       r="8.5389299" />
+    <ellipse
+       style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+       id="path952"
+       cx="30.070639"
+       cy="28.833015"
+       rx="4.5358982"
+       ry="8.5728388" />
+    <ellipse
+       style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+       id="path954"
+       cx="26.473625"
+       cy="24.711111"
+       rx="11.056757"
+       ry="5.5283785" />
+    <ellipse
+       style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+       id="path952-5"
+       cx="33.704639"
+       cy="15.709757"
+       rx="4.5358982"
+       ry="8.5728388" />
+  </g>
+  <rect
+     style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect1000"
+     width="31.061703"
+     height="4.7948809"
+     x="16.278164"
+     y="13.511433" />
+</svg>
diff --git a/svg-flatten/testdata/svg/pattern_fill.svg b/svg-flatten/testdata/svg/pattern_fill.svg
new file mode 100644
index 0000000..21789e6
--- /dev/null
+++ b/svg-flatten/testdata/svg/pattern_fill.svg
@@ -0,0 +1,135 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:xlink="http://www.w3.org/1999/xlink"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="pattern_fill.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5">
+    <pattern
+       inkscape:collect="always"
+       xlink:href="#Strips2_1"
+       id="pattern3491"
+       patternTransform="matrix(1.774288,0.91540651,-2.3098808,4.4771298,0,1.6295182e-8)" />
+    <pattern
+       inkscape:collect="always"
+       patternUnits="userSpaceOnUse"
+       width="1.5"
+       height="1"
+       patternTransform="translate(0,0) scale(10,10)"
+       id="Strips2_1"
+       inkscape:stockid="Stripes 2:1">
+      <rect
+         style="fill:black;stroke:none"
+         x="0"
+         y="-0.5"
+         width="1"
+         height="2"
+         id="rect2399" />
+    </pattern>
+    <pattern
+       inkscape:collect="always"
+       xlink:href="#Packedcircles"
+       id="pattern3461"
+       patternTransform="matrix(5.9164411,-2.7358919,2.7358917,5.9164404,9.4471894,8.2115447)" />
+    <pattern
+       inkscape:collect="always"
+       patternUnits="userSpaceOnUse"
+       width="1"
+       height="1.73205080756"
+       patternTransform="translate(0,0) scale(10,10)"
+       id="Packedcircles"
+       inkscape:stockid="Packed circles">
+      <circle
+         style="fill:black;stroke:none"
+         cx="0"
+         cy="0.5"
+         r="0.5"
+         id="circle2421" />
+      <circle
+         style="fill:black;stroke:none"
+         cx="1"
+         cy="0.5"
+         r="0.5"
+         id="circle2423" />
+      <circle
+         style="fill:black;stroke:none"
+         cx="0.5"
+         cy="1.36602540378"
+         r="0.5"
+         id="circle2425" />
+      <circle
+         style="fill:black;stroke:none"
+         cx="0.5"
+         cy="-0.366025403784"
+         r="0.5"
+         id="circle2427" />
+    </pattern>
+  </defs>
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.440884"
+     inkscape:cx="168.05451"
+     inkscape:cy="88.906258"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <circle
+     style="opacity:0.99435;fill:url(#pattern3461);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4.00001, 4.00001;stroke-dashoffset:2.19001;stop-color:#000000"
+     id="path1374"
+     cx="31.476124"
+     cy="30.316647"
+     r="13.079776" />
+  <rect
+     style="opacity:0.99435;fill:url(#pattern3491);fill-opacity:1;stroke:none;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4, 4;stroke-dashoffset:2.19;stop-color:#000000"
+     id="rect3463"
+     width="23.181004"
+     height="23.181004"
+     x="6.0242319"
+     y="6.351799" />
+  <rect
+     style="opacity:0.99435;fill:#ffffff;stroke:none;stroke-width:1.16072;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4.64288, 4.64288;stroke-dashoffset:2.54198;stop-color:#000000"
+     id="rect3493"
+     width="13.836466"
+     height="59.941151"
+     x="40.70414"
+     y="5.6588583"
+     transform="matrix(1,0,-0.67013266,0.74224134,0,0)" />
+</svg>
diff --git a/svg-flatten/testdata/svg/pattern_stroke.svg b/svg-flatten/testdata/svg/pattern_stroke.svg
new file mode 100644
index 0000000..5564df0
--- /dev/null
+++ b/svg-flatten/testdata/svg/pattern_stroke.svg
@@ -0,0 +1,89 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:xlink="http://www.w3.org/1999/xlink"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="pattern_stroke.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5">
+    <pattern
+       inkscape:collect="always"
+       xlink:href="#Checkerboard"
+       id="pattern3520"
+       patternTransform="scale(1.9742409,1.8180095)" />
+    <pattern
+       inkscape:collect="always"
+       patternUnits="userSpaceOnUse"
+       width="2"
+       height="2"
+       patternTransform="translate(0,0) scale(10,10)"
+       id="Checkerboard"
+       inkscape:stockid="Checkerboard">
+      <rect
+         style="fill:black;stroke:none"
+         x="0"
+         y="0"
+         width="1"
+         height="1"
+         id="rect2411" />
+      <rect
+         style="fill:black;stroke:none"
+         x="1"
+         y="1"
+         width="1"
+         height="1"
+         id="rect2413" />
+    </pattern>
+  </defs>
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.440884"
+     inkscape:cx="168.05451"
+     inkscape:cy="88.906258"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <circle
+     style="opacity:0.99435;fill:none;stroke:url(#pattern3520);stroke-width:8;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-dashoffset:10.95;stroke-opacity:1;stop-color:#000000"
+     id="path3513"
+     cx="25"
+     cy="25"
+     r="13.90474" />
+</svg>
diff --git a/svg-flatten/testdata/svg/pattern_stroke_dashed.svg b/svg-flatten/testdata/svg/pattern_stroke_dashed.svg
new file mode 100644
index 0000000..3d8145e
--- /dev/null
+++ b/svg-flatten/testdata/svg/pattern_stroke_dashed.svg
@@ -0,0 +1,89 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:xlink="http://www.w3.org/1999/xlink"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="pattern_stroke_dashed.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5">
+    <pattern
+       inkscape:collect="always"
+       xlink:href="#Checkerboard"
+       id="pattern3520"
+       patternTransform="scale(1.9742409,1.8180095)" />
+    <pattern
+       inkscape:collect="always"
+       patternUnits="userSpaceOnUse"
+       width="2"
+       height="2"
+       patternTransform="translate(0,0) scale(10,10)"
+       id="Checkerboard"
+       inkscape:stockid="Checkerboard">
+      <rect
+         style="fill:black;stroke:none"
+         x="0"
+         y="0"
+         width="1"
+         height="1"
+         id="rect2411" />
+      <rect
+         style="fill:black;stroke:none"
+         x="1"
+         y="1"
+         width="1"
+         height="1"
+         id="rect2413" />
+    </pattern>
+  </defs>
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.440884"
+     inkscape:cx="168.05451"
+     inkscape:cy="88.906258"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <circle
+     style="opacity:0.99435;fill:#000000;stroke:url(#pattern3520);stroke-width:12;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:6,6;stroke-dashoffset:0;stroke-opacity:1;stop-color:#000000"
+     id="path3513"
+     cx="25"
+     cy="25"
+     r="13.90474" />
+</svg>
diff --git a/svg-flatten/testdata/svg/rect.svg b/svg-flatten/testdata/svg/rect.svg
new file mode 100644
index 0000000..297ff7e
--- /dev/null
+++ b/svg-flatten/testdata/svg/rect.svg
@@ -0,0 +1,65 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="rect.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="2.6961137"
+     inkscape:cx="84.54603"
+     inkscape:cy="109.58858"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect832"
+     width="20"
+     height="15"
+     x="10"
+     y="15" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect832-3"
+     width="20"
+     height="15"
+     x="19.318335"
+     y="22.307203" />
+</svg>
diff --git a/svg-flatten/testdata/svg/rect_occlusion.svg b/svg-flatten/testdata/svg/rect_occlusion.svg
new file mode 100644
index 0000000..ce0e531
--- /dev/null
+++ b/svg-flatten/testdata/svg/rect_occlusion.svg
@@ -0,0 +1,79 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="rect_occlusion.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="6.608567"
+     inkscape:cx="109.89929"
+     inkscape:cy="79.489935"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect832"
+     width="20"
+     height="15"
+     x="10"
+     y="15" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect832-3"
+     width="20"
+     height="15"
+     x="19.318335"
+     y="22.307203" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect832-3-6"
+     width="20"
+     height="15"
+     x="19.318335"
+     y="22.307203" />
+  <rect
+     style="fill:#ffffff;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect832-3-6-7"
+     width="20"
+     height="15"
+     x="23.297699"
+     y="13.616298" />
+</svg>
diff --git a/svg-flatten/testdata/svg/rotation.svg b/svg-flatten/testdata/svg/rotation.svg
new file mode 100644
index 0000000..961ef0f
--- /dev/null
+++ b/svg-flatten/testdata/svg/rotation.svg
@@ -0,0 +1,66 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="rotation.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="97.954218"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect1022"
+     width="30"
+     height="10"
+     x="10"
+     y="10" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect1022-3"
+     width="30"
+     height="10"
+     x="-3.3479581"
+     y="-44.595863"
+     transform="rotate(120.85838)" />
+</svg>
diff --git a/svg-flatten/testdata/svg/rotation_90.svg b/svg-flatten/testdata/svg/rotation_90.svg
new file mode 100644
index 0000000..dcca867
--- /dev/null
+++ b/svg-flatten/testdata/svg/rotation_90.svg
@@ -0,0 +1,66 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="rotation_90.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="97.954218"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect1022"
+     width="30"
+     height="10"
+     x="10"
+     y="10" />
+  <rect
+     style="fill:#000000;stroke:none;stroke-width:0.0264583;stroke-linejoin:round;stop-color:#000000"
+     id="rect1022-3"
+     width="30"
+     height="10"
+     x="15"
+     y="-30"
+     transform="rotate(90)" />
+</svg>
diff --git a/svg-flatten/testdata/svg/scale.svg b/svg-flatten/testdata/svg/scale.svg
new file mode 100644
index 0000000..91b28d6
--- /dev/null
+++ b/svg-flatten/testdata/svg/scale.svg
@@ -0,0 +1,91 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="scale.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="97.954218"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <g
+     id="g1082"
+     transform="matrix(0.81799278,0,0,0.81799278,12.474166,-0.98587389)"
+     style="stroke-width:1.2225">
+    <rect
+       style="fill:#000000;stroke:none;stroke-width:0.0323454;stroke-linejoin:round;stop-color:#000000"
+       id="rect1022"
+       width="30"
+       height="10"
+       x="10"
+       y="10" />
+    <rect
+       style="fill:#000000;stroke:none;stroke-width:0.0323454;stroke-linejoin:round;stop-color:#000000"
+       id="rect1022-3"
+       width="30"
+       height="10"
+       x="-3.3479581"
+       y="-44.595863"
+       transform="rotate(120.85838)" />
+  </g>
+  <g
+     id="g1082-5"
+     transform="matrix(0.61146823,0,0,0.61146823,-0.17038858,14.878729)"
+     style="stroke-width:1.63541">
+    <rect
+       style="fill:#000000;stroke:none;stroke-width:0.0432701;stroke-linejoin:round;stop-color:#000000"
+       id="rect1022-6"
+       width="30"
+       height="10"
+       x="10"
+       y="10" />
+    <rect
+       style="fill:#000000;stroke:none;stroke-width:0.0432701;stroke-linejoin:round;stop-color:#000000"
+       id="rect1022-3-2"
+       width="30"
+       height="10"
+       x="-3.3479581"
+       y="-44.595863"
+       transform="rotate(120.85838)" />
+  </g>
+</svg>
diff --git a/svg-flatten/testdata/svg/shear.svg b/svg-flatten/testdata/svg/shear.svg
new file mode 100644
index 0000000..ec7875f
--- /dev/null
+++ b/svg-flatten/testdata/svg/shear.svg
@@ -0,0 +1,59 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="shear.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="97.954218"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8" />
+  <ellipse
+     style="fill:#000000;stroke:none;stroke-width:0.0307622;stroke-linejoin:round;stop-color:#000000"
+     id="path1130"
+     cx="47.739796"
+     cy="33.794945"
+     rx="14.123164"
+     ry="19.091663"
+     transform="matrix(1,0,-0.67287566,0.7397556,0,0)" />
+</svg>
diff --git a/svg-flatten/testdata/svg/stroke.svg b/svg-flatten/testdata/svg/stroke.svg
new file mode 100644
index 0000000..6a4b334
--- /dev/null
+++ b/svg-flatten/testdata/svg/stroke.svg
@@ -0,0 +1,67 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="stroke.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.2278448"
+     inkscape:cx="83.06317"
+     inkscape:cy="83.428533"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <rect
+     style="fill:#ffffff;stroke:#000000;stroke-width:9.99998;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stop-color:#000000"
+     id="rect1220"
+     width="30"
+     height="30.000029"
+     x="10"
+     y="10" />
+  <rect
+     style="fill:#ffffff;stroke:#000000;stroke-width:2;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stop-color:#000000"
+     id="rect1220-1"
+     width="11"
+     height="11"
+     x="19.5"
+     y="19.5" />
+</svg>
diff --git a/svg-flatten/testdata/svg/stroke_caps.svg b/svg-flatten/testdata/svg/stroke_caps.svg
new file mode 100644
index 0000000..6b89a34
--- /dev/null
+++ b/svg-flatten/testdata/svg/stroke_caps.svg
@@ -0,0 +1,68 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="stroke_caps.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.4020648"
+     inkscape:cx="99.189726"
+     inkscape:cy="80.388351"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:round;stroke-opacity:1;stroke-miterlimit:4;stroke-dasharray:none"
+     d="M 8.1209509,7.3978548 20.159949,7.3803896 9.2562029,14.249268 c 19.4757851,0.912458 -2.156778,6.498823 6.5634261,12.04404"
+     id="path1278"
+     sodipodi:nodetypes="cccc" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:round;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+     d="M 28.369293,7.3978548 40.408291,7.3803896 29.504545,14.249268 c 19.475785,0.912458 -2.156778,6.498823 6.563426,12.04404"
+     id="path1278-2"
+     sodipodi:nodetypes="cccc" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:square;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+     d="m 17.843819,42.603963 12.038998,0.01746 -10.903746,-6.868878 c 19.475785,-0.912458 -2.156778,-6.498823 6.563426,-12.04404"
+     id="path1278-7"
+     sodipodi:nodetypes="cccc" />
+</svg>
diff --git a/svg-flatten/testdata/svg/stroke_dashes.svg b/svg-flatten/testdata/svg/stroke_dashes.svg
new file mode 100644
index 0000000..c1d6f5c
--- /dev/null
+++ b/svg-flatten/testdata/svg/stroke_dashes.svg
@@ -0,0 +1,75 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="stroke_dashes.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.4020648"
+     inkscape:cx="99.189726"
+     inkscape:cy="80.388351"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:3.00000008;stroke-linecap:butt;stroke-linejoin:round;stroke-opacity:1;stroke-miterlimit:4;stroke-dasharray:3.00000008,3.00000008;stroke-dashoffset:0"
+     d="M 8.1209509,7.3978548 20.159949,7.3803896 9.2562029,14.249268 c 19.4757851,0.912458 -2.156778,6.498823 6.5634261,12.04404"
+     id="path1278"
+     sodipodi:nodetypes="cccc" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:2, 1, 0.50000000000000000, 1;stroke-opacity:1;stroke-dashoffset:0;opacity:0.99435036"
+     d="M 28.369293,7.3978548 40.408291,7.3803896 29.504545,14.249268 c 19.475785,0.912458 -2.156778,6.498823 6.563426,12.04404"
+     id="path1278-2"
+     sodipodi:nodetypes="cccc" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:0.1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:0.4,0.4;stroke-opacity:1;stroke-dashoffset:0"
+     d="m 17.843819,42.603963 12.038998,0.01746 -10.903746,-6.868878 c 19.475785,-0.912458 -2.156778,-6.498823 6.563426,-12.04404"
+     id="path1278-7"
+     sodipodi:nodetypes="cccc" />
+  <rect
+     style="opacity:0.99435;fill:none;stroke:#000000;stroke-width:1;stroke-linecap:butt;stroke-linejoin:round;stroke-miterlimit:4;stroke-dasharray:4, 4;stroke-dashoffset:2.19;stop-color:#000000"
+     id="rect1354"
+     width="43.84502"
+     height="43.932682"
+     x="3.0774899"
+     y="3.033659" />
+</svg>
diff --git a/svg-flatten/testdata/svg/stroke_joins.svg b/svg-flatten/testdata/svg/stroke_joins.svg
new file mode 100644
index 0000000..ee3e1ac
--- /dev/null
+++ b/svg-flatten/testdata/svg/stroke_joins.svg
@@ -0,0 +1,68 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="stroke_joins.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5" />
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.4020648"
+     inkscape:cx="99.189726"
+     inkscape:cy="80.388351"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:round;stroke-opacity:1;stroke-miterlimit:4;stroke-dasharray:none"
+     d="M 8.1209509,7.3978548 20.159949,7.3803896 9.2562029,14.249268 c 19.4757851,0.912458 -2.156778,6.498823 6.5634261,12.04404"
+     id="path1278"
+     sodipodi:nodetypes="cccc" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:bevel;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+     d="M 28.369293,7.3978548 40.408291,7.3803896 29.504545,14.249268 c 19.475785,0.912458 -2.156778,6.498823 6.563426,12.04404"
+     id="path1278-2"
+     sodipodi:nodetypes="cccc" />
+  <path
+     style="fill:none;stroke:#000000;stroke-width:5;stroke-linecap:butt;stroke-linejoin:miter;stroke-miterlimit:4;stroke-dasharray:none;stroke-opacity:1"
+     d="m 17.843819,42.603963 12.038998,0.01746 -10.903746,-6.868878 c 19.475785,-0.912458 -2.156778,-6.498823 6.563426,-12.04404"
+     id="path1278-7"
+     sodipodi:nodetypes="cccc" />
+</svg>
diff --git a/svg-flatten/testdata/svg/text.svg b/svg-flatten/testdata/svg/text.svg
new file mode 100644
index 0000000..bd8c390
--- /dev/null
+++ b/svg-flatten/testdata/svg/text.svg
@@ -0,0 +1,108 @@
+<?xml version="1.0" encoding="UTF-8" standalone="no"?>
+<svg
+   xmlns:dc="http://purl.org/dc/elements/1.1/"
+   xmlns:cc="http://creativecommons.org/ns#"
+   xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
+   xmlns:svg="http://www.w3.org/2000/svg"
+   xmlns="http://www.w3.org/2000/svg"
+   xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
+   xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
+   width="50mm"
+   height="50mm"
+   viewBox="0 0 50 50"
+   version="1.1"
+   id="svg8"
+   sodipodi:docname="text.svg"
+   inkscape:version="1.0.2 (e86c870879, 2021-01-15, custom)">
+  <metadata
+     id="metadata7">
+    <rdf:RDF>
+      <cc:Work
+         rdf:about="">
+        <dc:format>image/svg+xml</dc:format>
+        <dc:type
+           rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
+        <dc:title></dc:title>
+      </cc:Work>
+    </rdf:RDF>
+  </metadata>
+  <defs
+     id="defs5">
+    <rect
+       x="3.0026957"
+       y="3.2239678"
+       width="65.46684"
+       height="67.27124"
+       id="rect3560" />
+    <rect
+       x="3.0026958"
+       y="3.2239678"
+       width="65.466843"
+       height="67.27124"
+       id="rect3560-2" />
+    <rect
+       x="3.0026958"
+       y="3.2239678"
+       width="65.466843"
+       height="67.27124"
+       id="rect3573" />
+    <rect
+       x="3.0026958"
+       y="3.2239678"
+       width="65.466843"
+       height="67.27124"
+       id="rect3560-2-4" />
+    <rect
+       x="3.0026958"
+       y="3.2239678"
+       width="65.466843"
+       height="67.27124"
+       id="rect3624" />
+  </defs>
+  <sodipodi:namedview
+     pagecolor="#ffffff"
+     bordercolor="#666666"
+     borderopacity="1"
+     objecttolerance="10"
+     gridtolerance="10"
+     guidetolerance="10"
+     inkscape:pageopacity="0"
+     inkscape:pageshadow="2"
+     inkscape:window-width="1920"
+     inkscape:window-height="1019"
+     id="namedview3"
+     showgrid="false"
+     inkscape:zoom="3.440884"
+     inkscape:cx="92.105039"
+     inkscape:cy="86.46986"
+     inkscape:window-x="0"
+     inkscape:window-y="0"
+     inkscape:window-maximized="1"
+     inkscape:current-layer="svg8"
+     inkscape:document-units="mm"
+     showguides="false" />
+  <text
+     xml:space="preserve"
+     id="text3558"
+     style="font-size:7.05556px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect3560);"
+     transform="translate(4.917252,17.244243)"><tspan
+       x="3.0019531"
+       y="9.5799363"><tspan
+         style="font-style:normal;font-variant:normal;font-weight:bold;font-stretch:normal;font-family:'Liberation Sans';-inkscape-font-specification:'Liberation Sans Bold'">Gerbolyze</tspan></tspan></text>
+  <text
+     xml:space="preserve"
+     id="text3558-0"
+     style="font-size:7.05556px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect3560-2);"
+     transform="matrix(1,-0.42912384,0,1,4.8672981,38.568218)"><tspan
+       x="3.0019531"
+       y="9.5799363"><tspan
+         style="font-style:italic;font-variant:normal;font-weight:bold;font-stretch:normal;font-family:'Liberation Sans';-inkscape-font-specification:'Liberation Sans Bold'">Gerbolyze</tspan></tspan></text>
+  <text
+     xml:space="preserve"
+     id="text3558-0-1"
+     style="font-size:7.05556px;line-height:1.25;font-family:sans-serif;white-space:pre;shape-inside:url(#rect3560-2-4);"
+     transform="matrix(1,-0.42912384,0,1,4.8672981,13.281795)"><tspan
+       x="3.0019531"
+       y="9.5799363"><tspan
+         style="font-style:italic;font-variant:normal;font-weight:bold;font-stretch:normal;font-family:'Liberation Sans';-inkscape-font-specification:'Liberation Sans Bold'">Gerbolyze</tspan></tspan></text>
+</svg>
-- 
cgit