libv8 5.7.492.65.1 → 5.9.211.38.0beta0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.gitignore +1 -0
- data/.travis.yml +8 -2
- data/CHANGELOG.md +4 -0
- data/README.md +4 -3
- data/Rakefile +27 -10
- data/ext/libv8/builder.rb +4 -0
- data/lib/libv8/version.rb +1 -1
- data/libv8.gemspec +1 -1
- data/patches/0001-Build-a-standalone-static-library.patch +5 -5
- data/patches/0002-Don-t-compile-unnecessary-stuff.patch +20 -24
- data/patches/0003-Use-the-fPIC-flag-for-the-static-library.patch +5 -5
- data/patches/0004-Do-not-embed-debug-symbols-in-macOS-libraries.patch +5 -5
- data/patches/0005-Fix-GCC-7-build-errors.patch +147 -0
- data/scaleway.png +0 -0
- data/vendor/depot_tools/.gitattributes +52 -0
- data/vendor/depot_tools/.gitignore +10 -0
- data/vendor/depot_tools/README.md +7 -3
- data/vendor/depot_tools/apply_issue.bat +4 -3
- data/vendor/depot_tools/apply_issue.py +0 -10
- data/vendor/depot_tools/autoninja +12 -0
- data/vendor/depot_tools/autoninja.bat +9 -0
- data/vendor/depot_tools/autoninja.py +70 -0
- data/vendor/depot_tools/bootstrap/win/README.md +108 -26
- data/vendor/depot_tools/bootstrap/win/git-bash.template.sh +3 -3
- data/vendor/depot_tools/bootstrap/win/git.template.bat +2 -2
- data/vendor/depot_tools/bootstrap/win/manifest.txt +18 -0
- data/vendor/depot_tools/bootstrap/win/manifest_bleeding_edge.txt +18 -0
- data/vendor/depot_tools/bootstrap/win/python27.new.bat +49 -0
- data/vendor/depot_tools/bootstrap/win/win_tools.bat +55 -59
- data/vendor/depot_tools/bootstrap/win/win_tools.py +335 -0
- data/vendor/depot_tools/cipd +16 -2
- data/vendor/depot_tools/cipd.bat +20 -2
- data/vendor/depot_tools/cipd.ps1 +36 -22
- data/vendor/depot_tools/cipd_bin_setup.bat +6 -0
- data/vendor/depot_tools/cipd_bin_setup.sh +10 -0
- data/vendor/depot_tools/cipd_client_version +1 -1
- data/vendor/depot_tools/cipd_manifest.txt +9 -0
- data/vendor/depot_tools/cit.bat +12 -11
- data/vendor/depot_tools/clang-format.bat +4 -3
- data/vendor/depot_tools/clang_format_merge_driver.bat +12 -11
- data/vendor/depot_tools/commit_queue.bat +4 -3
- data/vendor/depot_tools/cpplint.bat +9 -3
- data/vendor/depot_tools/cpplint.py +3 -3
- data/vendor/depot_tools/depot-tools-auth.bat +4 -3
- data/vendor/depot_tools/download_from_google_storage.bat +5 -3
- data/vendor/depot_tools/download_from_google_storage.py +6 -1
- data/vendor/depot_tools/fetch.bat +5 -4
- data/vendor/depot_tools/fetch.py +4 -5
- data/vendor/depot_tools/gclient-new-workdir.py +82 -46
- data/vendor/depot_tools/gclient.bat +5 -4
- data/vendor/depot_tools/gclient.py +713 -319
- data/vendor/depot_tools/gclient_eval.py +284 -0
- data/vendor/depot_tools/gclient_utils.py +70 -4
- data/vendor/depot_tools/gerrit_client.py +26 -1
- data/vendor/depot_tools/gerrit_util.py +120 -127
- data/vendor/depot_tools/git-crrev-parse +1 -0
- data/vendor/depot_tools/git-gs +1 -1
- data/vendor/depot_tools/git_cl.py +731 -415
- data/vendor/depot_tools/git_common.py +23 -3
- data/vendor/depot_tools/git_drover.py +10 -1
- data/vendor/depot_tools/git_footers.py +62 -22
- data/vendor/depot_tools/git_hyper_blame.py +3 -2
- data/vendor/depot_tools/git_map.py +30 -3
- data/vendor/depot_tools/git_map_branches.py +18 -4
- data/vendor/depot_tools/git_number.py +8 -2
- data/vendor/depot_tools/git_retry.py +21 -0
- data/vendor/depot_tools/gn.bat +4 -3
- data/vendor/depot_tools/infra/config/cq.cfg +1 -5
- data/vendor/depot_tools/infra/config/recipes.cfg +18 -16
- data/vendor/depot_tools/led +12 -0
- data/vendor/depot_tools/led.bat +7 -0
- data/vendor/depot_tools/man/html/git-cl.html +9 -1
- data/vendor/depot_tools/man/html/git-drover.html +22 -18
- data/vendor/depot_tools/man/man1/git-cl.1 +8 -3
- data/vendor/depot_tools/man/man1/git-drover.1 +22 -20
- data/vendor/depot_tools/man/src/git-cl.txt +3 -0
- data/vendor/depot_tools/man/src/git-drover.txt +8 -0
- data/vendor/depot_tools/my_activity.py +8 -5
- data/vendor/depot_tools/owners.py +103 -11
- data/vendor/depot_tools/owners_finder.py +14 -2
- data/vendor/depot_tools/presubmit_canned_checks.py +25 -67
- data/vendor/depot_tools/presubmit_support.py +87 -35
- data/vendor/depot_tools/recipes/OWNERS +2 -0
- data/vendor/depot_tools/recipes/README.recipes.md +842 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/__init__.py +5 -3
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/api.py +181 -60
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/{example.expected → examples/full.expected}/apply_gerrit_ref.json +4 -2
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/basic.json +82 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/basic_with_branch_heads.json +149 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/buildbot.json +82 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/clobber.json +149 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/deprecated_got_revision_mapping.json +122 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/gerrit_no_rebase_patch_ref.json +149 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/gerrit_no_reset.json +149 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/no_shallow.json +149 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/reset_root_solution_revision.json +148 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/trychange.json +150 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/{example.expected → examples/full.expected}/trychange_oauth2.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/trychange_oauth2_buildbot.json +152 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/trychange_oauth2_json.json +150 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/trychange_oauth2_json_win.json +150 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob.json +156 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_fail.json +91 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_fail_patch.json +118 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_fail_patch_download.json +118 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_gerrit_angle.json +202 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_gerrit_angle_deprecated.json +158 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_gerrit_feature_branch.json +196 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_gerrit_v8.json +202 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_gerrit_v8_feature_branch.json +202 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_v8.json +162 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/tryjob_v8_head_by_default.json +162 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/examples/full.expected/with_tags.json +149 -0
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/{example.py → examples/full.py} +46 -8
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/resources/bot_update.py +139 -133
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/test_api.py +25 -13
- data/vendor/depot_tools/recipes/recipe_modules/cipd/api.py +40 -9
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/basic.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/basic_pkg.json +1 -1
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/describe-failed.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/describe-many-instances.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/mac64.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/pkg_bad_file.json +1 -1
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/pkg_bad_mode.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/pkg_bad_verfile.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.expected → examples/full.expected}/win64.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/{example.py → examples/full.py} +1 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/junk arch.json +7 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/junk bits.json +7 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/linux_arm_32.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/linux_arm_64.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/linux_intel_32.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/linux_intel_64.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/linux_mips_64.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/mac_intel_64.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/win_intel_32.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.expected/win_intel_64.json +14 -0
- data/vendor/depot_tools/recipes/recipe_modules/cipd/examples/platform_suffix.py +59 -0
- data/vendor/depot_tools/recipes/recipe_modules/depot_tools/api.py +5 -0
- data/vendor/depot_tools/recipes/recipe_modules/depot_tools/{example.expected → examples/full.expected}/basic.json +7 -0
- data/vendor/depot_tools/recipes/recipe_modules/depot_tools/{example.expected → examples/full.expected}/win.json +7 -0
- data/vendor/depot_tools/recipes/recipe_modules/depot_tools/{example.py → examples/full.py} +2 -0
- data/vendor/depot_tools/recipes/recipe_modules/gclient/__init__.py +1 -4
- data/vendor/depot_tools/recipes/recipe_modules/gclient/api.py +28 -14
- data/vendor/depot_tools/recipes/recipe_modules/gclient/config.py +10 -235
- data/vendor/depot_tools/recipes/recipe_modules/gclient/{example.expected → examples/full.expected}/basic.json +9 -9
- data/vendor/depot_tools/recipes/recipe_modules/gclient/{example.expected → examples/full.expected}/buildbot.json +9 -9
- data/vendor/depot_tools/recipes/recipe_modules/gclient/{example.expected → examples/full.expected}/revision.json +9 -9
- data/vendor/depot_tools/recipes/recipe_modules/gclient/{example.expected → examples/full.expected}/tryserver.json +9 -9
- data/vendor/depot_tools/recipes/recipe_modules/gclient/{example.py → examples/full.py} +5 -21
- data/vendor/depot_tools/recipes/recipe_modules/gclient/tests/patch_project.py +45 -0
- data/vendor/depot_tools/recipes/recipe_modules/gerrit/__init__.py +1 -0
- data/vendor/depot_tools/recipes/recipe_modules/gerrit/api.py +97 -2
- data/vendor/depot_tools/recipes/recipe_modules/gerrit/examples/full.expected/basic.json +283 -0
- data/vendor/depot_tools/recipes/recipe_modules/gerrit/{example.py → examples/full.py} +31 -2
- data/vendor/depot_tools/recipes/recipe_modules/gerrit/test_api.py +30 -1
- data/vendor/depot_tools/recipes/recipe_modules/git/__init__.py +1 -4
- data/vendor/depot_tools/recipes/recipe_modules/git/api.py +7 -35
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/basic.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/basic_branch.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/basic_file_name.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/basic_hash.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/basic_ref.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/basic_submodule_update_force.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/can_fail_build.json +1 -1
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/cannot_fail_build.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/cat-file_test.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/count-objects_delta.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/count-objects_failed.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/count-objects_with_bad_output.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/count-objects_with_bad_output_fails_build.json +1 -1
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/curl_trace_file.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/git-cache-checkout.json +6 -6
- data/vendor/depot_tools/recipes/recipe_modules/git/examples/full.expected/platform_win.json +223 -0
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/rebase_failed.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/remote_not_origin.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.expected → examples/full.expected}/set_got_revision.json +3 -3
- data/vendor/depot_tools/recipes/recipe_modules/git/{example.py → examples/full.py} +2 -1
- data/vendor/depot_tools/recipes/recipe_modules/git/resources/git_setup.py +12 -21
- data/vendor/depot_tools/recipes/recipe_modules/git_cl/__init__.py +1 -4
- data/vendor/depot_tools/recipes/recipe_modules/git_cl/api.py +13 -11
- data/vendor/depot_tools/recipes/recipe_modules/git_cl/{example.expected → examples/full.expected}/basic.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/git_cl/{example.py → examples/full.py} +4 -2
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/OWNERS +3 -0
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/__init__.py +7 -0
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/api.py +135 -0
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/examples/full.expected/basic.json +537 -0
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/examples/full.py +61 -0
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/resources/gerrit_client.py +192 -0
- data/vendor/depot_tools/recipes/recipe_modules/gitiles/test_api.py +95 -0
- data/vendor/depot_tools/recipes/recipe_modules/gsutil/{example.expected → examples/full.expected}/basic.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/gsutil/{example.py → examples/full.py} +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/__init__.py +0 -3
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.expected → examples/full.expected}/basic.json +3 -2
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.expected → examples/full.expected}/paths_buildbot_linux.json +3 -2
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.expected → examples/full.expected}/paths_buildbot_mac.json +3 -2
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.expected → examples/full.expected}/paths_buildbot_win.json +3 -2
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.expected → examples/full.expected}/paths_generic_linux.json +3 -2
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.expected → examples/full.expected}/paths_generic_mac.json +3 -2
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/examples/full.expected/paths_generic_win.json +16 -0
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/examples/full.expected/paths_kitchen_linux.json +16 -0
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/examples/full.expected/paths_kitchen_mac.json +16 -0
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/examples/full.expected/paths_kitchen_win.json +16 -0
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/{example.py → examples/full.py} +2 -1
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/path_config.py +12 -3
- data/vendor/depot_tools/recipes/recipe_modules/presubmit/__init__.py +1 -0
- data/vendor/depot_tools/recipes/recipe_modules/presubmit/api.py +2 -2
- data/vendor/depot_tools/recipes/recipe_modules/presubmit/{example.expected → examples/full.expected}/basic.json +1 -1
- data/vendor/depot_tools/recipes/recipe_modules/presubmit/{example.py → examples/full.py} +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/rietveld/__init__.py +0 -4
- data/vendor/depot_tools/recipes/recipe_modules/rietveld/{example.expected → examples/full.expected}/basic.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/rietveld/{example.expected → examples/full.expected}/buildbot.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/rietveld/examples/full.expected/no_auth.json +27 -0
- data/vendor/depot_tools/recipes/recipe_modules/rietveld/{example.py → examples/full.py} +9 -1
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/__init__.py +3 -5
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/api.py +21 -96
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected → examples/full.expected}/basic_tags.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected → examples/full.expected}/set_failure_hash_with_no_steps.json +0 -0
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/examples/full.expected/with_gerrit_patch.json +56 -0
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected/with_gerrit_patch_deprecated.json → examples/full.expected/with_git_patch.json} +2 -2
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected/with_gerrit_patch.json → examples/full.expected/with_git_patch_luci.json} +2 -2
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected → examples/full.expected}/with_rietveld_patch.json +0 -20
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected → examples/full.expected}/with_rietveld_patch_new.json +0 -20
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected → examples/full.expected}/with_wrong_patch.json +1 -13
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.expected → examples/full.expected}/with_wrong_patch_new.json +1 -13
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/{example.py → examples/full.py} +3 -17
- data/vendor/depot_tools/recipes/recipes.py +141 -96
- data/vendor/depot_tools/rietveld.py +8 -1
- data/vendor/depot_tools/roll-dep-svn.bat +12 -10
- data/vendor/depot_tools/roll-dep.bat +5 -3
- data/vendor/depot_tools/scm.py +8 -1
- data/vendor/depot_tools/setup_color.py +0 -0
- data/vendor/depot_tools/split_cl.py +193 -0
- data/vendor/depot_tools/third_party/schema/.editorconfig +15 -0
- data/vendor/depot_tools/third_party/schema/.gitignore +174 -0
- data/vendor/depot_tools/third_party/schema/.travis.yml +37 -0
- data/vendor/depot_tools/third_party/schema/LICENSE-MIT +19 -0
- data/vendor/depot_tools/third_party/schema/MANIFEST.in +1 -0
- data/vendor/depot_tools/third_party/schema/README.chromium +12 -0
- data/vendor/depot_tools/third_party/schema/README.rst +382 -0
- data/vendor/depot_tools/third_party/schema/__init__.py +1 -0
- data/vendor/depot_tools/third_party/schema/schema.py +338 -0
- data/vendor/depot_tools/third_party/schema/setup.cfg +5 -0
- data/vendor/depot_tools/third_party/schema/setup.py +30 -0
- data/vendor/depot_tools/third_party/schema/test_schema.py +556 -0
- data/vendor/depot_tools/third_party/schema/tox.ini +33 -0
- data/vendor/depot_tools/third_party/upload.py +4 -0
- data/vendor/depot_tools/update_depot_tools +4 -16
- data/vendor/depot_tools/update_depot_tools.bat +4 -17
- data/vendor/depot_tools/update_depot_tools_toggle.py +38 -0
- data/vendor/depot_tools/vpython +12 -0
- data/vendor/depot_tools/vpython.bat +7 -0
- data/vendor/depot_tools/win_toolchain/get_toolchain_if_necessary.py +17 -5
- data/vendor/depot_tools/win_toolchain/package_from_installed.py +63 -33
- metadata +161 -113
- data/vendor/depot_tools/bootstrap/gclient.bat +0 -22
- data/vendor/depot_tools/bootstrap/win/get_file.js +0 -119
- data/vendor/depot_tools/bootstrap/win/git_bootstrap.py +0 -193
- data/vendor/depot_tools/bootstrap/win/git_version.txt +0 -1
- data/vendor/depot_tools/bootstrap/win/git_version_bleeding_edge.txt +0 -1
- data/vendor/depot_tools/bootstrap/win/python276.new.bat +0 -8
- data/vendor/depot_tools/bootstrap/win/svn.new.bat +0 -4
- data/vendor/depot_tools/bootstrap/win/svnversion.new.bat +0 -4
- data/vendor/depot_tools/bootstrap/win/unzip.js +0 -91
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/basic.json +0 -53
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/basic_output_manifest.json +0 -60
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/basic_with_branch_heads.json +0 -54
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/buildbot.json +0 -53
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/clobber.json +0 -54
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/gerrit_no_rebase_patch_ref.json +0 -54
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/gerrit_no_reset.json +0 -54
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/no_shallow.json +0 -54
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/reset_root_solution_revision.json +0 -53
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/trychange.json +0 -55
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/trychange_oauth2_buildbot.json +0 -57
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/trychange_oauth2_json.json +0 -55
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/trychange_oauth2_json_win.json +0 -55
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob.json +0 -59
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_fail.json +0 -62
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_fail_patch.json +0 -83
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_fail_patch_download.json +0 -84
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_gerrit_angle.json +0 -60
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_gerrit_angle_deprecated.json +0 -60
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_v8.json +0 -62
- data/vendor/depot_tools/recipes/recipe_modules/bot_update/example.expected/tryjob_v8_head_by_default.json +0 -62
- data/vendor/depot_tools/recipes/recipe_modules/gclient/bundle_extra_paths.txt +0 -4
- data/vendor/depot_tools/recipes/recipe_modules/gerrit/example.expected/basic.json +0 -66
- data/vendor/depot_tools/recipes/recipe_modules/git/bundle_extra_paths.txt +0 -28
- data/vendor/depot_tools/recipes/recipe_modules/git/example.expected/platform_win.json +0 -237
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/example.expected/paths_generic_win.json +0 -15
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/example.expected/paths_kitchen_linux.json +0 -15
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/example.expected/paths_kitchen_mac.json +0 -15
- data/vendor/depot_tools/recipes/recipe_modules/infra_paths/example.expected/paths_kitchen_win.json +0 -15
- data/vendor/depot_tools/recipes/recipe_modules/presubmit/bundle_extra_paths.txt +0 -30
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/example.expected/with_git_patch.json +0 -109
- data/vendor/depot_tools/recipes/recipe_modules/tryserver/example.expected/with_git_patch_luci.json +0 -8
@@ -0,0 +1 @@
|
|
1
|
+
from .schema import *
|
@@ -0,0 +1,338 @@
|
|
1
|
+
"""schema is a library for validating Python data structures, such as those
|
2
|
+
obtained from config-files, forms, external services or command-line
|
3
|
+
parsing, converted from JSON/YAML (or something else) to Python data-types."""
|
4
|
+
|
5
|
+
import re
|
6
|
+
|
7
|
+
__version__ = '0.6.6'
|
8
|
+
__all__ = ['Schema',
|
9
|
+
'And', 'Or', 'Regex', 'Optional', 'Use',
|
10
|
+
'SchemaError',
|
11
|
+
'SchemaWrongKeyError',
|
12
|
+
'SchemaMissingKeyError',
|
13
|
+
'SchemaUnexpectedTypeError']
|
14
|
+
|
15
|
+
|
16
|
+
class SchemaError(Exception):
|
17
|
+
"""Error during Schema validation."""
|
18
|
+
|
19
|
+
def __init__(self, autos, errors=None):
|
20
|
+
self.autos = autos if type(autos) is list else [autos]
|
21
|
+
self.errors = errors if type(errors) is list else [errors]
|
22
|
+
Exception.__init__(self, self.code)
|
23
|
+
|
24
|
+
@property
|
25
|
+
def code(self):
|
26
|
+
"""
|
27
|
+
Removes duplicates values in auto and error list.
|
28
|
+
parameters.
|
29
|
+
"""
|
30
|
+
def uniq(seq):
|
31
|
+
"""
|
32
|
+
Utility function that removes duplicate.
|
33
|
+
"""
|
34
|
+
seen = set()
|
35
|
+
seen_add = seen.add
|
36
|
+
# This way removes duplicates while preserving the order.
|
37
|
+
return [x for x in seq if x not in seen and not seen_add(x)]
|
38
|
+
data_set = uniq(i for i in self.autos if i is not None)
|
39
|
+
error_list = uniq(i for i in self.errors if i is not None)
|
40
|
+
if error_list:
|
41
|
+
return '\n'.join(error_list)
|
42
|
+
return '\n'.join(data_set)
|
43
|
+
|
44
|
+
|
45
|
+
class SchemaWrongKeyError(SchemaError):
|
46
|
+
"""Error Should be raised when an unexpected key is detected within the
|
47
|
+
data set being."""
|
48
|
+
pass
|
49
|
+
|
50
|
+
|
51
|
+
class SchemaMissingKeyError(SchemaError):
|
52
|
+
"""Error should be raised when a mandatory key is not found within the
|
53
|
+
data set being vaidated"""
|
54
|
+
pass
|
55
|
+
|
56
|
+
|
57
|
+
class SchemaUnexpectedTypeError(SchemaError):
|
58
|
+
"""Error should be raised when a type mismatch is detected within the
|
59
|
+
data set being validated."""
|
60
|
+
pass
|
61
|
+
|
62
|
+
|
63
|
+
class And(object):
|
64
|
+
"""
|
65
|
+
Utility function to combine validation directives in AND Boolean fashion.
|
66
|
+
"""
|
67
|
+
def __init__(self, *args, **kw):
|
68
|
+
self._args = args
|
69
|
+
assert set(kw).issubset(['error', 'schema', 'ignore_extra_keys'])
|
70
|
+
self._error = kw.get('error')
|
71
|
+
self._ignore_extra_keys = kw.get('ignore_extra_keys', False)
|
72
|
+
# You can pass your inherited Schema class.
|
73
|
+
self._schema = kw.get('schema', Schema)
|
74
|
+
|
75
|
+
def __repr__(self):
|
76
|
+
return '%s(%s)' % (self.__class__.__name__,
|
77
|
+
', '.join(repr(a) for a in self._args))
|
78
|
+
|
79
|
+
def validate(self, data):
|
80
|
+
"""
|
81
|
+
Validate data using defined sub schema/expressions ensuring all
|
82
|
+
values are valid.
|
83
|
+
:param data: to be validated with sub defined schemas.
|
84
|
+
:return: returns validated data
|
85
|
+
"""
|
86
|
+
for s in [self._schema(s, error=self._error,
|
87
|
+
ignore_extra_keys=self._ignore_extra_keys)
|
88
|
+
for s in self._args]:
|
89
|
+
data = s.validate(data)
|
90
|
+
return data
|
91
|
+
|
92
|
+
|
93
|
+
class Or(And):
|
94
|
+
"""Utility function to combine validation directives in a OR Boolean
|
95
|
+
fashion."""
|
96
|
+
def validate(self, data):
|
97
|
+
"""
|
98
|
+
Validate data using sub defined schema/expressions ensuring at least
|
99
|
+
one value is valid.
|
100
|
+
:param data: data to be validated by provided schema.
|
101
|
+
:return: return validated data if not validation
|
102
|
+
"""
|
103
|
+
x = SchemaError([], [])
|
104
|
+
for s in [self._schema(s, error=self._error,
|
105
|
+
ignore_extra_keys=self._ignore_extra_keys)
|
106
|
+
for s in self._args]:
|
107
|
+
try:
|
108
|
+
return s.validate(data)
|
109
|
+
except SchemaError as _x:
|
110
|
+
x = _x
|
111
|
+
raise SchemaError(['%r did not validate %r' % (self, data)] + x.autos,
|
112
|
+
[self._error.format(data) if self._error else None] +
|
113
|
+
x.errors)
|
114
|
+
|
115
|
+
|
116
|
+
class Regex(object):
|
117
|
+
"""
|
118
|
+
Enables schema.py to validate string using regular expressions.
|
119
|
+
"""
|
120
|
+
# Map all flags bits to a more readable description
|
121
|
+
NAMES = ['re.ASCII', 're.DEBUG', 're.VERBOSE', 're.UNICODE', 're.DOTALL',
|
122
|
+
're.MULTILINE', 're.LOCALE', 're.IGNORECASE', 're.TEMPLATE']
|
123
|
+
|
124
|
+
def __init__(self, pattern_str, flags=0, error=None):
|
125
|
+
self._pattern_str = pattern_str
|
126
|
+
flags_list = [Regex.NAMES[i] for i, f in # Name for each bit
|
127
|
+
enumerate('{0:09b}'.format(flags)) if f != '0']
|
128
|
+
|
129
|
+
if flags_list:
|
130
|
+
self._flags_names = ', flags=' + '|'.join(flags_list)
|
131
|
+
else:
|
132
|
+
self._flags_names = ''
|
133
|
+
|
134
|
+
self._pattern = re.compile(pattern_str, flags=flags)
|
135
|
+
self._error = error
|
136
|
+
|
137
|
+
def __repr__(self):
|
138
|
+
return '%s(%r%s)' % (
|
139
|
+
self.__class__.__name__, self._pattern_str, self._flags_names
|
140
|
+
)
|
141
|
+
|
142
|
+
def validate(self, data):
|
143
|
+
"""
|
144
|
+
Validated data using defined regex.
|
145
|
+
:param data: data to be validated
|
146
|
+
:return: return validated data.
|
147
|
+
"""
|
148
|
+
e = self._error
|
149
|
+
|
150
|
+
try:
|
151
|
+
if self._pattern.search(data):
|
152
|
+
return data
|
153
|
+
else:
|
154
|
+
raise SchemaError('%r does not match %r' % (self, data), e)
|
155
|
+
except TypeError:
|
156
|
+
raise SchemaError('%r is not string nor buffer' % data, e)
|
157
|
+
|
158
|
+
|
159
|
+
class Use(object):
|
160
|
+
"""
|
161
|
+
For more general use cases, you can use the Use class to transform
|
162
|
+
the data while it is being validate.
|
163
|
+
"""
|
164
|
+
def __init__(self, callable_, error=None):
|
165
|
+
assert callable(callable_)
|
166
|
+
self._callable = callable_
|
167
|
+
self._error = error
|
168
|
+
|
169
|
+
def __repr__(self):
|
170
|
+
return '%s(%r)' % (self.__class__.__name__, self._callable)
|
171
|
+
|
172
|
+
def validate(self, data):
|
173
|
+
try:
|
174
|
+
return self._callable(data)
|
175
|
+
except SchemaError as x:
|
176
|
+
raise SchemaError([None] + x.autos,
|
177
|
+
[self._error.format(data)
|
178
|
+
if self._error else None] + x.errors)
|
179
|
+
except BaseException as x:
|
180
|
+
f = _callable_str(self._callable)
|
181
|
+
raise SchemaError('%s(%r) raised %r' % (f, data, x),
|
182
|
+
self._error.format(data)
|
183
|
+
if self._error else None)
|
184
|
+
|
185
|
+
|
186
|
+
COMPARABLE, CALLABLE, VALIDATOR, TYPE, DICT, ITERABLE = range(6)
|
187
|
+
|
188
|
+
|
189
|
+
def _priority(s):
|
190
|
+
"""Return priority for a given object."""
|
191
|
+
if type(s) in (list, tuple, set, frozenset):
|
192
|
+
return ITERABLE
|
193
|
+
if type(s) is dict:
|
194
|
+
return DICT
|
195
|
+
if issubclass(type(s), type):
|
196
|
+
return TYPE
|
197
|
+
if hasattr(s, 'validate'):
|
198
|
+
return VALIDATOR
|
199
|
+
if callable(s):
|
200
|
+
return CALLABLE
|
201
|
+
else:
|
202
|
+
return COMPARABLE
|
203
|
+
|
204
|
+
|
205
|
+
class Schema(object):
|
206
|
+
"""
|
207
|
+
Entry point of the library, use this class to instantiate validation
|
208
|
+
schema for the data that will be validated.
|
209
|
+
"""
|
210
|
+
def __init__(self, schema, error=None, ignore_extra_keys=False):
|
211
|
+
self._schema = schema
|
212
|
+
self._error = error
|
213
|
+
self._ignore_extra_keys = ignore_extra_keys
|
214
|
+
|
215
|
+
def __repr__(self):
|
216
|
+
return '%s(%r)' % (self.__class__.__name__, self._schema)
|
217
|
+
|
218
|
+
@staticmethod
|
219
|
+
def _dict_key_priority(s):
|
220
|
+
"""Return priority for a given key object."""
|
221
|
+
if isinstance(s, Optional):
|
222
|
+
return _priority(s._schema) + 0.5
|
223
|
+
return _priority(s)
|
224
|
+
|
225
|
+
def validate(self, data):
|
226
|
+
Schema = self.__class__
|
227
|
+
s = self._schema
|
228
|
+
e = self._error
|
229
|
+
i = self._ignore_extra_keys
|
230
|
+
flavor = _priority(s)
|
231
|
+
if flavor == ITERABLE:
|
232
|
+
data = Schema(type(s), error=e).validate(data)
|
233
|
+
o = Or(*s, error=e, schema=Schema, ignore_extra_keys=i)
|
234
|
+
return type(data)(o.validate(d) for d in data)
|
235
|
+
if flavor == DICT:
|
236
|
+
data = Schema(dict, error=e).validate(data)
|
237
|
+
new = type(data)() # new - is a dict of the validated values
|
238
|
+
coverage = set() # matched schema keys
|
239
|
+
# for each key and value find a schema entry matching them, if any
|
240
|
+
sorted_skeys = sorted(s, key=self._dict_key_priority)
|
241
|
+
for key, value in data.items():
|
242
|
+
for skey in sorted_skeys:
|
243
|
+
svalue = s[skey]
|
244
|
+
try:
|
245
|
+
nkey = Schema(skey, error=e).validate(key)
|
246
|
+
except SchemaError:
|
247
|
+
pass
|
248
|
+
else:
|
249
|
+
try:
|
250
|
+
nvalue = Schema(svalue, error=e,
|
251
|
+
ignore_extra_keys=i).validate(value)
|
252
|
+
except SchemaError as x:
|
253
|
+
k = "Key '%s' error:" % nkey
|
254
|
+
raise SchemaError([k] + x.autos, [e] + x.errors)
|
255
|
+
else:
|
256
|
+
new[nkey] = nvalue
|
257
|
+
coverage.add(skey)
|
258
|
+
break
|
259
|
+
required = set(k for k in s if type(k) is not Optional)
|
260
|
+
if not required.issubset(coverage):
|
261
|
+
missing_keys = required - coverage
|
262
|
+
s_missing_keys = \
|
263
|
+
', '.join(repr(k) for k in sorted(missing_keys, key=repr))
|
264
|
+
raise \
|
265
|
+
SchemaMissingKeyError('Missing keys: ' + s_missing_keys, e)
|
266
|
+
if not self._ignore_extra_keys and (len(new) != len(data)):
|
267
|
+
wrong_keys = set(data.keys()) - set(new.keys())
|
268
|
+
s_wrong_keys = \
|
269
|
+
', '.join(repr(k) for k in sorted(wrong_keys, key=repr))
|
270
|
+
raise \
|
271
|
+
SchemaWrongKeyError(
|
272
|
+
'Wrong keys %s in %r' % (s_wrong_keys, data),
|
273
|
+
e.format(data) if e else None)
|
274
|
+
|
275
|
+
# Apply default-having optionals that haven't been used:
|
276
|
+
defaults = set(k for k in s if type(k) is Optional and
|
277
|
+
hasattr(k, 'default')) - coverage
|
278
|
+
for default in defaults:
|
279
|
+
new[default.key] = default.default
|
280
|
+
|
281
|
+
return new
|
282
|
+
if flavor == TYPE:
|
283
|
+
if isinstance(data, s):
|
284
|
+
return data
|
285
|
+
else:
|
286
|
+
raise SchemaUnexpectedTypeError(
|
287
|
+
'%r should be instance of %r' % (data, s.__name__),
|
288
|
+
e.format(data) if e else None)
|
289
|
+
if flavor == VALIDATOR:
|
290
|
+
try:
|
291
|
+
return s.validate(data)
|
292
|
+
except SchemaError as x:
|
293
|
+
raise SchemaError([None] + x.autos, [e] + x.errors)
|
294
|
+
except BaseException as x:
|
295
|
+
raise SchemaError(
|
296
|
+
'%r.validate(%r) raised %r' % (s, data, x),
|
297
|
+
self._error.format(data) if self._error else None)
|
298
|
+
if flavor == CALLABLE:
|
299
|
+
f = _callable_str(s)
|
300
|
+
try:
|
301
|
+
if s(data):
|
302
|
+
return data
|
303
|
+
except SchemaError as x:
|
304
|
+
raise SchemaError([None] + x.autos, [e] + x.errors)
|
305
|
+
except BaseException as x:
|
306
|
+
raise SchemaError(
|
307
|
+
'%s(%r) raised %r' % (f, data, x),
|
308
|
+
self._error.format(data) if self._error else None)
|
309
|
+
raise SchemaError('%s(%r) should evaluate to True' % (f, data), e)
|
310
|
+
if s == data:
|
311
|
+
return data
|
312
|
+
else:
|
313
|
+
raise SchemaError('%r does not match %r' % (s, data),
|
314
|
+
e.format(data) if e else None)
|
315
|
+
|
316
|
+
|
317
|
+
class Optional(Schema):
|
318
|
+
"""Marker for an optional part of the validation Schema."""
|
319
|
+
_MARKER = object()
|
320
|
+
|
321
|
+
def __init__(self, *args, **kwargs):
|
322
|
+
default = kwargs.pop('default', self._MARKER)
|
323
|
+
super(Optional, self).__init__(*args, **kwargs)
|
324
|
+
if default is not self._MARKER:
|
325
|
+
# See if I can come up with a static key to use for myself:
|
326
|
+
if _priority(self._schema) != COMPARABLE:
|
327
|
+
raise TypeError(
|
328
|
+
'Optional keys with defaults must have simple, '
|
329
|
+
'predictable values, like literal strings or ints. '
|
330
|
+
'"%r" is too complex.' % (self._schema,))
|
331
|
+
self.default = default
|
332
|
+
self.key = self._schema
|
333
|
+
|
334
|
+
|
335
|
+
def _callable_str(callable_):
|
336
|
+
if hasattr(callable_, '__name__'):
|
337
|
+
return callable_.__name__
|
338
|
+
return str(callable_)
|
@@ -0,0 +1,30 @@
|
|
1
|
+
from setuptools import setup
|
2
|
+
|
3
|
+
import codecs
|
4
|
+
import schema
|
5
|
+
|
6
|
+
|
7
|
+
setup(
|
8
|
+
name=schema.__name__,
|
9
|
+
version=schema.__version__,
|
10
|
+
author="Vladimir Keleshev",
|
11
|
+
author_email="vladimir@keleshev.com",
|
12
|
+
description="Simple data validation library",
|
13
|
+
license="MIT",
|
14
|
+
keywords="schema json validation",
|
15
|
+
url="https://github.com/keleshev/schema",
|
16
|
+
py_modules=['schema'],
|
17
|
+
long_description=codecs.open('README.rst', 'r', 'utf-8').read(),
|
18
|
+
classifiers=[
|
19
|
+
"Development Status :: 3 - Alpha",
|
20
|
+
"Topic :: Utilities",
|
21
|
+
"Programming Language :: Python :: 2.6",
|
22
|
+
"Programming Language :: Python :: 2.7",
|
23
|
+
"Programming Language :: Python :: 3.2",
|
24
|
+
"Programming Language :: Python :: 3.3",
|
25
|
+
"Programming Language :: Python :: 3.4",
|
26
|
+
"Programming Language :: Python :: 3.5",
|
27
|
+
"Programming Language :: Python :: Implementation :: PyPy",
|
28
|
+
"License :: OSI Approved :: MIT License",
|
29
|
+
],
|
30
|
+
)
|
@@ -0,0 +1,556 @@
|
|
1
|
+
from __future__ import with_statement
|
2
|
+
from collections import defaultdict, namedtuple
|
3
|
+
from operator import methodcaller
|
4
|
+
import os
|
5
|
+
import re
|
6
|
+
import sys
|
7
|
+
import copy
|
8
|
+
|
9
|
+
from pytest import raises
|
10
|
+
|
11
|
+
from schema import (Schema, Use, And, Or, Regex, Optional,
|
12
|
+
SchemaError, SchemaWrongKeyError,
|
13
|
+
SchemaMissingKeyError, SchemaUnexpectedTypeError)
|
14
|
+
|
15
|
+
if sys.version_info[0] == 3:
|
16
|
+
basestring = str # Python 3 does not have basestring
|
17
|
+
unicode = str # Python 3 does not have unicode
|
18
|
+
|
19
|
+
|
20
|
+
SE = raises(SchemaError)
|
21
|
+
|
22
|
+
|
23
|
+
def ve(_):
|
24
|
+
raise ValueError()
|
25
|
+
|
26
|
+
|
27
|
+
def se(_):
|
28
|
+
raise SchemaError('first auto', 'first error')
|
29
|
+
|
30
|
+
|
31
|
+
def test_schema():
|
32
|
+
|
33
|
+
assert Schema(1).validate(1) == 1
|
34
|
+
with SE: Schema(1).validate(9)
|
35
|
+
|
36
|
+
assert Schema(int).validate(1) == 1
|
37
|
+
with SE: Schema(int).validate('1')
|
38
|
+
assert Schema(Use(int)).validate('1') == 1
|
39
|
+
with SE: Schema(int).validate(int)
|
40
|
+
|
41
|
+
assert Schema(str).validate('hai') == 'hai'
|
42
|
+
with SE: Schema(str).validate(1)
|
43
|
+
assert Schema(Use(str)).validate(1) == '1'
|
44
|
+
|
45
|
+
assert Schema(list).validate(['a', 1]) == ['a', 1]
|
46
|
+
assert Schema(dict).validate({'a': 1}) == {'a': 1}
|
47
|
+
with SE: Schema(dict).validate(['a', 1])
|
48
|
+
|
49
|
+
assert Schema(lambda n: 0 < n < 5).validate(3) == 3
|
50
|
+
with SE: Schema(lambda n: 0 < n < 5).validate(-1)
|
51
|
+
|
52
|
+
|
53
|
+
def test_validate_file():
|
54
|
+
assert Schema(
|
55
|
+
Use(open)).validate('LICENSE-MIT').read().startswith('Copyright')
|
56
|
+
with SE: Schema(Use(open)).validate('NON-EXISTENT')
|
57
|
+
assert Schema(os.path.exists).validate('.') == '.'
|
58
|
+
with SE: Schema(os.path.exists).validate('./non-existent/')
|
59
|
+
assert Schema(os.path.isfile).validate('LICENSE-MIT') == 'LICENSE-MIT'
|
60
|
+
with SE: Schema(os.path.isfile).validate('NON-EXISTENT')
|
61
|
+
|
62
|
+
|
63
|
+
def test_and():
|
64
|
+
assert And(int, lambda n: 0 < n < 5).validate(3) == 3
|
65
|
+
with SE: And(int, lambda n: 0 < n < 5).validate(3.33)
|
66
|
+
assert And(Use(int), lambda n: 0 < n < 5).validate(3.33) == 3
|
67
|
+
with SE: And(Use(int), lambda n: 0 < n < 5).validate('3.33')
|
68
|
+
|
69
|
+
|
70
|
+
def test_or():
|
71
|
+
assert Or(int, dict).validate(5) == 5
|
72
|
+
assert Or(int, dict).validate({}) == {}
|
73
|
+
with SE: Or(int, dict).validate('hai')
|
74
|
+
assert Or(int).validate(4)
|
75
|
+
with SE: Or().validate(2)
|
76
|
+
|
77
|
+
|
78
|
+
def test_regex():
|
79
|
+
# Simple case: validate string
|
80
|
+
assert Regex(r'foo').validate('afoot') == 'afoot'
|
81
|
+
with SE: Regex(r'bar').validate('afoot')
|
82
|
+
|
83
|
+
# More complex case: validate string
|
84
|
+
assert Regex(r'^[a-z]+$').validate('letters') == 'letters'
|
85
|
+
with SE:
|
86
|
+
Regex(r'^[a-z]+$').validate('letters + spaces') == 'letters + spaces'
|
87
|
+
|
88
|
+
# Validate dict key
|
89
|
+
assert (Schema({Regex(r'^foo'): str})
|
90
|
+
.validate({'fookey': 'value'}) == {'fookey': 'value'})
|
91
|
+
with SE: Schema({Regex(r'^foo'): str}).validate({'barkey': 'value'})
|
92
|
+
|
93
|
+
# Validate dict value
|
94
|
+
assert (Schema({str: Regex(r'^foo')}).validate({'key': 'foovalue'}) ==
|
95
|
+
{'key': 'foovalue'})
|
96
|
+
with SE: Schema({str: Regex(r'^foo')}).validate({'key': 'barvalue'})
|
97
|
+
|
98
|
+
# Error if the value does not have a buffer interface
|
99
|
+
with SE: Regex(r'bar').validate(1)
|
100
|
+
with SE: Regex(r'bar').validate({})
|
101
|
+
with SE: Regex(r'bar').validate([])
|
102
|
+
with SE: Regex(r'bar').validate(None)
|
103
|
+
|
104
|
+
# Validate that the pattern has a buffer interface
|
105
|
+
assert Regex(re.compile(r'foo')).validate('foo') == 'foo'
|
106
|
+
assert Regex(unicode('foo')).validate('foo') == 'foo'
|
107
|
+
with raises(TypeError): Regex(1).validate('bar')
|
108
|
+
with raises(TypeError): Regex({}).validate('bar')
|
109
|
+
with raises(TypeError): Regex([]).validate('bar')
|
110
|
+
with raises(TypeError): Regex(None).validate('bar')
|
111
|
+
|
112
|
+
|
113
|
+
def test_validate_list():
|
114
|
+
assert Schema([1, 0]).validate([1, 0, 1, 1]) == [1, 0, 1, 1]
|
115
|
+
assert Schema([1, 0]).validate([]) == []
|
116
|
+
with SE: Schema([1, 0]).validate(0)
|
117
|
+
with SE: Schema([1, 0]).validate([2])
|
118
|
+
assert And([1, 0], lambda l: len(l) > 2).validate([0, 1, 0]) == [0, 1, 0]
|
119
|
+
with SE: And([1, 0], lambda l: len(l) > 2).validate([0, 1])
|
120
|
+
|
121
|
+
|
122
|
+
def test_list_tuple_set_frozenset():
|
123
|
+
assert Schema([int]).validate([1, 2])
|
124
|
+
with SE: Schema([int]).validate(['1', 2])
|
125
|
+
assert Schema(set([int])).validate(set([1, 2])) == set([1, 2])
|
126
|
+
with SE: Schema(set([int])).validate([1, 2]) # not a set
|
127
|
+
with SE: Schema(set([int])).validate(['1', 2])
|
128
|
+
assert Schema(tuple([int])).validate(tuple([1, 2])) == tuple([1, 2])
|
129
|
+
with SE: Schema(tuple([int])).validate([1, 2]) # not a set
|
130
|
+
|
131
|
+
|
132
|
+
def test_strictly():
|
133
|
+
assert Schema(int).validate(1) == 1
|
134
|
+
with SE: Schema(int).validate('1')
|
135
|
+
|
136
|
+
|
137
|
+
def test_dict():
|
138
|
+
assert Schema({'key': 5}).validate({'key': 5}) == {'key': 5}
|
139
|
+
with SE: Schema({'key': 5}).validate({'key': 'x'})
|
140
|
+
with SE: Schema({'key': 5}).validate(['key', 5])
|
141
|
+
assert Schema({'key': int}).validate({'key': 5}) == {'key': 5}
|
142
|
+
assert Schema({'n': int, 'f': float}).validate(
|
143
|
+
{'n': 5, 'f': 3.14}) == {'n': 5, 'f': 3.14}
|
144
|
+
with SE: Schema({'n': int, 'f': float}).validate(
|
145
|
+
{'n': 3.14, 'f': 5})
|
146
|
+
with SE:
|
147
|
+
try:
|
148
|
+
Schema({}).validate({'abc': None, 1: None})
|
149
|
+
except SchemaWrongKeyError as e:
|
150
|
+
assert e.args[0].startswith("Wrong keys 'abc', 1 in")
|
151
|
+
raise
|
152
|
+
with SE:
|
153
|
+
try:
|
154
|
+
Schema({'key': 5}).validate({})
|
155
|
+
except SchemaMissingKeyError as e:
|
156
|
+
assert e.args[0] == "Missing keys: 'key'"
|
157
|
+
raise
|
158
|
+
with SE:
|
159
|
+
try:
|
160
|
+
Schema({'key': 5}).validate({'n': 5})
|
161
|
+
except SchemaMissingKeyError as e:
|
162
|
+
assert e.args[0] == "Missing keys: 'key'"
|
163
|
+
raise
|
164
|
+
with SE:
|
165
|
+
try:
|
166
|
+
Schema({}).validate({'n': 5})
|
167
|
+
except SchemaWrongKeyError as e:
|
168
|
+
assert e.args[0] == "Wrong keys 'n' in {'n': 5}"
|
169
|
+
raise
|
170
|
+
with SE:
|
171
|
+
try:
|
172
|
+
Schema({'key': 5}).validate({'key': 5, 'bad': 5})
|
173
|
+
except SchemaWrongKeyError as e:
|
174
|
+
assert e.args[0] in ["Wrong keys 'bad' in {'key': 5, 'bad': 5}",
|
175
|
+
"Wrong keys 'bad' in {'bad': 5, 'key': 5}"]
|
176
|
+
raise
|
177
|
+
with SE:
|
178
|
+
try:
|
179
|
+
Schema({}).validate({'a': 5, 'b': 5})
|
180
|
+
except SchemaError as e:
|
181
|
+
assert e.args[0] in ["Wrong keys 'a', 'b' in {'a': 5, 'b': 5}",
|
182
|
+
"Wrong keys 'a', 'b' in {'b': 5, 'a': 5}"]
|
183
|
+
raise
|
184
|
+
|
185
|
+
with SE:
|
186
|
+
try:
|
187
|
+
Schema({int: int}).validate({'': ''})
|
188
|
+
except SchemaUnexpectedTypeError as e:
|
189
|
+
assert e.args[0] in ["'' should be instance of 'int'"]
|
190
|
+
|
191
|
+
|
192
|
+
def test_dict_keys():
|
193
|
+
assert Schema({str: int}).validate(
|
194
|
+
{'a': 1, 'b': 2}) == {'a': 1, 'b': 2}
|
195
|
+
with SE: Schema({str: int}).validate({1: 1, 'b': 2})
|
196
|
+
assert Schema({Use(str): Use(int)}).validate(
|
197
|
+
{1: 3.14, 3.14: 1}) == {'1': 3, '3.14': 1}
|
198
|
+
|
199
|
+
|
200
|
+
def test_ignore_extra_keys():
|
201
|
+
assert Schema({'key': 5}, ignore_extra_keys=True).validate(
|
202
|
+
{'key': 5, 'bad': 4}) == {'key': 5}
|
203
|
+
assert Schema({'key': 5, 'dk': {'a': 'a'}}, ignore_extra_keys=True).validate(
|
204
|
+
{'key': 5, 'bad': 'b', 'dk': {'a': 'a', 'bad': 'b'}}) == \
|
205
|
+
{'key': 5, 'dk': {'a': 'a'}}
|
206
|
+
assert Schema([{'key': 'v'}], ignore_extra_keys=True).validate(
|
207
|
+
[{'key': 'v', 'bad': 'bad'}]) == [{'key': 'v'}]
|
208
|
+
assert Schema([{'key': 'v'}], ignore_extra_keys=True).validate(
|
209
|
+
[{'key': 'v', 'bad': 'bad'}]) == [{'key': 'v'}]
|
210
|
+
|
211
|
+
|
212
|
+
def test_ignore_extra_keys_validation_and_return_keys():
|
213
|
+
assert Schema({'key': 5, object: object}, ignore_extra_keys=True).validate(
|
214
|
+
{'key': 5, 'bad': 4}) == {'key': 5, 'bad': 4}
|
215
|
+
assert Schema({'key': 5, 'dk': {'a': 'a', object: object}},
|
216
|
+
ignore_extra_keys=True).validate(
|
217
|
+
{'key': 5, 'dk': {'a': 'a', 'bad': 'b'}}) == \
|
218
|
+
{'key': 5, 'dk': {'a': 'a', 'bad': 'b'}}
|
219
|
+
|
220
|
+
|
221
|
+
def test_dict_optional_keys():
|
222
|
+
with SE: Schema({'a': 1, 'b': 2}).validate({'a': 1})
|
223
|
+
assert Schema({'a': 1, Optional('b'): 2}).validate({'a': 1}) == {'a': 1}
|
224
|
+
assert Schema({'a': 1, Optional('b'): 2}).validate(
|
225
|
+
{'a': 1, 'b': 2}) == {'a': 1, 'b': 2}
|
226
|
+
# Make sure Optionals are favored over types:
|
227
|
+
assert Schema({basestring: 1,
|
228
|
+
Optional('b'): 2}).validate({'a': 1, 'b': 2}) == {'a': 1, 'b': 2}
|
229
|
+
|
230
|
+
|
231
|
+
def test_dict_optional_defaults():
|
232
|
+
# Optionals fill out their defaults:
|
233
|
+
assert Schema({Optional('a', default=1): 11,
|
234
|
+
Optional('b', default=2): 22}).validate({'a': 11}) == {'a': 11, 'b': 2}
|
235
|
+
|
236
|
+
# Optionals take precedence over types. Here, the "a" is served by the
|
237
|
+
# Optional:
|
238
|
+
assert Schema({Optional('a', default=1): 11,
|
239
|
+
basestring: 22}).validate({'b': 22}) == {'a': 1, 'b': 22}
|
240
|
+
|
241
|
+
with raises(TypeError):
|
242
|
+
Optional(And(str, Use(int)), default=7)
|
243
|
+
|
244
|
+
|
245
|
+
def test_dict_subtypes():
|
246
|
+
d = defaultdict(int, key=1)
|
247
|
+
v = Schema({'key': 1}).validate(d)
|
248
|
+
assert v == d
|
249
|
+
assert isinstance(v, defaultdict)
|
250
|
+
# Please add tests for Counter and OrderedDict once support for Python2.6
|
251
|
+
# is dropped!
|
252
|
+
|
253
|
+
|
254
|
+
def test_dict_key_error():
|
255
|
+
try:
|
256
|
+
Schema({'k': int}).validate({'k': 'x'})
|
257
|
+
except SchemaError as e:
|
258
|
+
assert e.code == "Key 'k' error:\n'x' should be instance of 'int'"
|
259
|
+
try:
|
260
|
+
Schema({'k': {'k2': int}}).validate({'k': {'k2': 'x'}})
|
261
|
+
except SchemaError as e:
|
262
|
+
code = "Key 'k' error:\nKey 'k2' error:\n'x' should be instance of 'int'"
|
263
|
+
assert e.code == code
|
264
|
+
try:
|
265
|
+
Schema({'k': {'k2': int}}, error='k2 should be int').validate({'k': {'k2': 'x'}})
|
266
|
+
except SchemaError as e:
|
267
|
+
assert e.code == 'k2 should be int'
|
268
|
+
|
269
|
+
|
270
|
+
def test_complex():
|
271
|
+
s = Schema({'<file>': And([Use(open)], lambda l: len(l)),
|
272
|
+
'<path>': os.path.exists,
|
273
|
+
Optional('--count'): And(int, lambda n: 0 <= n <= 5)})
|
274
|
+
data = s.validate({'<file>': ['./LICENSE-MIT'], '<path>': './'})
|
275
|
+
assert len(data) == 2
|
276
|
+
assert len(data['<file>']) == 1
|
277
|
+
assert data['<file>'][0].read().startswith('Copyright')
|
278
|
+
assert data['<path>'] == './'
|
279
|
+
|
280
|
+
|
281
|
+
def test_nice_errors():
|
282
|
+
try:
|
283
|
+
Schema(int, error='should be integer').validate('x')
|
284
|
+
except SchemaError as e:
|
285
|
+
assert e.errors == ['should be integer']
|
286
|
+
try:
|
287
|
+
Schema(Use(float), error='should be a number').validate('x')
|
288
|
+
except SchemaError as e:
|
289
|
+
assert e.code == 'should be a number'
|
290
|
+
try:
|
291
|
+
Schema({Optional('i'): Use(int, error='should be a number')}).validate({'i': 'x'})
|
292
|
+
except SchemaError as e:
|
293
|
+
assert e.code == 'should be a number'
|
294
|
+
|
295
|
+
|
296
|
+
def test_use_error_handling():
|
297
|
+
try:
|
298
|
+
Use(ve).validate('x')
|
299
|
+
except SchemaError as e:
|
300
|
+
assert e.autos == ["ve('x') raised ValueError()"]
|
301
|
+
assert e.errors == [None]
|
302
|
+
try:
|
303
|
+
Use(ve, error='should not raise').validate('x')
|
304
|
+
except SchemaError as e:
|
305
|
+
assert e.autos == ["ve('x') raised ValueError()"]
|
306
|
+
assert e.errors == ['should not raise']
|
307
|
+
try:
|
308
|
+
Use(se).validate('x')
|
309
|
+
except SchemaError as e:
|
310
|
+
assert e.autos == [None, 'first auto']
|
311
|
+
assert e.errors == [None, 'first error']
|
312
|
+
try:
|
313
|
+
Use(se, error='second error').validate('x')
|
314
|
+
except SchemaError as e:
|
315
|
+
assert e.autos == [None, 'first auto']
|
316
|
+
assert e.errors == ['second error', 'first error']
|
317
|
+
|
318
|
+
|
319
|
+
def test_or_error_handling():
|
320
|
+
try:
|
321
|
+
Or(ve).validate('x')
|
322
|
+
except SchemaError as e:
|
323
|
+
assert e.autos[0].startswith('Or(')
|
324
|
+
assert e.autos[0].endswith(") did not validate 'x'")
|
325
|
+
assert e.autos[1] == "ve('x') raised ValueError()"
|
326
|
+
assert len(e.autos) == 2
|
327
|
+
assert e.errors == [None, None]
|
328
|
+
try:
|
329
|
+
Or(ve, error='should not raise').validate('x')
|
330
|
+
except SchemaError as e:
|
331
|
+
assert e.autos[0].startswith('Or(')
|
332
|
+
assert e.autos[0].endswith(") did not validate 'x'")
|
333
|
+
assert e.autos[1] == "ve('x') raised ValueError()"
|
334
|
+
assert len(e.autos) == 2
|
335
|
+
assert e.errors == ['should not raise', 'should not raise']
|
336
|
+
try:
|
337
|
+
Or('o').validate('x')
|
338
|
+
except SchemaError as e:
|
339
|
+
assert e.autos == ["Or('o') did not validate 'x'",
|
340
|
+
"'o' does not match 'x'"]
|
341
|
+
assert e.errors == [None, None]
|
342
|
+
try:
|
343
|
+
Or('o', error='second error').validate('x')
|
344
|
+
except SchemaError as e:
|
345
|
+
assert e.autos == ["Or('o') did not validate 'x'",
|
346
|
+
"'o' does not match 'x'"]
|
347
|
+
assert e.errors == ['second error', 'second error']
|
348
|
+
|
349
|
+
|
350
|
+
def test_and_error_handling():
|
351
|
+
try:
|
352
|
+
And(ve).validate('x')
|
353
|
+
except SchemaError as e:
|
354
|
+
assert e.autos == ["ve('x') raised ValueError()"]
|
355
|
+
assert e.errors == [None]
|
356
|
+
try:
|
357
|
+
And(ve, error='should not raise').validate('x')
|
358
|
+
except SchemaError as e:
|
359
|
+
assert e.autos == ["ve('x') raised ValueError()"]
|
360
|
+
assert e.errors == ['should not raise']
|
361
|
+
try:
|
362
|
+
And(str, se).validate('x')
|
363
|
+
except SchemaError as e:
|
364
|
+
assert e.autos == [None, 'first auto']
|
365
|
+
assert e.errors == [None, 'first error']
|
366
|
+
try:
|
367
|
+
And(str, se, error='second error').validate('x')
|
368
|
+
except SchemaError as e:
|
369
|
+
assert e.autos == [None, 'first auto']
|
370
|
+
assert e.errors == ['second error', 'first error']
|
371
|
+
|
372
|
+
|
373
|
+
def test_schema_error_handling():
|
374
|
+
try:
|
375
|
+
Schema(Use(ve)).validate('x')
|
376
|
+
except SchemaError as e:
|
377
|
+
assert e.autos == [None, "ve('x') raised ValueError()"]
|
378
|
+
assert e.errors == [None, None]
|
379
|
+
try:
|
380
|
+
Schema(Use(ve), error='should not raise').validate('x')
|
381
|
+
except SchemaError as e:
|
382
|
+
assert e.autos == [None, "ve('x') raised ValueError()"]
|
383
|
+
assert e.errors == ['should not raise', None]
|
384
|
+
try:
|
385
|
+
Schema(Use(se)).validate('x')
|
386
|
+
except SchemaError as e:
|
387
|
+
assert e.autos == [None, None, 'first auto']
|
388
|
+
assert e.errors == [None, None, 'first error']
|
389
|
+
try:
|
390
|
+
Schema(Use(se), error='second error').validate('x')
|
391
|
+
except SchemaError as e:
|
392
|
+
assert e.autos == [None, None, 'first auto']
|
393
|
+
assert e.errors == ['second error', None, 'first error']
|
394
|
+
|
395
|
+
|
396
|
+
def test_use_json():
|
397
|
+
import json
|
398
|
+
gist_schema = Schema(And(Use(json.loads), # first convert from JSON
|
399
|
+
{Optional('description'): basestring,
|
400
|
+
'public': bool,
|
401
|
+
'files': {basestring: {'content': basestring}}}))
|
402
|
+
gist = '''{"description": "the description for this gist",
|
403
|
+
"public": true,
|
404
|
+
"files": {
|
405
|
+
"file1.txt": {"content": "String file contents"},
|
406
|
+
"other.txt": {"content": "Another file contents"}}}'''
|
407
|
+
assert gist_schema.validate(gist)
|
408
|
+
|
409
|
+
|
410
|
+
def test_error_reporting():
|
411
|
+
s = Schema({'<files>': [Use(open, error='<files> should be readable')],
|
412
|
+
'<path>': And(os.path.exists, error='<path> should exist'),
|
413
|
+
'--count': Or(None, And(Use(int), lambda n: 0 < n < 5),
|
414
|
+
error='--count should be integer 0 < n < 5')},
|
415
|
+
error='Error:')
|
416
|
+
s.validate({'<files>': [], '<path>': './', '--count': 3})
|
417
|
+
|
418
|
+
try:
|
419
|
+
s.validate({'<files>': [], '<path>': './', '--count': '10'})
|
420
|
+
except SchemaError as e:
|
421
|
+
assert e.code == 'Error:\n--count should be integer 0 < n < 5'
|
422
|
+
try:
|
423
|
+
s.validate({'<files>': [], '<path>': './hai', '--count': '2'})
|
424
|
+
except SchemaError as e:
|
425
|
+
assert e.code == 'Error:\n<path> should exist'
|
426
|
+
try:
|
427
|
+
s.validate({'<files>': ['hai'], '<path>': './', '--count': '2'})
|
428
|
+
except SchemaError as e:
|
429
|
+
assert e.code == 'Error:\n<files> should be readable'
|
430
|
+
|
431
|
+
|
432
|
+
def test_schema_repr(): # what about repr with `error`s?
|
433
|
+
schema = Schema([Or(None, And(str, Use(float)))])
|
434
|
+
repr_ = "Schema([Or(None, And(<type 'str'>, Use(<type 'float'>)))])"
|
435
|
+
# in Python 3 repr contains <class 'str'>, not <type 'str'>
|
436
|
+
assert repr(schema).replace('class', 'type') == repr_
|
437
|
+
|
438
|
+
|
439
|
+
def test_validate_object():
|
440
|
+
schema = Schema({object: str})
|
441
|
+
assert schema.validate({42: 'str'}) == {42: 'str'}
|
442
|
+
with SE: schema.validate({42: 777})
|
443
|
+
|
444
|
+
|
445
|
+
def test_issue_9_prioritized_key_comparison():
|
446
|
+
validate = Schema({'key': 42, object: 42}).validate
|
447
|
+
assert validate({'key': 42, 777: 42}) == {'key': 42, 777: 42}
|
448
|
+
|
449
|
+
|
450
|
+
def test_issue_9_prioritized_key_comparison_in_dicts():
|
451
|
+
# http://stackoverflow.com/questions/14588098/docopt-schema-validation
|
452
|
+
s = Schema({'ID': Use(int, error='ID should be an int'),
|
453
|
+
'FILE': Or(None, Use(open, error='FILE should be readable')),
|
454
|
+
Optional(str): object})
|
455
|
+
data = {'ID': 10, 'FILE': None, 'other': 'other', 'other2': 'other2'}
|
456
|
+
assert s.validate(data) == data
|
457
|
+
data = {'ID': 10, 'FILE': None}
|
458
|
+
assert s.validate(data) == data
|
459
|
+
|
460
|
+
|
461
|
+
def test_missing_keys_exception_with_non_str_dict_keys():
|
462
|
+
s = Schema({And(str, Use(str.lower), 'name'): And(str, len)})
|
463
|
+
with SE: s.validate(dict())
|
464
|
+
with SE:
|
465
|
+
try:
|
466
|
+
Schema({1: 'x'}).validate(dict())
|
467
|
+
except SchemaMissingKeyError as e:
|
468
|
+
assert e.args[0] == "Missing keys: 1"
|
469
|
+
raise
|
470
|
+
|
471
|
+
|
472
|
+
def test_issue_56_cant_rely_on_callables_to_have_name():
|
473
|
+
s = Schema(methodcaller('endswith', '.csv'))
|
474
|
+
assert s.validate('test.csv') == 'test.csv'
|
475
|
+
with SE:
|
476
|
+
try:
|
477
|
+
s.validate('test.py')
|
478
|
+
except SchemaError as e:
|
479
|
+
assert "operator.methodcaller" in e.args[0]
|
480
|
+
raise
|
481
|
+
|
482
|
+
|
483
|
+
def test_exception_handling_with_bad_validators():
|
484
|
+
BadValidator = namedtuple("BadValidator", ["validate"])
|
485
|
+
s = Schema(BadValidator("haha"))
|
486
|
+
with SE:
|
487
|
+
try:
|
488
|
+
s.validate("test")
|
489
|
+
except SchemaError as e:
|
490
|
+
assert "TypeError" in e.args[0]
|
491
|
+
raise
|
492
|
+
|
493
|
+
|
494
|
+
def test_issue_83_iterable_validation_return_type():
|
495
|
+
TestSetType = type("TestSetType", (set,), dict())
|
496
|
+
data = TestSetType(["test", "strings"])
|
497
|
+
s = Schema(set([str]))
|
498
|
+
assert isinstance(s.validate(data), TestSetType)
|
499
|
+
|
500
|
+
|
501
|
+
def test_optional_key_convert_failed_randomly_while_with_another_optional_object():
|
502
|
+
"""
|
503
|
+
In this test, created_at string "2015-10-10 00:00:00" is expected to be converted
|
504
|
+
to a datetime instance.
|
505
|
+
- it works when the schema is
|
506
|
+
|
507
|
+
s = Schema({
|
508
|
+
'created_at': _datetime_validator,
|
509
|
+
Optional(basestring): object,
|
510
|
+
})
|
511
|
+
|
512
|
+
- but when wrapping the key 'created_at' with Optional, it fails randomly
|
513
|
+
:return:
|
514
|
+
"""
|
515
|
+
import datetime
|
516
|
+
fmt = '%Y-%m-%d %H:%M:%S'
|
517
|
+
_datetime_validator = Or(None, Use(lambda i: datetime.datetime.strptime(i, fmt)))
|
518
|
+
# FIXME given tests enough
|
519
|
+
for i in range(1024):
|
520
|
+
s = Schema({
|
521
|
+
Optional('created_at'): _datetime_validator,
|
522
|
+
Optional('updated_at'): _datetime_validator,
|
523
|
+
Optional('birth'): _datetime_validator,
|
524
|
+
Optional(basestring): object,
|
525
|
+
})
|
526
|
+
data = {
|
527
|
+
'created_at': '2015-10-10 00:00:00'
|
528
|
+
}
|
529
|
+
validated_data = s.validate(data)
|
530
|
+
# is expected to be converted to a datetime instance, but fails randomly
|
531
|
+
# (most of the time)
|
532
|
+
assert isinstance(validated_data['created_at'], datetime.datetime)
|
533
|
+
# assert isinstance(validated_data['created_at'], basestring)
|
534
|
+
|
535
|
+
|
536
|
+
def test_copy():
|
537
|
+
s1 = SchemaError('a', None)
|
538
|
+
s2 = copy.deepcopy(s1)
|
539
|
+
assert s1 is not s2
|
540
|
+
assert type(s1) is type(s2)
|
541
|
+
|
542
|
+
|
543
|
+
def test_inheritance():
|
544
|
+
def convert(data):
|
545
|
+
if isinstance(data, int):
|
546
|
+
return data + 1
|
547
|
+
return data
|
548
|
+
|
549
|
+
class MySchema(Schema):
|
550
|
+
def validate(self, data):
|
551
|
+
return super(MySchema, self).validate(convert(data))
|
552
|
+
|
553
|
+
s = {'k': int, 'd': {'k': int, 'l': [{'l': [int]}]}}
|
554
|
+
v = {'k': 1, 'd': {'k': 2, 'l': [{'l': [3, 4, 5]}]}}
|
555
|
+
d = MySchema(s).validate(v)
|
556
|
+
assert d['k'] == 2 and d['d']['k'] == 3 and d['d']['l'][0]['l'] == [4, 5, 6]
|