puppet 4.3.2 → 4.4.0
Sign up to get free protection for your applications and to get access to all the features.
Potentially problematic release.
This version of puppet might be problematic. Click here for more details.
- checksums.yaml +4 -4
- data/COMMITTERS.md +2 -2
- data/CONTRIBUTING.md +6 -6
- data/LICENSE +1 -1
- data/README.md +8 -9
- data/conf/auth.conf +2 -2
- data/ext/README.environment +1 -1
- data/ext/debian/README.source +1 -1
- data/ext/debian/control +1 -1
- data/ext/debian/copyright +4 -4
- data/ext/debian/puppetmaster.README.debian +11 -9
- data/ext/emacs/puppet-mode.el +1 -1
- data/ext/envpuppet +2 -2
- data/ext/ips/puppetagent.xml +1 -1
- data/ext/ips/puppetmaster.xml +1 -1
- data/ext/project_data.yaml +8 -0
- data/ext/puppet-test +3 -3
- data/ext/rack/example-passenger-vhost.conf +1 -1
- data/ext/redhat/puppet.spec.erb +2 -2
- data/ext/regexp_nodes/regexp_nodes.rb +1 -1
- data/ext/solaris/pkginfo +1 -1
- data/ext/solaris/smf/puppet.xml +1 -1
- data/ext/suse/puppet.spec +2 -2
- data/ext/upload_facts.rb +1 -1
- data/ext/windows/puppet_interactive.bat +6 -0
- data/ext/windows/puppet_shell.bat +9 -0
- data/ext/windows/run_puppet_interactive.bat +9 -0
- data/ext/yaml_nodes.rb +1 -1
- data/install.rb +30 -20
- data/lib/puppet/agent.rb +1 -1
- data/lib/puppet/application/agent.rb +4 -2
- data/lib/puppet/application/apply.rb +7 -4
- data/lib/puppet/application/cert.rb +1 -1
- data/lib/puppet/application/device.rb +1 -1
- data/lib/puppet/application/filebucket.rb +1 -1
- data/lib/puppet/application/inspect.rb +1 -1
- data/lib/puppet/application/lookup.rb +4 -4
- data/lib/puppet/application/master.rb +2 -2
- data/lib/puppet/application/resource.rb +1 -1
- data/lib/puppet/configurer.rb +100 -22
- data/lib/puppet/data_providers/hiera_config.rb +28 -3
- data/lib/puppet/data_providers/hiera_interpolate.rb +30 -15
- data/lib/puppet/data_providers/hiera_support.rb +1 -1
- data/lib/puppet/data_providers/json_data_provider_factory.rb +2 -2
- data/lib/puppet/data_providers/yaml_data_provider_factory.rb +2 -2
- data/lib/puppet/defaults.rb +65 -19
- data/lib/puppet/environments.rb +3 -1
- data/lib/puppet/face/config.rb +1 -1
- data/lib/puppet/face/epp.rb +1 -1
- data/lib/puppet/face/help/man.erb +1 -1
- data/lib/puppet/face/module/install.rb +6 -6
- data/lib/puppet/face/parser.rb +12 -9
- data/lib/puppet/face/status.rb +2 -1
- data/lib/puppet/feature/cfpropertylist.rb +3 -0
- data/lib/puppet/feature/telnet.rb +9 -0
- data/lib/puppet/file_serving/http_metadata.rb +46 -0
- data/lib/puppet/file_serving/metadata.rb +18 -2
- data/lib/puppet/file_serving/terminus_selector.rb +2 -0
- data/lib/puppet/file_system.rb +2 -2
- data/lib/puppet/file_system/file_impl.rb +2 -2
- data/lib/puppet/file_system/memory_impl.rb +1 -1
- data/lib/puppet/file_system/uniquefile.rb +1 -1
- data/lib/puppet/forge.rb +1 -1
- data/lib/puppet/forge/repository.rb +1 -31
- data/lib/puppet/functions.rb +45 -6
- data/lib/puppet/functions/assert_type.rb +9 -9
- data/lib/puppet/functions/each.rb +5 -13
- data/lib/puppet/functions/filter.rb +5 -14
- data/lib/puppet/functions/map.rb +6 -14
- data/lib/puppet/functions/reduce.rb +5 -13
- data/lib/puppet/functions/reverse_each.rb +82 -0
- data/lib/puppet/functions/scanf.rb +15 -18
- data/lib/puppet/functions/slice.rb +22 -36
- data/lib/puppet/functions/split.rb +2 -2
- data/lib/puppet/functions/step.rb +88 -0
- data/lib/puppet/functions/type.rb +70 -0
- data/lib/puppet/graph/rb_tree_map.rb +1 -1
- data/lib/puppet/indirector/catalog/compiler.rb +188 -5
- data/lib/puppet/indirector/file_content/http.rb +15 -0
- data/lib/puppet/indirector/file_metadata/http.rb +27 -0
- data/lib/puppet/indirector/generic_http.rb +16 -0
- data/lib/puppet/indirector/node/exec.rb +1 -1
- data/lib/puppet/indirector/node/ldap.rb +1 -1
- data/lib/puppet/indirector/rest.rb +2 -1
- data/lib/puppet/info_service/class_information_service.rb +13 -12
- data/lib/puppet/loaders.rb +1 -0
- data/lib/puppet/module.rb +3 -0
- data/lib/puppet/module_tool/skeleton/templates/generator/Gemfile +9 -2
- data/lib/puppet/module_tool/skeleton/templates/generator/spec/classes/init_spec.rb.erb +1 -1
- data/lib/puppet/module_tool/skeleton/templates/generator/tests/init.pp.erb +2 -2
- data/lib/puppet/module_tool/tar/mini.rb +3 -3
- data/lib/puppet/network/http/pool.rb +9 -0
- data/lib/puppet/node.rb +1 -1
- data/lib/puppet/node/environment.rb +11 -2
- data/lib/puppet/parser/ast/pops_bridge.rb +19 -22
- data/lib/puppet/parser/compiler.rb +3 -3
- data/lib/puppet/parser/environment_compiler.rb +0 -1
- data/lib/puppet/parser/functions.rb +28 -16
- data/lib/puppet/parser/functions/fqdn_rand.rb +1 -1
- data/lib/puppet/parser/functions/inline_template.rb +1 -1
- data/lib/puppet/parser/functions/map.rb +1 -1
- data/lib/puppet/parser/functions/scanf.rb +15 -26
- data/lib/puppet/parser/functions/slice.rb +17 -24
- data/lib/puppet/parser/functions/split.rb +1 -1
- data/lib/puppet/parser/resource.rb +19 -17
- data/lib/puppet/parser/scope.rb +176 -5
- data/lib/puppet/plugins/data_providers/data_provider.rb +54 -13
- data/lib/puppet/pops.rb +0 -8
- data/lib/puppet/pops/adaptable.rb +4 -1
- data/lib/puppet/pops/adapters.rb +38 -13
- data/lib/puppet/pops/binder/binder.rb +21 -17
- data/lib/puppet/pops/binder/binder_issues.rb +8 -6
- data/lib/puppet/pops/binder/bindings_checker.rb +12 -8
- data/lib/puppet/pops/binder/bindings_composer.rb +16 -12
- data/lib/puppet/pops/binder/bindings_factory.rb +108 -104
- data/lib/puppet/pops/binder/bindings_model.rb +49 -47
- data/lib/puppet/pops/binder/config/diagnostic_producer.rb +10 -6
- data/lib/puppet/pops/binder/injector.rb +53 -48
- data/lib/puppet/pops/binder/key_factory.rb +10 -6
- data/lib/puppet/pops/binder/producers.rb +67 -62
- data/lib/puppet/pops/evaluator/access_operator.rb +95 -93
- data/lib/puppet/pops/evaluator/closure.rb +84 -68
- data/lib/puppet/pops/evaluator/collector_transformer.rb +18 -14
- data/lib/puppet/pops/evaluator/collectors/exported_collector.rb +0 -1
- data/lib/puppet/pops/evaluator/compare_operator.rb +13 -9
- data/lib/puppet/pops/evaluator/epp_evaluator.rb +9 -8
- data/lib/puppet/pops/evaluator/evaluator_impl.rb +78 -76
- data/lib/puppet/pops/evaluator/json_strict_literal_evaluator.rb +85 -0
- data/lib/puppet/pops/evaluator/relationship_operator.rb +13 -11
- data/lib/puppet/pops/evaluator/runtime3_converter.rb +5 -0
- data/lib/puppet/pops/evaluator/runtime3_support.rb +41 -45
- data/lib/puppet/pops/issue_reporter.rb +6 -4
- data/lib/puppet/pops/issues.rb +34 -11
- data/lib/puppet/pops/loader/base_loader.rb +1 -1
- data/lib/puppet/pops/loader/loader.rb +1 -1
- data/lib/puppet/pops/loader/loader_paths.rb +15 -0
- data/lib/puppet/pops/loader/module_loaders.rb +17 -13
- data/lib/puppet/pops/loader/puppet_function_instantiator.rb +16 -12
- data/lib/puppet/pops/loader/ruby_function_instantiator.rb +16 -3
- data/lib/puppet/pops/loader/type_definition_instantiator.rb +55 -0
- data/lib/puppet/pops/loaders.rb +51 -9
- data/lib/puppet/pops/lookup.rb +14 -12
- data/lib/puppet/pops/merge_strategy.rb +16 -19
- data/lib/puppet/pops/model/factory.rb +26 -2
- data/lib/puppet/pops/model/model.rb +8 -8
- data/lib/puppet/pops/model/model_label_provider.rb +13 -7
- data/lib/puppet/pops/model/model_meta.rb +17 -0
- data/lib/puppet/pops/model/model_tree_dumper.rb +8 -0
- data/lib/puppet/pops/parser/egrammar.ra +38 -14
- data/lib/puppet/pops/parser/eparser.rb +1353 -1276
- data/lib/puppet/pops/parser/epp_support.rb +11 -7
- data/lib/puppet/pops/parser/evaluating_parser.rb +14 -10
- data/lib/puppet/pops/parser/heredoc_support.rb +15 -11
- data/lib/puppet/pops/parser/lexer2.rb +26 -19
- data/lib/puppet/pops/parser/lexer_support.rb +85 -7
- data/lib/puppet/pops/parser/locator.rb +21 -0
- data/lib/puppet/pops/parser/parser_support.rb +19 -16
- data/lib/puppet/pops/parser/slurp_support.rb +11 -7
- data/lib/puppet/pops/types/class_loader.rb +23 -19
- data/lib/puppet/pops/types/enumeration.rb +9 -26
- data/lib/puppet/pops/types/iterable.rb +308 -0
- data/lib/puppet/pops/types/recursion_guard.rb +82 -0
- data/lib/puppet/pops/types/type_acceptor.rb +25 -0
- data/lib/puppet/pops/types/type_asserter.rb +10 -9
- data/lib/puppet/pops/types/type_calculator.rb +138 -381
- data/lib/puppet/pops/types/type_factory.rb +91 -57
- data/lib/puppet/pops/types/type_formatter.rb +334 -0
- data/lib/puppet/pops/types/type_mismatch_describer.rb +226 -59
- data/lib/puppet/pops/types/type_parser.rb +159 -112
- data/lib/puppet/pops/types/types.rb +2057 -1247
- data/lib/puppet/pops/utils.rb +11 -10
- data/lib/puppet/pops/validation.rb +11 -9
- data/lib/puppet/pops/validation/checker4_0.rb +83 -55
- data/lib/puppet/pops/validation/validator_factory_4_0.rb +8 -4
- data/lib/puppet/provider/aixobject.rb +1 -1
- data/lib/puppet/provider/augeas/augeas.rb +1 -1
- data/lib/puppet/provider/cron/crontab.rb +1 -1
- data/lib/puppet/provider/exec/windows.rb +1 -1
- data/lib/puppet/provider/macauthorization/macauthorization.rb +10 -9
- data/lib/puppet/provider/nameservice/directoryservice.rb +35 -50
- data/lib/puppet/provider/package/appdmg.rb +3 -2
- data/lib/puppet/provider/package/dnf.rb +1 -1
- data/lib/puppet/provider/package/pip.rb +5 -8
- data/lib/puppet/provider/package/pip3.rb +1 -1
- data/lib/puppet/provider/package/pkg.rb +1 -1
- data/lib/puppet/provider/package/pkgdmg.rb +3 -2
- data/lib/puppet/provider/package/pkgng.rb +13 -4
- data/lib/puppet/provider/package/windows.rb +1 -1
- data/lib/puppet/provider/package/yum.rb +1 -1
- data/lib/puppet/provider/package/zypper.rb +19 -0
- data/lib/puppet/provider/service/debian.rb +2 -2
- data/lib/puppet/provider/service/launchd.rb +6 -18
- data/lib/puppet/provider/service/systemd.rb +9 -2
- data/lib/puppet/provider/sshkey/parsed.rb +1 -1
- data/lib/puppet/provider/user/aix.rb +1 -1
- data/lib/puppet/provider/user/directoryservice.rb +33 -58
- data/lib/puppet/provider/zfs/zfs.rb +1 -1
- data/lib/puppet/provider/zpool/zpool.rb +1 -1
- data/lib/puppet/reference/configuration.rb +1 -1
- data/lib/puppet/reference/providers.rb +1 -1
- data/lib/puppet/resource.rb +15 -12
- data/lib/puppet/resource/capability_finder.rb +20 -13
- data/lib/puppet/resource/catalog.rb +60 -3
- data/lib/puppet/resource/status.rb +11 -2
- data/lib/puppet/resource/type.rb +28 -38
- data/lib/puppet/settings.rb +1 -1
- data/lib/puppet/settings/config_file.rb +1 -1
- data/lib/puppet/settings/environment_conf.rb +13 -5
- data/lib/puppet/ssl/certificate_factory.rb +3 -3
- data/lib/puppet/ssl/certificate_request.rb +4 -4
- data/lib/puppet/ssl/certificate_signer.rb +1 -1
- data/lib/puppet/ssl/validator/default_validator.rb +1 -1
- data/lib/puppet/test/test_helper.rb +16 -4
- data/lib/puppet/transaction.rb +15 -2
- data/lib/puppet/transaction/additional_resource_generator.rb +6 -2
- data/lib/puppet/transaction/report.rb +31 -1
- data/lib/puppet/transaction/resource_harness.rb +0 -25
- data/lib/puppet/type.rb +11 -11
- data/lib/puppet/type/augeas.rb +1 -1
- data/lib/puppet/type/cron.rb +12 -12
- data/lib/puppet/type/file.rb +91 -39
- data/lib/puppet/type/file/checksum_value.rb +53 -0
- data/lib/puppet/type/file/content.rb +26 -111
- data/lib/puppet/type/file/data_sync.rb +84 -0
- data/lib/puppet/type/file/ensure.rb +17 -14
- data/lib/puppet/type/file/selcontext.rb +1 -1
- data/lib/puppet/type/file/source.rb +103 -18
- data/lib/puppet/type/filebucket.rb +1 -1
- data/lib/puppet/type/interface.rb +8 -3
- data/lib/puppet/type/macauthorization.rb +1 -1
- data/lib/puppet/type/package.rb +6 -0
- data/lib/puppet/type/schedule.rb +1 -1
- data/lib/puppet/type/stage.rb +1 -1
- data/lib/puppet/type/user.rb +19 -17
- data/lib/puppet/type/yumrepo.rb +20 -0
- data/lib/puppet/util.rb +109 -22
- data/lib/puppet/util/autoload.rb +16 -11
- data/lib/puppet/util/checksums.rb +74 -31
- data/lib/puppet/util/execution.rb +1 -1
- data/lib/puppet/util/http_proxy.rb +72 -0
- data/lib/puppet/util/log.rb +2 -0
- data/lib/puppet/util/logging.rb +43 -1
- data/lib/puppet/util/monkey_patches.rb +2 -2
- data/lib/puppet/util/multi_match.rb +51 -0
- data/lib/puppet/util/network_device/cisco/device.rb +10 -2
- data/lib/puppet/util/network_device/cisco/interface.rb +21 -8
- data/lib/puppet/util/network_device/transport/ssh.rb +7 -3
- data/lib/puppet/util/network_device/transport/telnet.rb +39 -36
- data/lib/puppet/util/plist.rb +130 -0
- data/lib/puppet/util/resource_template.rb +1 -1
- data/lib/puppet/util/run_mode.rb +2 -2
- data/lib/puppet/util/skip_tags.rb +9 -0
- data/lib/puppet/util/windows/access_control_entry.rb +1 -1
- data/lib/puppet/util/windows/access_control_list.rb +3 -3
- data/lib/puppet/util/windows/adsi.rb +4 -4
- data/lib/puppet/util/windows/api_types.rb +24 -18
- data/lib/puppet/util/windows/com.rb +3 -3
- data/lib/puppet/util/windows/error.rb +1 -1
- data/lib/puppet/util/windows/file.rb +8 -8
- data/lib/puppet/util/windows/principal.rb +23 -14
- data/lib/puppet/util/windows/process.rb +78 -11
- data/lib/puppet/util/windows/registry.rb +1 -1
- data/lib/puppet/util/windows/root_certs.rb +5 -5
- data/lib/puppet/util/windows/security.rb +33 -35
- data/lib/puppet/util/windows/security_descriptor.rb +1 -1
- data/lib/puppet/util/windows/sid.rb +42 -4
- data/lib/puppet/util/windows/taskscheduler.rb +15 -15
- data/lib/puppet/util/windows/user.rb +10 -10
- data/lib/puppet/vendor/deep_merge/deep_merge.gemspec +1 -1
- data/lib/puppet/vendor/pathspec/LICENSE +2 -2
- data/lib/puppet/vendor/pathspec/README.md +1 -1
- data/lib/puppet/vendor/rgen/README.rdoc +1 -1
- data/lib/puppet/vendor/semantic/lib/semantic/dependency/module_release.rb +14 -0
- data/lib/puppet/version.rb +1 -1
- data/lib/semver.rb +17 -1
- data/man/man5/puppet.conf.5 +12 -12
- data/man/man8/extlookup2hiera.8 +1 -1
- data/man/man8/puppet-agent.8 +2 -2
- data/man/man8/puppet-apply.8 +2 -2
- data/man/man8/puppet-ca.8 +2 -2
- data/man/man8/puppet-catalog.8 +2 -2
- data/man/man8/puppet-cert.8 +2 -2
- data/man/man8/puppet-certificate.8 +2 -2
- data/man/man8/puppet-certificate_request.8 +2 -2
- data/man/man8/puppet-certificate_revocation_list.8 +2 -2
- data/man/man8/puppet-config.8 +3 -3
- data/man/man8/puppet-describe.8 +1 -1
- data/man/man8/puppet-device.8 +1 -1
- data/man/man8/puppet-doc.8 +1 -1
- data/man/man8/puppet-epp.8 +2 -2
- data/man/man8/puppet-facts.8 +2 -2
- data/man/man8/puppet-file.8 +2 -2
- data/man/man8/puppet-filebucket.8 +2 -2
- data/man/man8/puppet-help.8 +2 -2
- data/man/man8/puppet-inspect.8 +2 -2
- data/man/man8/puppet-key.8 +2 -2
- data/man/man8/puppet-man.8 +2 -2
- data/man/man8/puppet-master.8 +2 -2
- data/man/man8/puppet-module.8 +9 -9
- data/man/man8/puppet-node.8 +2 -2
- data/man/man8/puppet-parser.8 +2 -2
- data/man/man8/puppet-plugin.8 +2 -2
- data/man/man8/puppet-report.8 +2 -2
- data/man/man8/puppet-resource.8 +2 -2
- data/man/man8/puppet-resource_type.8 +2 -2
- data/man/man8/puppet-status.8 +3 -3
- data/man/man8/puppet.8 +1 -1
- data/spec/fixtures/module.tar.gz +0 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/bad_data/lib/puppet/functions/bad_data/data.rb +1 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/bad_data/manifests/init.pp +0 -1
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_json/data/empty.json +0 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_json/hiera.yaml +5 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_json/manifests/init.pp +2 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_json/metadata.json +9 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_yaml/data/empty.yaml +1 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_yaml/hiera.yaml +5 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_yaml/manifests/init.pp +2 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/empty_yaml/metadata.json +9 -0
- data/spec/fixtures/unit/functions/lookup/environments/production/modules/hieraprovider/data/first.json +2 -1
- data/spec/fixtures/unit/module/trailing-comma.json +1 -1
- data/spec/fixtures/unit/pops/loaders/loaders/dependent_modules_with_metadata/modules/usee/manifests/init.pp +3 -1
- data/spec/fixtures/unit/pops/loaders/loaders/dependent_modules_with_metadata/modules/usee/types/zero.pp +1 -0
- data/spec/fixtures/unit/pops/loaders/loaders/dependent_modules_with_metadata/modules/user/types/withuseeone.pp +1 -0
- data/spec/fixtures/unit/pops/loaders/loaders/dependent_modules_with_metadata/modules/user/types/withuseezero.pp +1 -0
- data/spec/fixtures/unit/provider/package/yum/yum-check-update-broken-notices.txt +187 -0
- data/spec/fixtures/unit/provider/sshkey/parsed/sample_with_blank_lines +8 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_md5/should_fetch_if_not_on_the_local_disk.yml +205 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_md5/should_not_update_if_content_on_disk_is_up-to-date.yml +213 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_md5/should_update_if_content_differs_on_disk.yml +213 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_mtime/should_fetch_if_mtime_is_older_on_disk.yml +205 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_mtime/should_fetch_if_no_header_specified.yml +197 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_mtime/should_fetch_if_not_on_the_local_disk.yml +205 -0
- data/spec/fixtures/vcr/cassettes/Puppet_Type_File/when_sourcing/from_http/using_mtime/should_not_update_if_mtime_is_newer_on_disk.yml +205 -0
- data/spec/integration/defaults_spec.rb +14 -2
- data/spec/integration/file_system/uniquefile_spec.rb +29 -0
- data/spec/integration/module_tool/tar/mini_spec.rb +28 -0
- data/spec/integration/node/environment_spec.rb +13 -0
- data/spec/integration/parser/dynamic_scoping_spec.rb +67 -0
- data/spec/integration/parser/parameter_defaults_spec.rb +336 -0
- data/spec/integration/parser/undef_param_spec.rb +8 -0
- data/spec/integration/provider/yumrepo_spec.rb +1 -1
- data/spec/integration/test/test_helper_spec.rb +28 -0
- data/spec/integration/transaction/report_spec.rb +16 -0
- data/spec/integration/transaction_spec.rb +11 -0
- data/spec/integration/type/file_spec.rb +194 -4
- data/spec/integration/type/package_spec.rb +5 -1
- data/spec/integration/type/tidy_spec.rb +21 -9
- data/spec/integration/util/execution_spec.rb +22 -0
- data/spec/integration/util/windows/principal_spec.rb +90 -4
- data/spec/integration/util/windows/process_spec.rb +31 -0
- data/spec/integration/util/windows/security_spec.rb +6 -6
- data/spec/integration/util/windows/user_spec.rb +1 -1
- data/spec/integration/util_spec.rb +49 -27
- data/spec/lib/puppet_spec/compiler.rb +17 -0
- data/spec/lib/puppet_spec/files.rb +2 -2
- data/spec/lib/puppet_spec/pops.rb +13 -0
- data/spec/shared_behaviours/iterative_functions.rb +1 -1
- data/spec/shared_contexts/types_setup.rb +96 -0
- data/spec/unit/agent_spec.rb +1 -0
- data/spec/unit/application/agent_spec.rb +10 -0
- data/spec/unit/application/apply_spec.rb +9 -0
- data/spec/unit/configurer/downloader_spec.rb +5 -5
- data/spec/unit/configurer_spec.rb +271 -39
- data/spec/unit/data_providers/hiera_interpolation_spec.rb +57 -0
- data/spec/unit/defaults_spec.rb +15 -0
- data/spec/unit/environments_spec.rb +24 -4
- data/spec/unit/face/parser_spec.rb +43 -2
- data/spec/unit/file_serving/http_metadata_spec.rb +85 -0
- data/spec/unit/file_serving/metadata_spec.rb +50 -0
- data/spec/unit/file_serving/terminus_selector_spec.rb +12 -2
- data/spec/unit/file_system_spec.rb +26 -0
- data/spec/unit/functions/assert_type_spec.rb +36 -2
- data/spec/unit/functions/defined_spec.rb +2 -2
- data/spec/unit/functions/epp_spec.rb +11 -3
- data/spec/unit/functions/lookup_spec.rb +58 -13
- data/spec/unit/functions/regsubst_spec.rb +1 -1
- data/spec/unit/functions/reverse_each_spec.rb +108 -0
- data/spec/unit/functions/step_spec.rb +113 -0
- data/spec/unit/functions/type_spec.rb +35 -0
- data/spec/unit/functions4_spec.rb +61 -5
- data/spec/unit/indirector/catalog/compiler_spec.rb +705 -4
- data/spec/unit/indirector/file_bucket_file/file_spec.rb +1 -1
- data/spec/unit/indirector/indirection_spec.rb +1 -1
- data/spec/unit/info_service_spec.rb +94 -32
- data/spec/unit/module_spec.rb +14 -0
- data/spec/unit/module_tool/applications/builder_spec.rb +4 -4
- data/spec/unit/network/authstore_spec.rb +1 -1
- data/spec/unit/network/http/connection_spec.rb +1 -0
- data/spec/unit/network/http/pool_spec.rb +30 -0
- data/spec/unit/node_spec.rb +1 -1
- data/spec/unit/parser/compiler_spec.rb +16 -0
- data/spec/unit/parser/scope_spec.rb +28 -11
- data/spec/unit/pops/evaluator/access_ops_spec.rb +3 -3
- data/spec/unit/pops/evaluator/comparison_ops_spec.rb +3 -0
- data/spec/unit/pops/evaluator/evaluating_parser_spec.rb +7 -1
- data/spec/unit/pops/evaluator/evaluator_rspec_helper.rb +4 -4
- data/spec/unit/pops/evaluator/json_strict_literal_evaluator_spec.rb +63 -0
- data/spec/unit/pops/evaluator/runtime3_converter_spec.rb +6 -0
- data/spec/unit/pops/loaders/dependency_loader_spec.rb +53 -0
- data/spec/unit/pops/loaders/loaders_spec.rb +44 -1
- data/spec/unit/pops/parser/lexer2_spec.rb +112 -3
- data/spec/unit/pops/parser/parse_calls_spec.rb +8 -0
- data/spec/unit/pops/parser/parser_spec.rb +10 -0
- data/spec/unit/pops/parser/source_pos_adapter_spec.rb +26 -0
- data/spec/unit/pops/types/iterable_spec.rb +262 -0
- data/spec/unit/pops/types/recursion_guard_spec.rb +91 -0
- data/spec/unit/pops/types/type_acceptor_spec.rb +105 -0
- data/spec/unit/pops/types/type_asserter_spec.rb +43 -0
- data/spec/unit/pops/types/type_calculator_spec.rb +275 -373
- data/spec/unit/pops/types/type_formatter_spec.rb +280 -0
- data/spec/unit/pops/types/type_mismatch_describer_spec.rb +152 -0
- data/spec/unit/pops/types/type_parser_spec.rb +58 -13
- data/spec/unit/pops/types/types_spec.rb +241 -0
- data/spec/unit/pops/validator/validator_spec.rb +100 -43
- data/spec/unit/provider/cron/parsed_spec.rb +1 -0
- data/spec/unit/provider/macauthorization_spec.rb +5 -2
- data/spec/unit/provider/nameservice/directoryservice_spec.rb +14 -19
- data/spec/unit/provider/package/appdmg_spec.rb +3 -3
- data/spec/unit/provider/package/dnf_spec.rb +16 -0
- data/spec/unit/provider/package/pip3_spec.rb +60 -42
- data/spec/unit/provider/package/pip_spec.rb +47 -34
- data/spec/unit/provider/package/pkgdmg_spec.rb +18 -9
- data/spec/unit/provider/package/pkgng_spec.rb +4 -2
- data/spec/unit/provider/package/yum_spec.rb +11 -0
- data/spec/unit/provider/package/zypper_spec.rb +14 -0
- data/spec/unit/provider/service/launchd_spec.rb +17 -35
- data/spec/unit/provider/service/systemd_spec.rb +7 -0
- data/spec/unit/provider/sshkey/parsed_spec.rb +20 -19
- data/spec/unit/provider/user/directoryservice_spec.rb +40 -59
- data/spec/unit/resource/capability_finder_spec.rb +28 -15
- data/spec/unit/resource/catalog_spec.rb +33 -1
- data/spec/unit/resource/type_spec.rb +149 -7
- data/spec/unit/resource_spec.rb +96 -57
- data/spec/unit/settings/environment_conf_spec.rb +18 -1
- data/spec/unit/ssl/certificate_revocation_list_spec.rb +3 -3
- data/spec/unit/transaction/report_spec.rb +27 -0
- data/spec/unit/transaction/resource_harness_spec.rb +0 -47
- data/spec/unit/transaction_spec.rb +5 -0
- data/spec/unit/type/file/checksum_spec.rb +6 -0
- data/spec/unit/type/file/checksum_value_spec.rb +286 -0
- data/spec/unit/type/file/content_spec.rb +12 -193
- data/spec/unit/type/file/source_spec.rb +211 -119
- data/spec/unit/type/file_spec.rb +133 -34
- data/spec/unit/type/interface_spec.rb +32 -0
- data/spec/unit/type/macauthorization_spec.rb +4 -1
- data/spec/unit/type/yumrepo_spec.rb +2 -2
- data/spec/unit/util/filetype_spec.rb +1 -1
- data/spec/unit/util/http_proxy_spec.rb +2 -2
- data/spec/unit/util/log/destinations_spec.rb +0 -2
- data/spec/unit/util/logging_spec.rb +69 -0
- data/spec/unit/util/multi_match_spec.rb +39 -0
- data/spec/unit/util/network_device/cisco/device_spec.rb +253 -216
- data/spec/unit/util/network_device/transport/telnet_spec.rb +60 -58
- data/spec/unit/util/plist_spec.rb +110 -0
- data/spec/unit/util/resource_template_spec.rb +2 -2
- data/spec/unit/util/run_mode_spec.rb +27 -3
- data/spec/unit/util/windows/adsi_spec.rb +4 -4
- data/spec/unit/util/windows/api_types_spec.rb +42 -0
- data/spec/unit/util/windows/security_descriptor_spec.rb +3 -3
- data/spec/unit/util/windows/sid_spec.rb +1 -1
- data/spec/unit/util_spec.rb +123 -13
- data/tasks/cfpropertylist.rake +15 -0
- metadata +114 -26
- data/lib/puppet/vendor/load_plist.rb +0 -1
- data/lib/puppet/vendor/plist/CHANGELOG +0 -82
- data/lib/puppet/vendor/plist/MIT-LICENSE +0 -21
- data/lib/puppet/vendor/plist/PUPPET_README.md +0 -6
- data/lib/puppet/vendor/plist/README +0 -36
- data/lib/puppet/vendor/plist/Rakefile +0 -144
- data/lib/puppet/vendor/plist/docs/USAGE +0 -104
- data/lib/puppet/vendor/plist/docs/jamis-template.rb +0 -591
- data/lib/puppet/vendor/plist/lib/plist.rb +0 -22
- data/lib/puppet/vendor/plist/lib/plist/generator.rb +0 -224
- data/lib/puppet/vendor/plist/lib/plist/parser.rb +0 -225
- data/lib/puppet/vendor/plist/test/assets/AlbumData.xml +0 -203
- data/lib/puppet/vendor/plist/test/assets/Cookies.plist +0 -104
- data/lib/puppet/vendor/plist/test/assets/commented.plist +0 -9
- data/lib/puppet/vendor/plist/test/assets/example_data.bin +0 -0
- data/lib/puppet/vendor/plist/test/assets/example_data.jpg +0 -0
- data/lib/puppet/vendor/plist/test/assets/example_data.plist +0 -259
- data/lib/puppet/vendor/plist/test/assets/test_data_elements.plist +0 -24
- data/lib/puppet/vendor/plist/test/assets/test_empty_key.plist +0 -13
- data/lib/puppet/vendor/plist/test/test_data_elements.rb +0 -115
- data/lib/puppet/vendor/plist/test/test_generator.rb +0 -59
- data/lib/puppet/vendor/plist/test/test_generator_basic_types.rb +0 -58
- data/lib/puppet/vendor/plist/test/test_generator_collections.rb +0 -82
- data/lib/puppet/vendor/plist/test/test_parser.rb +0 -90
@@ -1,11 +1,26 @@
|
|
1
|
+
require_relative 'iterable'
|
2
|
+
require_relative 'enumeration'
|
3
|
+
require_relative 'recursion_guard'
|
4
|
+
require_relative 'type_acceptor'
|
5
|
+
require_relative 'type_asserter'
|
6
|
+
require_relative 'type_assertion_error'
|
7
|
+
require_relative 'type_formatter'
|
8
|
+
require_relative 'type_calculator'
|
9
|
+
require_relative 'type_factory'
|
10
|
+
require_relative 'type_parser'
|
11
|
+
require_relative 'class_loader'
|
12
|
+
require_relative 'type_mismatch_describer'
|
13
|
+
|
1
14
|
require 'rgen/metamodel_builder'
|
2
15
|
|
16
|
+
module Puppet::Pops
|
17
|
+
module Types
|
3
18
|
# The Types model is a model of Puppet Language types.
|
4
19
|
#
|
5
20
|
# The exact relationship between types is not visible in this model wrt. the PDataType which is an abstraction
|
6
21
|
# of Scalar, Array[Data], and Hash[Scalar, Data] nested to any depth. This means it is not possible to
|
7
|
-
# infer the type by simply looking at the inheritance hierarchy. The {
|
8
|
-
# be used to answer questions about types. The {
|
22
|
+
# infer the type by simply looking at the inheritance hierarchy. The {TypeCalculator} should
|
23
|
+
# be used to answer questions about types. The {TypeFactory} should be used to create an instance
|
9
24
|
# of a type whenever one is needed.
|
10
25
|
#
|
11
26
|
# The implementation of the Types model contains methods that are required for the type objects to behave as
|
@@ -14,1535 +29,2330 @@ require 'rgen/metamodel_builder'
|
|
14
29
|
#
|
15
30
|
# @api public
|
16
31
|
#
|
17
|
-
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
24
|
-
|
25
|
-
|
26
|
-
|
27
|
-
|
28
|
-
|
29
|
-
|
30
|
-
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
35
|
-
|
36
|
-
|
37
|
-
|
38
|
-
|
39
|
-
|
32
|
+
# TODO: See PUP-2978 for possible performance optimization
|
33
|
+
class TypedModelObject < Object
|
34
|
+
include Visitable
|
35
|
+
include Adaptable
|
36
|
+
end
|
37
|
+
|
38
|
+
# Base type for all types
|
39
|
+
# @api public
|
40
|
+
#
|
41
|
+
class PAnyType < TypedModelObject
|
42
|
+
# Accept a visitor that will be sent the message `visit`, once with `self` as the
|
43
|
+
# argument. The visitor will then visit all types that this type contains.
|
44
|
+
#
|
45
|
+
def accept(visitor, guard)
|
46
|
+
visitor.visit(self, guard)
|
47
|
+
end
|
48
|
+
|
49
|
+
# Checks if _o_ is a type that is assignable to this type.
|
50
|
+
# If _o_ is a `Class` then it is first converted to a type.
|
51
|
+
# If _o_ is a Variant, then it is considered assignable when all its types are assignable
|
52
|
+
#
|
53
|
+
# The check for assignable must be guarded against self recursion since `self`, the given type _o_,
|
54
|
+
# or both, might be a `TypeAlias`. The initial caller of this method will typically never care
|
55
|
+
# about this and hence pass only the first argument, but as soon as a check of a contained type
|
56
|
+
# encounters a `TypeAlias`, then a `RecursionGuard` instance is created and passed on in all
|
57
|
+
# subsequent calls. The recursion is allowed to continue until self recursion has been detected in
|
58
|
+
# both `self` and in the given type. At that point the given type is considered to be assignable
|
59
|
+
# to `self` since all checks up to that point were positive.
|
60
|
+
#
|
61
|
+
# @param o [Class,PAnyType] the class or type to test
|
62
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
63
|
+
# @return [Boolean] `true` when _o_ is assignable to this type
|
64
|
+
# @api public
|
65
|
+
def assignable?(o, guard = nil)
|
66
|
+
case o
|
67
|
+
when Class
|
68
|
+
# Safe to call _assignable directly since a Class never is a Unit or Variant
|
69
|
+
_assignable?(TypeCalculator.singleton.type(o), guard)
|
70
|
+
when PUnitType
|
71
|
+
true
|
72
|
+
when PTypeAliasType
|
73
|
+
# An alias may contain self recursive constructs.
|
74
|
+
if o.self_recursion?
|
75
|
+
guard ||= RecursionGuard.new
|
76
|
+
if guard.add_that(o) == RecursionGuard::SELF_RECURSION_IN_BOTH
|
77
|
+
# Recursion detected both in self and other. This means that other is assignable
|
78
|
+
# to self. This point would not have been reached otherwise
|
40
79
|
true
|
41
|
-
when PVariantType
|
42
|
-
# Assignable if all contained types are assignable
|
43
|
-
o.types.all? { |vt| assignable?(vt) }
|
44
|
-
when PNotUndefType
|
45
|
-
if !(o.type.nil? || o.type.assignable?(PUndefType::DEFAULT))
|
46
|
-
assignable?(o.type)
|
47
|
-
else
|
48
|
-
_assignable?(o)
|
49
|
-
end
|
50
80
|
else
|
51
|
-
|
81
|
+
assignable?(o.resolved_type, guard)
|
52
82
|
end
|
53
|
-
|
83
|
+
else
|
84
|
+
assignable?(o.resolved_type, guard)
|
85
|
+
end
|
86
|
+
when PVariantType
|
87
|
+
# Assignable if all contained types are assignable
|
88
|
+
o.types.all? { |vt| assignable?(vt, guard) }
|
89
|
+
when PNotUndefType
|
90
|
+
if !(o.type.nil? || o.type.assignable?(PUndefType::DEFAULT))
|
91
|
+
assignable?(o.type, guard)
|
92
|
+
else
|
93
|
+
_assignable?(o, guard)
|
94
|
+
end
|
95
|
+
else
|
96
|
+
_assignable?(o, guard)
|
97
|
+
end
|
98
|
+
end
|
54
99
|
|
55
|
-
|
56
|
-
|
57
|
-
|
58
|
-
|
59
|
-
|
60
|
-
|
100
|
+
# Returns `true` if this instance is a callable that accepts the given _args_
|
101
|
+
#
|
102
|
+
# @param args [PAnyType] the arguments to test
|
103
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
104
|
+
# @return [Boolean] `true` if this instance is a callable that accepts the given _args_
|
105
|
+
def callable?(args, guard = nil)
|
106
|
+
args.is_a?(PAnyType) && kind_of_callable? && args.callable_args?(self, guard)
|
107
|
+
end
|
61
108
|
|
62
|
-
|
63
|
-
|
64
|
-
|
65
|
-
|
66
|
-
|
67
|
-
|
109
|
+
# Returns `true` if this instance is considered valid as arguments to the given `callable`
|
110
|
+
# @param callable [PAnyType] the callable
|
111
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
112
|
+
# @return [Boolean] `true` if this instance is considered valid as arguments to the given `callable`
|
113
|
+
# @api private
|
114
|
+
def callable_args?(callable, guard)
|
115
|
+
false
|
116
|
+
end
|
68
117
|
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
73
|
-
|
118
|
+
# Generalizes value specific types. Types that are not value specific will return `self` otherwise
|
119
|
+
# the generalized type is returned.
|
120
|
+
#
|
121
|
+
# @return [PAnyType] The generalized type
|
122
|
+
# @api public
|
123
|
+
def generalize
|
124
|
+
# Applicable to all types that have no variables
|
125
|
+
self
|
126
|
+
end
|
74
127
|
|
75
|
-
|
76
|
-
|
77
|
-
|
78
|
-
|
79
|
-
|
80
|
-
|
81
|
-
|
82
|
-
|
83
|
-
|
128
|
+
# Normalizes the type. This does not change the characteristics of the type but it will remove duplicates
|
129
|
+
# and constructs like NotUndef[T] where T is not assignable from Undef and change Variant[*T] where all
|
130
|
+
# T are enums into an Enum.
|
131
|
+
#
|
132
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
133
|
+
# @return [PAnyType] The iterable type that this type is assignable to or `nil`
|
134
|
+
# @api public
|
135
|
+
def normalize(guard = nil)
|
136
|
+
self
|
137
|
+
end
|
84
138
|
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
90
|
-
|
91
|
-
|
139
|
+
# Responds `true` for all callables, variants of callables and unless _optional_ is
|
140
|
+
# false, all optional callables.
|
141
|
+
# @param optional [Boolean]
|
142
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
143
|
+
# @return [Boolean] `true`if this type is considered callable
|
144
|
+
# @api private
|
145
|
+
def kind_of_callable?(optional = true, guard = nil)
|
146
|
+
false
|
147
|
+
end
|
92
148
|
|
93
|
-
|
94
|
-
|
95
|
-
|
149
|
+
# Returns `true` if an instance of this type is iterable, `false` otherwise
|
150
|
+
# The method #iterable_type must produce a `PIterableType` instance when this
|
151
|
+
# method returns `true`
|
152
|
+
#
|
153
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
154
|
+
# @return [Boolean] flag to indicate if instances of this type is iterable.
|
155
|
+
def iterable?(guard = nil)
|
156
|
+
false
|
157
|
+
end
|
96
158
|
|
97
|
-
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
159
|
+
# Returns the `PIterableType` that this type should be assignable to, or `nil` if no such type exists.
|
160
|
+
# A type that returns a `PIterableType` must respond `true` to `#iterable?`.
|
161
|
+
#
|
162
|
+
# @example
|
163
|
+
# Any Collection[T] is assignable to an Iterable[T]
|
164
|
+
# A String is assignable to an Iterable[String] iterating over the strings characters
|
165
|
+
# An Integer is assignable to an Iterable[Integer] iterating over the 'times' enumerator
|
166
|
+
# A Type[T] is assignable to an Iterable[Type[T]] if T is an Integer or Enum
|
167
|
+
#
|
168
|
+
# @param guard [RecursionGuard] guard against recursion. Only used by internal calls
|
169
|
+
# @return [PIterableType,nil] The iterable type that this type is assignable to or `nil`
|
170
|
+
# @api private
|
171
|
+
def iterable_type(guard = nil)
|
172
|
+
nil
|
173
|
+
end
|
102
174
|
|
103
|
-
|
104
|
-
|
105
|
-
|
175
|
+
def hash
|
176
|
+
self.class.hash
|
177
|
+
end
|
106
178
|
|
107
|
-
|
179
|
+
# Returns true if the given argument _o_ is an instance of this type
|
180
|
+
# @return [Boolean]
|
181
|
+
def instance?(o)
|
182
|
+
true
|
183
|
+
end
|
108
184
|
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
def simple_name
|
113
|
-
n = self.class.name
|
114
|
-
n[n.rindex('::')+3..n.size-5]
|
115
|
-
end
|
185
|
+
def eql?(o)
|
186
|
+
self.class == o.class
|
187
|
+
end
|
116
188
|
|
117
|
-
|
118
|
-
|
119
|
-
|
189
|
+
def ==(o)
|
190
|
+
eql?(o)
|
191
|
+
end
|
120
192
|
|
121
|
-
|
122
|
-
|
123
|
-
|
124
|
-
|
193
|
+
# Strips the class name from all module prefixes, the leading 'P' and the ending 'Type'. I.e.
|
194
|
+
# an instance of PVariantType will return 'Variant'
|
195
|
+
# @return [String] the simple name of this type
|
196
|
+
def simple_name
|
197
|
+
n = self.class.name
|
198
|
+
n[n.rindex('::')+3..n.size-5]
|
199
|
+
end
|
125
200
|
|
126
|
-
|
201
|
+
def to_alias_expanded_s
|
202
|
+
TypeFormatter.new.alias_expanded_string(self)
|
203
|
+
end
|
127
204
|
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
end
|
205
|
+
def to_s
|
206
|
+
TypeFormatter.string(self)
|
207
|
+
end
|
132
208
|
|
133
|
-
|
209
|
+
# The default instance of this type. Each type in the type system has this constant
|
210
|
+
# declared.
|
211
|
+
#
|
212
|
+
DEFAULT = PAnyType.new
|
134
213
|
|
135
|
-
|
136
|
-
def class_from_string(str)
|
137
|
-
begin
|
138
|
-
str.split(NAME_SEGMENT_SEPARATOR).reduce(Object) do |memo, name_segment|
|
139
|
-
memo.const_get(name_segment)
|
140
|
-
end
|
141
|
-
rescue NameError
|
142
|
-
return nil
|
143
|
-
end
|
144
|
-
end
|
214
|
+
protected
|
145
215
|
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
#
|
151
|
-
# @api private
|
152
|
-
#
|
153
|
-
def tuple_entry_at(tuple_t, from, to, index)
|
154
|
-
regular = (tuple_t.types.size - 1)
|
155
|
-
if index < regular
|
156
|
-
tuple_t.types[index]
|
157
|
-
elsif index < regular + to
|
158
|
-
# in the varargs part
|
159
|
-
tuple_t.types[-1]
|
160
|
-
else
|
161
|
-
nil
|
162
|
-
end
|
163
|
-
end
|
216
|
+
# @api private
|
217
|
+
def _assignable?(o, guard)
|
218
|
+
o.is_a?(PAnyType)
|
219
|
+
end
|
164
220
|
|
221
|
+
NAME_SEGMENT_SEPARATOR = '::'.freeze
|
165
222
|
|
166
|
-
|
167
|
-
|
168
|
-
|
169
|
-
|
170
|
-
|
171
|
-
# @api private
|
172
|
-
def type_to_range(size_type)
|
173
|
-
return [1,1] if size_type.nil?
|
174
|
-
from = size_type.from
|
175
|
-
to = size_type.to
|
176
|
-
[from.nil? ? 1 : from, to.nil? ? Float::INFINITY : to]
|
223
|
+
# @api private
|
224
|
+
def class_from_string(str)
|
225
|
+
begin
|
226
|
+
str.split(NAME_SEGMENT_SEPARATOR).reduce(Object) do |memo, name_segment|
|
227
|
+
memo.const_get(name_segment)
|
177
228
|
end
|
229
|
+
rescue NameError
|
230
|
+
return nil
|
178
231
|
end
|
232
|
+
end
|
179
233
|
|
180
|
-
|
181
|
-
|
182
|
-
|
183
|
-
|
184
|
-
|
185
|
-
|
186
|
-
|
187
|
-
|
188
|
-
|
234
|
+
# Produces the tuple entry at the given index given a tuple type, its from/to constraints on the last
|
235
|
+
# type, and an index.
|
236
|
+
# Produces nil if the index is out of bounds
|
237
|
+
# from must be less than to, and from may not be less than 0
|
238
|
+
#
|
239
|
+
# @api private
|
240
|
+
#
|
241
|
+
def tuple_entry_at(tuple_t, from, to, index)
|
242
|
+
regular = (tuple_t.types.size - 1)
|
243
|
+
if index < regular
|
244
|
+
tuple_t.types[index]
|
245
|
+
elsif index < regular + to
|
246
|
+
# in the varargs part
|
247
|
+
tuple_t.types[-1]
|
248
|
+
else
|
249
|
+
nil
|
250
|
+
end
|
251
|
+
end
|
189
252
|
|
190
|
-
def instance?(o)
|
191
|
-
if o.is_a?(PAnyType)
|
192
|
-
type.nil? || type.assignable?(o)
|
193
|
-
else
|
194
|
-
assignable?(TypeCalculator.infer(o))
|
195
|
-
end
|
196
|
-
end
|
197
253
|
|
198
|
-
|
199
|
-
|
200
|
-
|
254
|
+
# Transform size_type to min, max
|
255
|
+
# if size_type == nil the constraint is 1,1
|
256
|
+
# if size_type.from == nil min size = 1
|
257
|
+
# if size_type.to == nil max size == Infinity
|
258
|
+
#
|
259
|
+
# @api private
|
260
|
+
def type_to_range(size_type)
|
261
|
+
return [1,1] if size_type.nil?
|
262
|
+
from = size_type.from
|
263
|
+
to = size_type.to
|
264
|
+
[from.nil? ? 1 : from, to.nil? ? Float::INFINITY : to]
|
265
|
+
end
|
201
266
|
|
202
|
-
|
203
|
-
|
204
|
-
|
267
|
+
# Applies a transformation by sending the given _method_ and _method_args_ to each of the types of the given array
|
268
|
+
# and collecting the results in a new array. If all transformation calls returned the type instance itself (i.e. no
|
269
|
+
# transformation took place), then this method will return `self`. If a transformation did occur, then this method
|
270
|
+
# will either return the transformed array or in case a block was given, the result of calling a given block with
|
271
|
+
# the transformed array.
|
272
|
+
#
|
273
|
+
# @param types [Array<PAnyType>] the array of types to transform
|
274
|
+
# @param method [Symbol] The method to call on each type
|
275
|
+
# @param method_args [Object] The arguments to pass to the method, if any
|
276
|
+
# @return [Object] self, the transformed array, or the result of calling a given block with the transformed array
|
277
|
+
# @yieldparam altered_types [Array<PAnyType>] the altered type array
|
278
|
+
# @api private
|
279
|
+
def alter_type_array(types, method, *method_args)
|
280
|
+
modified = false
|
281
|
+
modified_types = types.map do |t|
|
282
|
+
t_mod = t.send(method, *method_args)
|
283
|
+
modified = !t.equal?(t_mod) unless modified
|
284
|
+
t_mod
|
285
|
+
end
|
286
|
+
if modified
|
287
|
+
block_given? ? yield(modified_types) : modified_types
|
288
|
+
else
|
289
|
+
self
|
290
|
+
end
|
291
|
+
end
|
292
|
+
end
|
205
293
|
|
206
|
-
|
207
|
-
|
208
|
-
|
294
|
+
# @abstract Encapsulates common behavior for a type that contains one type
|
295
|
+
# @api public
|
296
|
+
class PTypeWithContainedType < PAnyType
|
297
|
+
attr_reader :type
|
209
298
|
|
210
|
-
|
211
|
-
|
212
|
-
|
213
|
-
end
|
299
|
+
def initialize(type)
|
300
|
+
@type = type
|
301
|
+
end
|
214
302
|
|
215
|
-
|
303
|
+
def accept(visitor, guard)
|
304
|
+
super
|
305
|
+
@type.accept(visitor, guard) unless @type.nil?
|
306
|
+
end
|
216
307
|
|
217
|
-
|
308
|
+
def generalize
|
309
|
+
if @type.nil?
|
310
|
+
self.class::DEFAULT
|
311
|
+
else
|
312
|
+
ge_type = @type.generalize
|
313
|
+
@type.equal?(ge_type) ? self : self.class.new(ge_type)
|
314
|
+
end
|
315
|
+
end
|
218
316
|
|
219
|
-
|
220
|
-
|
221
|
-
|
222
|
-
|
223
|
-
|
224
|
-
|
225
|
-
end
|
317
|
+
def normalize(guard = nil)
|
318
|
+
if @type.nil?
|
319
|
+
self.class::DEFAULT
|
320
|
+
else
|
321
|
+
ne_type = @type.normalize(guard)
|
322
|
+
@type.equal?(ne_type) ? self : self.class.new(ne_type)
|
226
323
|
end
|
324
|
+
end
|
227
325
|
|
228
|
-
|
229
|
-
|
326
|
+
def hash
|
327
|
+
self.class.hash * 31 * @type.hash
|
328
|
+
end
|
230
329
|
|
231
|
-
|
232
|
-
|
233
|
-
|
330
|
+
def eql?(o)
|
331
|
+
self.class == o.class && @type == o.type
|
332
|
+
end
|
333
|
+
end
|
234
334
|
|
235
|
-
|
236
|
-
|
237
|
-
|
335
|
+
# The type of types.
|
336
|
+
# @api public
|
337
|
+
#
|
338
|
+
class PType < PTypeWithContainedType
|
339
|
+
def instance?(o)
|
340
|
+
if o.is_a?(PAnyType)
|
341
|
+
type.nil? || type.assignable?(o)
|
342
|
+
else
|
343
|
+
assignable?(TypeCalculator.infer(o))
|
344
|
+
end
|
345
|
+
end
|
238
346
|
|
239
|
-
|
240
|
-
|
241
|
-
|
347
|
+
def iterable?(guard = nil)
|
348
|
+
case @type
|
349
|
+
when PEnumType
|
350
|
+
true
|
351
|
+
when PIntegerType
|
352
|
+
@type.finite_range?
|
353
|
+
else
|
354
|
+
false
|
355
|
+
end
|
356
|
+
end
|
242
357
|
|
243
|
-
|
244
|
-
|
245
|
-
|
358
|
+
def iterable_type(guard = nil)
|
359
|
+
# The types PIntegerType and PEnumType are Iterable
|
360
|
+
case @type
|
361
|
+
when PEnumType
|
362
|
+
# @type describes the element type perfectly since the iteration is made over the
|
363
|
+
# contained choices.
|
364
|
+
PIterableType.new(@type)
|
365
|
+
when PIntegerType
|
366
|
+
# @type describes the element type perfectly since the iteration is made over the
|
367
|
+
# specified range.
|
368
|
+
@type.finite_range? ? PIterableType.new(@type) : nil
|
369
|
+
else
|
370
|
+
nil
|
371
|
+
end
|
372
|
+
end
|
246
373
|
|
247
|
-
|
248
|
-
|
249
|
-
|
374
|
+
def eql?(o)
|
375
|
+
self.class == o.class && @type == o.type
|
376
|
+
end
|
250
377
|
|
251
|
-
|
378
|
+
def simple_name
|
379
|
+
# since this the class is inconsistently named PType and not PTypeType
|
380
|
+
'Type'
|
381
|
+
end
|
252
382
|
|
253
|
-
|
383
|
+
DEFAULT = PType.new(nil)
|
254
384
|
|
255
|
-
|
256
|
-
def _assignable?(o)
|
257
|
-
o.is_a?(PAnyType) && !o.assignable?(PUndefType::DEFAULT) && (@type.nil? || @type.assignable?(o))
|
258
|
-
end
|
259
|
-
end
|
385
|
+
protected
|
260
386
|
|
261
|
-
|
262
|
-
|
263
|
-
|
264
|
-
|
265
|
-
|
266
|
-
|
387
|
+
# @api private
|
388
|
+
def _assignable?(o, guard)
|
389
|
+
return false unless o.is_a?(PType)
|
390
|
+
return true if @type.nil? # wide enough to handle all types
|
391
|
+
return false if o.type.nil? # wider than t
|
392
|
+
@type.assignable?(o.type, guard)
|
393
|
+
end
|
394
|
+
end
|
267
395
|
|
268
|
-
|
269
|
-
|
270
|
-
|
271
|
-
|
272
|
-
end
|
396
|
+
class PNotUndefType < PTypeWithContainedType
|
397
|
+
def initialize(type = nil)
|
398
|
+
super(type.class == PAnyType ? nil : type)
|
399
|
+
end
|
273
400
|
|
274
|
-
|
401
|
+
def instance?(o)
|
402
|
+
!(o.nil? || o == :undef) && (@type.nil? || @type.instance?(o))
|
403
|
+
end
|
275
404
|
|
276
|
-
|
277
|
-
|
278
|
-
|
279
|
-
|
280
|
-
|
405
|
+
def normalize(guard = nil)
|
406
|
+
n = super
|
407
|
+
if n.type.nil?
|
408
|
+
n
|
409
|
+
else
|
410
|
+
if n.type.is_a?(POptionalType)
|
411
|
+
# No point in having an optional in a NotUndef
|
412
|
+
PNotUndef.new(n.type.type).normalize
|
413
|
+
elsif !n.type.assignable?(PUndefType::DEFAULT)
|
414
|
+
# THe type is NotUndef anyway, so it can be stripped of
|
415
|
+
n.type
|
416
|
+
else
|
417
|
+
n
|
418
|
+
end
|
281
419
|
end
|
420
|
+
end
|
282
421
|
|
283
|
-
|
284
|
-
# @api private
|
285
|
-
#
|
286
|
-
class PUnitType < PAnyType
|
287
|
-
def instance?(o)
|
288
|
-
true
|
289
|
-
end
|
422
|
+
DEFAULT = PNotUndefType.new
|
290
423
|
|
291
|
-
|
424
|
+
protected
|
292
425
|
|
293
|
-
|
294
|
-
|
295
|
-
|
296
|
-
|
297
|
-
|
298
|
-
end
|
426
|
+
# @api private
|
427
|
+
def _assignable?(o, guard)
|
428
|
+
o.is_a?(PAnyType) && !o.assignable?(PUndefType::DEFAULT, guard) && (@type.nil? || @type.assignable?(o, guard))
|
429
|
+
end
|
430
|
+
end
|
299
431
|
|
300
|
-
|
301
|
-
|
302
|
-
|
303
|
-
|
304
|
-
|
305
|
-
|
432
|
+
# @api public
|
433
|
+
#
|
434
|
+
class PUndefType < PAnyType
|
435
|
+
def instance?(o)
|
436
|
+
o.nil? || o == :undef
|
437
|
+
end
|
306
438
|
|
307
|
-
|
439
|
+
# @api private
|
440
|
+
def callable_args?(callable_t, guard)
|
441
|
+
# if callable_t is Optional (or indeed PUndefType), this means that 'missing callable' is accepted
|
442
|
+
callable_t.assignable?(DEFAULT, guard)
|
443
|
+
end
|
308
444
|
|
309
|
-
|
310
|
-
# @api private
|
311
|
-
def _assignable?(o)
|
312
|
-
o.is_a?(PDefaultType)
|
313
|
-
end
|
314
|
-
end
|
445
|
+
DEFAULT = PUndefType.new
|
315
446
|
|
316
|
-
|
317
|
-
|
318
|
-
|
319
|
-
|
320
|
-
|
321
|
-
|
322
|
-
self.class == o.class || o.class == PVariantType && o == PVariantType::DATA
|
323
|
-
end
|
447
|
+
protected
|
448
|
+
# @api private
|
449
|
+
def _assignable?(o, guard)
|
450
|
+
o.is_a?(PUndefType)
|
451
|
+
end
|
452
|
+
end
|
324
453
|
|
325
|
-
|
326
|
-
|
327
|
-
|
454
|
+
# A type private to the type system that describes "ignored type" - i.e. "I am what you are"
|
455
|
+
# @api private
|
456
|
+
#
|
457
|
+
class PUnitType < PAnyType
|
458
|
+
def instance?(o)
|
459
|
+
true
|
460
|
+
end
|
328
461
|
|
329
|
-
|
462
|
+
DEFAULT = PUnitType.new
|
330
463
|
|
331
|
-
|
464
|
+
protected
|
465
|
+
# @api private
|
466
|
+
def _assignable?(o, guard)
|
467
|
+
true
|
468
|
+
end
|
469
|
+
end
|
332
470
|
|
333
|
-
|
334
|
-
|
335
|
-
|
336
|
-
|
337
|
-
|
338
|
-
|
339
|
-
true
|
340
|
-
when Types::PNotUndefType
|
341
|
-
assignable?(o.type || PUndefType::DEFAULT)
|
342
|
-
else
|
343
|
-
PVariantType::DATA.assignable?(o)
|
344
|
-
end
|
345
|
-
end
|
346
|
-
end
|
471
|
+
# @api public
|
472
|
+
#
|
473
|
+
class PDefaultType < PAnyType
|
474
|
+
def instance?(o)
|
475
|
+
o == :default
|
476
|
+
end
|
347
477
|
|
348
|
-
|
349
|
-
# @api public
|
350
|
-
#
|
351
|
-
class PScalarType < PAnyType
|
478
|
+
DEFAULT = PDefaultType.new
|
352
479
|
|
353
|
-
|
354
|
-
|
355
|
-
|
480
|
+
protected
|
481
|
+
# @api private
|
482
|
+
def _assignable?(o, guard)
|
483
|
+
o.is_a?(PDefaultType)
|
484
|
+
end
|
485
|
+
end
|
356
486
|
|
357
|
-
|
487
|
+
# A flexible data type, being assignable to its subtypes as well as PArrayType and PHashType with element type assignable to PDataType.
|
488
|
+
#
|
489
|
+
# @api public
|
490
|
+
#
|
491
|
+
class PDataType < PAnyType
|
492
|
+
def eql?(o)
|
493
|
+
self.class == o.class || o == PVariantType::DATA
|
494
|
+
end
|
358
495
|
|
359
|
-
|
496
|
+
def instance?(o)
|
497
|
+
PVariantType::DATA.instance?(o)
|
498
|
+
end
|
360
499
|
|
361
|
-
|
362
|
-
|
363
|
-
|
364
|
-
|
500
|
+
DEFAULT = PDataType.new
|
501
|
+
|
502
|
+
protected
|
503
|
+
|
504
|
+
# Data is assignable by other Data and by Array[Data] and Hash[Scalar, Data]
|
505
|
+
# @api private
|
506
|
+
def _assignable?(o, guard)
|
507
|
+
# We cannot put the NotUndefType[Data] in the @data_variant_t since that causes an endless recursion
|
508
|
+
case o
|
509
|
+
when Types::PDataType
|
510
|
+
true
|
511
|
+
when Types::PNotUndefType
|
512
|
+
assignable?(o.type || PUndefType::DEFAULT, guard)
|
513
|
+
else
|
514
|
+
PVariantType::DATA.assignable?(o, guard)
|
365
515
|
end
|
516
|
+
end
|
517
|
+
end
|
366
518
|
|
367
|
-
|
368
|
-
|
369
|
-
|
370
|
-
|
371
|
-
include Enumerable
|
519
|
+
# Type that is PDataType compatible, but is not a PCollectionType.
|
520
|
+
# @api public
|
521
|
+
#
|
522
|
+
class PScalarType < PAnyType
|
372
523
|
|
373
|
-
|
524
|
+
def instance?(o)
|
525
|
+
assignable?(TypeCalculator.infer(o))
|
526
|
+
end
|
374
527
|
|
375
|
-
|
376
|
-
@values = values.sort.freeze
|
377
|
-
end
|
528
|
+
DEFAULT = PScalarType.new
|
378
529
|
|
379
|
-
|
380
|
-
# block with each of the strings for this enum
|
381
|
-
def each
|
382
|
-
if block_given?
|
383
|
-
values.each { |x| yield x }
|
384
|
-
else
|
385
|
-
values.to_enum
|
386
|
-
end
|
387
|
-
end
|
530
|
+
protected
|
388
531
|
|
389
|
-
|
390
|
-
|
391
|
-
|
532
|
+
# @api private
|
533
|
+
def _assignable?(o, guard)
|
534
|
+
o.is_a?(PScalarType)
|
535
|
+
end
|
536
|
+
end
|
392
537
|
|
393
|
-
|
394
|
-
|
395
|
-
|
538
|
+
# A string type describing the set of strings having one of the given values
|
539
|
+
# @api public
|
540
|
+
#
|
541
|
+
class PEnumType < PScalarType
|
542
|
+
attr_reader :values
|
396
543
|
|
397
|
-
|
544
|
+
def initialize(values)
|
545
|
+
@values = values.sort.freeze
|
546
|
+
end
|
398
547
|
|
399
|
-
|
548
|
+
# Returns Enumerator if no block is given, otherwise, calls the given
|
549
|
+
# block with each of the strings for this enum
|
550
|
+
def each(&block)
|
551
|
+
r = Iterable.on(self)
|
552
|
+
block_given? ? r.each(&block) : r
|
553
|
+
end
|
400
554
|
|
401
|
-
|
402
|
-
|
403
|
-
|
404
|
-
svalues = values
|
405
|
-
if svalues.empty?
|
406
|
-
return true if o.is_a?(PStringType) || o.is_a?(PEnumType) || o.is_a?(PPatternType)
|
407
|
-
end
|
408
|
-
case o
|
409
|
-
when PStringType
|
410
|
-
# if the set of strings are all found in the set of enums
|
411
|
-
!o.values.empty? && o.values.all? { |s| svalues.any? { |e| e == s }}
|
412
|
-
when PEnumType
|
413
|
-
!o.values.empty? && o.values.all? { |s| svalues.any? {|e| e == s }}
|
414
|
-
else
|
415
|
-
false
|
416
|
-
end
|
417
|
-
end
|
418
|
-
end
|
555
|
+
def iterable?(guard = nil)
|
556
|
+
true
|
557
|
+
end
|
419
558
|
|
420
|
-
|
421
|
-
#
|
422
|
-
|
423
|
-
|
424
|
-
from = -Float::INFINITY if from.nil? || from == :default
|
425
|
-
to = Float::INFINITY if to.nil? || to == :default
|
426
|
-
raise ArgumentError, "'from' must be less or equal to 'to'. Got (#{from}, #{to}" if from.is_a?(Numeric) && to.is_a?(Numeric) && from > to
|
427
|
-
@from = from
|
428
|
-
@to = to
|
429
|
-
end
|
559
|
+
def iterable_type(guard = nil)
|
560
|
+
# An instance of an Enum is a String
|
561
|
+
PStringType::ITERABLE_TYPE
|
562
|
+
end
|
430
563
|
|
431
|
-
|
432
|
-
|
433
|
-
|
434
|
-
@from == -Float::INFINITY ? nil : @from
|
435
|
-
end
|
564
|
+
def hash
|
565
|
+
@values.hash
|
566
|
+
end
|
436
567
|
|
437
|
-
|
438
|
-
|
439
|
-
|
440
|
-
@to == Float::INFINITY ? nil : @to
|
441
|
-
end
|
568
|
+
def eql?(o)
|
569
|
+
self.class == o.class && @values == o.values
|
570
|
+
end
|
442
571
|
|
443
|
-
|
444
|
-
# @return [Float,Integer]
|
445
|
-
def numeric_from
|
446
|
-
@from
|
447
|
-
end
|
572
|
+
DEFAULT = PEnumType.new(EMPTY_ARRAY)
|
448
573
|
|
449
|
-
|
450
|
-
# @return [Float,Integer]
|
451
|
-
def numeric_to
|
452
|
-
@to
|
453
|
-
end
|
574
|
+
protected
|
454
575
|
|
455
|
-
|
456
|
-
|
457
|
-
|
576
|
+
# @api private
|
577
|
+
def _assignable?(o, guard)
|
578
|
+
return true if self == o
|
579
|
+
svalues = values
|
580
|
+
if svalues.empty?
|
581
|
+
return true if o.is_a?(PStringType) || o.is_a?(PEnumType) || o.is_a?(PPatternType)
|
582
|
+
end
|
583
|
+
case o
|
584
|
+
when PStringType
|
585
|
+
# if the set of strings are all found in the set of enums
|
586
|
+
!o.values.empty? && o.values.all? { |s| svalues.any? { |e| e == s }}
|
587
|
+
when PEnumType
|
588
|
+
!o.values.empty? && o.values.all? { |s| svalues.any? {|e| e == s }}
|
589
|
+
else
|
590
|
+
false
|
591
|
+
end
|
592
|
+
end
|
593
|
+
end
|
458
594
|
|
459
|
-
|
460
|
-
|
461
|
-
|
595
|
+
# @api public
|
596
|
+
#
|
597
|
+
class PNumericType < PScalarType
|
598
|
+
def initialize(from, to = Float::INFINITY)
|
599
|
+
from = -Float::INFINITY if from.nil? || from == :default
|
600
|
+
to = Float::INFINITY if to.nil? || to == :default
|
601
|
+
raise ArgumentError, "'from' must be less or equal to 'to'. Got (#{from}, #{to}" if from.is_a?(Numeric) && to.is_a?(Numeric) && from > to
|
602
|
+
@from = from
|
603
|
+
@to = to
|
604
|
+
end
|
462
605
|
|
463
|
-
|
464
|
-
|
465
|
-
|
606
|
+
# Checks if this numeric range intersects with another
|
607
|
+
#
|
608
|
+
# @param o [PNumericType] the range to compare with
|
609
|
+
# @return [Boolean] `true` if this range intersects with the other range
|
610
|
+
# @api public
|
611
|
+
def intersect?(o)
|
612
|
+
self.class == o.class && !(@to < o.from || o.to < @from)
|
613
|
+
end
|
466
614
|
|
467
|
-
|
468
|
-
|
469
|
-
|
615
|
+
# Returns the lower bound of the numeric range or `nil` if no lower bound is set.
|
616
|
+
# @return [Float,Integer]
|
617
|
+
def from
|
618
|
+
@from == -Float::INFINITY ? nil : @from
|
619
|
+
end
|
470
620
|
|
471
|
-
|
621
|
+
# Returns the upper bound of the numeric range or `nil` if no upper bound is set.
|
622
|
+
# @return [Float,Integer]
|
623
|
+
def to
|
624
|
+
@to == Float::INFINITY ? nil : @to
|
625
|
+
end
|
472
626
|
|
473
|
-
|
627
|
+
# Same as #from but will return `-Float::Infinity` instead of `nil` if no lower bound is set.
|
628
|
+
# @return [Float,Integer]
|
629
|
+
def numeric_from
|
630
|
+
@from
|
631
|
+
end
|
474
632
|
|
475
|
-
|
476
|
-
|
477
|
-
|
478
|
-
|
479
|
-
|
480
|
-
end
|
481
|
-
end
|
633
|
+
# Same as #to but will return `Float::Infinity` instead of `nil` if no lower bound is set.
|
634
|
+
# @return [Float,Integer]
|
635
|
+
def numeric_to
|
636
|
+
@to
|
637
|
+
end
|
482
638
|
|
483
|
-
|
484
|
-
|
485
|
-
|
486
|
-
# The integer type is enumerable when it defines a range
|
487
|
-
include Enumerable
|
639
|
+
def hash
|
640
|
+
@from.hash * 31 + @to.hash
|
641
|
+
end
|
488
642
|
|
489
|
-
|
490
|
-
|
491
|
-
|
643
|
+
def eql?(o)
|
644
|
+
self.class == o.class && @from == o.numeric_from && @to == o.numeric_to
|
645
|
+
end
|
492
646
|
|
493
|
-
|
494
|
-
|
495
|
-
|
647
|
+
def instance?(o)
|
648
|
+
o.is_a?(Numeric) && o >= @from && o <= @to
|
649
|
+
end
|
496
650
|
|
497
|
-
|
498
|
-
|
499
|
-
|
651
|
+
def unbounded?
|
652
|
+
@from == -Float::INFINITY && @to == Float::INFINITY
|
653
|
+
end
|
500
654
|
|
501
|
-
|
502
|
-
def size
|
503
|
-
return Float::INFINITY if @from == -Float::INFINITY || @to == Float::INFINITY
|
504
|
-
1+(to-from).abs
|
505
|
-
end
|
655
|
+
DEFAULT = PNumericType.new(-Float::INFINITY)
|
506
656
|
|
507
|
-
|
508
|
-
# The number may be Infinity or -Infinity.
|
509
|
-
def range
|
510
|
-
[@from, @to]
|
511
|
-
end
|
657
|
+
protected
|
512
658
|
|
513
|
-
|
514
|
-
|
515
|
-
|
516
|
-
|
517
|
-
|
518
|
-
|
519
|
-
|
520
|
-
end
|
521
|
-
end
|
659
|
+
# @api_private
|
660
|
+
def _assignable?(o, guard)
|
661
|
+
return false unless o.is_a?(self.class)
|
662
|
+
# If o min and max are within the range of t
|
663
|
+
@from <= o.numeric_from && @to >= o.numeric_to
|
664
|
+
end
|
665
|
+
end
|
522
666
|
|
523
|
-
|
524
|
-
|
525
|
-
|
526
|
-
|
527
|
-
|
528
|
-
|
667
|
+
# @api public
|
668
|
+
#
|
669
|
+
class PIntegerType < PNumericType
|
670
|
+
# Will respond `true` for any range that is bounded at both ends.
|
671
|
+
#
|
672
|
+
# @return [Boolean] `true` if the type describes a finite range.
|
673
|
+
def finite_range?
|
674
|
+
@from != -Float::INFINITY && @to != Float::INFINITY
|
675
|
+
end
|
529
676
|
|
530
|
-
|
531
|
-
|
677
|
+
def generalize
|
678
|
+
DEFAULT
|
679
|
+
end
|
532
680
|
|
533
|
-
|
534
|
-
|
535
|
-
|
536
|
-
def generalize
|
537
|
-
DEFAULT
|
538
|
-
end
|
681
|
+
def instance?(o)
|
682
|
+
o.is_a?(Integer) && o >= numeric_from && o <= numeric_to
|
683
|
+
end
|
539
684
|
|
540
|
-
|
541
|
-
|
542
|
-
|
685
|
+
# Checks if this range is adjacent to the given range
|
686
|
+
#
|
687
|
+
# @param o [PIntegerType] the range to compare with
|
688
|
+
# @return [Boolean] `true` if this range is adjacent to the other range
|
689
|
+
# @api public
|
690
|
+
def adjacent?(o)
|
691
|
+
o.is_a?(PIntegerType) && (@to + 1 == o.from || o.to + 1 == @from)
|
692
|
+
end
|
543
693
|
|
544
|
-
|
694
|
+
# Concatenates this range with another range provided that the ranges intersect or
|
695
|
+
# are adjacent. When that's not the case, this method will return `nil`
|
696
|
+
#
|
697
|
+
# @param o [PIntegerType] the range to concatenate with this range
|
698
|
+
# @return [PIntegerType,nil] the concatenated range or `nil` when the ranges were apart
|
699
|
+
# @api public
|
700
|
+
def merge(o)
|
701
|
+
if intersect?(o) || adjacent?(o)
|
702
|
+
min = @from <= o.from ? @from : o.from
|
703
|
+
max = @to >= o.to ? @to : o.to
|
704
|
+
PIntegerType.new(min, max)
|
705
|
+
else
|
706
|
+
nil
|
545
707
|
end
|
708
|
+
end
|
546
709
|
|
547
|
-
|
548
|
-
|
549
|
-
|
550
|
-
attr_reader :element_type, :size_type
|
551
|
-
|
552
|
-
def initialize(element_type, size_type = nil)
|
553
|
-
@element_type = element_type
|
554
|
-
@size_type = size_type
|
555
|
-
end
|
710
|
+
def iterable?(guard = nil)
|
711
|
+
true
|
712
|
+
end
|
556
713
|
|
557
|
-
|
558
|
-
|
559
|
-
|
714
|
+
def iterable_type(guard = nil)
|
715
|
+
# It's unknown if the iterable will be a range (min, max) or a "times" (0, max)
|
716
|
+
PIterableType.new(PIntegerType::DEFAULT)
|
717
|
+
end
|
560
718
|
|
561
|
-
|
562
|
-
|
563
|
-
|
719
|
+
# Returns Float.Infinity if one end of the range is unbound
|
720
|
+
def size
|
721
|
+
return Float::INFINITY if @from == -Float::INFINITY || @to == Float::INFINITY
|
722
|
+
1+(to-from).abs
|
723
|
+
end
|
564
724
|
|
565
|
-
|
566
|
-
|
567
|
-
|
568
|
-
|
725
|
+
# Returns the range as an array ordered so the smaller number is always first.
|
726
|
+
# The number may be Infinity or -Infinity.
|
727
|
+
def range
|
728
|
+
[@from, @to]
|
729
|
+
end
|
569
730
|
|
570
|
-
|
571
|
-
|
572
|
-
|
731
|
+
# Returns Enumerator if no block is given
|
732
|
+
# Returns nil if size is infinity (does not yield)
|
733
|
+
def each(&block)
|
734
|
+
r = Iterable.on(self)
|
735
|
+
block_given? ? r.each(&block) : r
|
736
|
+
end
|
573
737
|
|
574
|
-
|
575
|
-
|
576
|
-
|
738
|
+
# Returns a range where both to and from are positive numbers. Negative
|
739
|
+
# numbers are converted to zero
|
740
|
+
# @return [PIntegerType] a positive range
|
741
|
+
def to_size
|
742
|
+
@from >= 0 ? self : PIntegerType.new(0, @to < 0 ? 0 : @to)
|
743
|
+
end
|
577
744
|
|
745
|
+
DEFAULT = PIntegerType.new(-Float::INFINITY)
|
746
|
+
end
|
578
747
|
|
579
|
-
|
580
|
-
|
581
|
-
|
748
|
+
# @api public
|
749
|
+
#
|
750
|
+
class PFloatType < PNumericType
|
751
|
+
def generalize
|
752
|
+
DEFAULT
|
753
|
+
end
|
582
754
|
|
583
|
-
|
755
|
+
def instance?(o)
|
756
|
+
o.is_a?(Float) && o >= numeric_from && o <= numeric_to
|
757
|
+
end
|
584
758
|
|
585
|
-
|
586
|
-
|
587
|
-
|
588
|
-
|
589
|
-
|
590
|
-
|
591
|
-
|
592
|
-
|
593
|
-
|
594
|
-
|
595
|
-
|
596
|
-
|
597
|
-
|
598
|
-
from = to = o.elements.size
|
599
|
-
(@size_type || DEFAULT_SIZE).assignable?(PIntegerType.new(from, to))
|
600
|
-
else
|
601
|
-
false
|
602
|
-
end
|
603
|
-
end
|
759
|
+
# Concatenates this range with another range provided that the ranges intersect. When that's not the case, this
|
760
|
+
# method will return `nil`
|
761
|
+
#
|
762
|
+
# @param o [PFloatType] the range to concatenate with this range
|
763
|
+
# @return [PFloatType,nil] the concatenated range or `nil` when the ranges were apart
|
764
|
+
# @api public
|
765
|
+
def merge(o)
|
766
|
+
if intersect?(o)
|
767
|
+
min = @from <= o.from ? @from : o.from
|
768
|
+
max = @to >= o.to ? @to : o.to
|
769
|
+
PFloatType.new(min, max)
|
770
|
+
else
|
771
|
+
nil
|
604
772
|
end
|
773
|
+
end
|
605
774
|
|
606
|
-
|
607
|
-
|
608
|
-
class PStringType < PScalarType
|
609
|
-
attr_reader :size_type, :values
|
775
|
+
DEFAULT = PFloatType.new(-Float::INFINITY)
|
776
|
+
end
|
610
777
|
|
611
|
-
|
612
|
-
|
613
|
-
|
778
|
+
# @api public
|
779
|
+
#
|
780
|
+
class PCollectionType < PAnyType
|
781
|
+
attr_reader :element_type, :size_type
|
614
782
|
|
615
|
-
|
616
|
-
|
617
|
-
|
618
|
-
|
783
|
+
def initialize(element_type, size_type = nil)
|
784
|
+
@element_type = element_type
|
785
|
+
@size_type = size_type
|
786
|
+
end
|
619
787
|
|
620
|
-
|
621
|
-
|
622
|
-
|
788
|
+
def accept(visitor, guard)
|
789
|
+
super
|
790
|
+
@size_type.accept(visitor, guard) unless @size_type.nil?
|
791
|
+
@element_type.accept(visitor, guard) unless @element_type.nil?
|
792
|
+
end
|
623
793
|
|
624
|
-
|
625
|
-
|
626
|
-
|
794
|
+
def generalize
|
795
|
+
if @element_type.nil?
|
796
|
+
DEFAULT
|
797
|
+
else
|
798
|
+
ge_type = @element_type.generalize
|
799
|
+
@size_type.nil? && @element_type.equal?(ge_type) ? self : self.class.new(ge_type, nil)
|
800
|
+
end
|
801
|
+
end
|
627
802
|
|
628
|
-
|
629
|
-
|
630
|
-
|
631
|
-
|
632
|
-
|
633
|
-
|
634
|
-
end
|
635
|
-
end
|
636
|
-
|
637
|
-
DEFAULT = PStringType.new(nil)
|
638
|
-
NON_EMPTY = PStringType.new(PIntegerType.new(1))
|
639
|
-
|
640
|
-
protected
|
641
|
-
|
642
|
-
# @api private
|
643
|
-
def _assignable?(o)
|
644
|
-
if values.empty?
|
645
|
-
# A general string is assignable by any other string or pattern restricted string
|
646
|
-
# if the string has a size constraint it does not match since there is no reasonable way
|
647
|
-
# to compute the min/max length a pattern will match. For enum, it is possible to test that
|
648
|
-
# each enumerator value is within range
|
649
|
-
case o
|
650
|
-
when PStringType
|
651
|
-
# true if size compliant
|
652
|
-
(@size_type || PCollectionType::DEFAULT_SIZE).assignable?(o.size_type || PCollectionType::DEFAULT_SIZE)
|
653
|
-
|
654
|
-
when PPatternType
|
655
|
-
# true if size constraint is at least 0 to +Infinity (which is the same as the default)
|
656
|
-
@size_type.nil? || @size_type.assignable?(PCollectionType::DEFAULT_SIZE)
|
657
|
-
|
658
|
-
when PEnumType
|
659
|
-
if o.values.empty?
|
660
|
-
# enum represents all enums, and thus all strings, a sized constrained string can thus not
|
661
|
-
# be assigned any enum (unless it is max size).
|
662
|
-
@size_type.nil? || @size_type.assignable?(PCollectionType::DEFAULT_SIZE)
|
663
|
-
else
|
664
|
-
# true if all enum values are within range
|
665
|
-
orange = o.values.map(&:size).minmax
|
666
|
-
srange = (@size_type || PCollectionType::DEFAULT_SIZE).range
|
667
|
-
# If o min and max are within the range of t
|
668
|
-
srange[0] <= orange[0] && srange[1] >= orange[1]
|
669
|
-
end
|
670
|
-
else
|
671
|
-
# no other type matches string
|
672
|
-
false
|
673
|
-
end
|
674
|
-
elsif o.is_a?(PStringType)
|
675
|
-
# A specific string acts as a set of strings - must have exactly the same strings
|
676
|
-
# In this case, size does not matter since the definition is very precise anyway
|
677
|
-
values == o.values
|
678
|
-
else
|
679
|
-
# All others are false, since no other type describes the same set of specific strings
|
680
|
-
false
|
681
|
-
end
|
682
|
-
end
|
803
|
+
def normalize(guard = nil)
|
804
|
+
if @element_type.nil?
|
805
|
+
DEFAULT
|
806
|
+
else
|
807
|
+
ne_type = @element_type.normalize(guard)
|
808
|
+
@element_type.equal?(ne_type) ? self : self.class.new(ne_type, @size_type)
|
683
809
|
end
|
810
|
+
end
|
684
811
|
|
685
|
-
|
686
|
-
|
687
|
-
|
688
|
-
attr_reader :pattern
|
812
|
+
def instance?(o)
|
813
|
+
assignable?(TypeCalculator.infer(o))
|
814
|
+
end
|
689
815
|
|
690
|
-
|
691
|
-
|
692
|
-
|
816
|
+
# Returns an array with from (min) size to (max) size
|
817
|
+
def size_range
|
818
|
+
(@size_type || DEFAULT_SIZE).range
|
819
|
+
end
|
693
820
|
|
694
|
-
|
695
|
-
|
696
|
-
|
821
|
+
def hash
|
822
|
+
@element_type.hash * 31 + @size_type.hash
|
823
|
+
end
|
697
824
|
|
698
|
-
|
699
|
-
|
700
|
-
|
825
|
+
def iterable?(guard = nil)
|
826
|
+
true
|
827
|
+
end
|
701
828
|
|
702
|
-
|
703
|
-
|
704
|
-
|
829
|
+
def iterable_type(guard = nil)
|
830
|
+
@element_type.nil? ? PIterableType::DEFAULT : PIterableType.new(@element_type)
|
831
|
+
end
|
832
|
+
|
833
|
+
def eql?(o)
|
834
|
+
self.class == o.class && @element_type == o.element_type && @size_type == o.size_type
|
835
|
+
end
|
705
836
|
|
706
|
-
DEFAULT = PRegexpType.new(nil)
|
707
837
|
|
708
|
-
|
838
|
+
DEFAULT_SIZE = PIntegerType.new(0)
|
839
|
+
ZERO_SIZE = PIntegerType.new(0, 0)
|
840
|
+
DEFAULT = PCollectionType.new(nil)
|
709
841
|
|
710
|
-
|
711
|
-
|
712
|
-
|
713
|
-
|
714
|
-
|
842
|
+
protected
|
843
|
+
|
844
|
+
# @api private
|
845
|
+
#
|
846
|
+
def _assignable?(o, guard)
|
847
|
+
case o
|
848
|
+
when PCollectionType
|
849
|
+
(@size_type || DEFAULT_SIZE).assignable?(o.size_type || DEFAULT_SIZE, guard)
|
850
|
+
when PTupleType
|
851
|
+
# compute the tuple's min/max size, and check if that size matches
|
852
|
+
from, to = type_to_range(o.size_type)
|
853
|
+
from = o.types.size - 1 + from
|
854
|
+
to = o.types.size - 1 + to
|
855
|
+
(@size_type || DEFAULT_SIZE).assignable?(PIntegerType.new(from, to), guard)
|
856
|
+
when PStructType
|
857
|
+
from = to = o.elements.size
|
858
|
+
(@size_type || DEFAULT_SIZE).assignable?(PIntegerType.new(from, to), guard)
|
859
|
+
else
|
860
|
+
false
|
715
861
|
end
|
862
|
+
end
|
863
|
+
end
|
716
864
|
|
717
|
-
|
718
|
-
|
719
|
-
|
720
|
-
|
721
|
-
#
|
722
|
-
class PPatternType < PScalarType
|
723
|
-
attr_reader :patterns
|
865
|
+
class PIterableType < PTypeWithContainedType
|
866
|
+
def element_type
|
867
|
+
@type
|
868
|
+
end
|
724
869
|
|
725
|
-
|
726
|
-
|
870
|
+
def instance?(o)
|
871
|
+
if @type.nil? || @type.assignable?(PAnyType::DEFAULT)
|
872
|
+
# Any element_type will do
|
873
|
+
case o
|
874
|
+
when Iterable, String, Hash, Array, Range, PEnumType
|
875
|
+
true
|
876
|
+
when Integer
|
877
|
+
o >= 0
|
878
|
+
when PIntegerType
|
879
|
+
o.finite_range?
|
880
|
+
else
|
881
|
+
false
|
727
882
|
end
|
883
|
+
else
|
884
|
+
assignable?(TypeCalculator.infer(o))
|
885
|
+
end
|
886
|
+
end
|
728
887
|
|
729
|
-
|
730
|
-
|
731
|
-
|
888
|
+
def iterable?(guard = nil)
|
889
|
+
true
|
890
|
+
end
|
732
891
|
|
733
|
-
|
734
|
-
|
735
|
-
|
892
|
+
def iterable_type(guard = nil)
|
893
|
+
self
|
894
|
+
end
|
736
895
|
|
737
|
-
|
896
|
+
DEFAULT = PIterableType.new(nil)
|
738
897
|
|
739
|
-
|
898
|
+
protected
|
740
899
|
|
741
|
-
|
742
|
-
|
743
|
-
|
744
|
-
|
745
|
-
|
746
|
-
|
900
|
+
# @api private
|
901
|
+
def _assignable?(o, guard)
|
902
|
+
if @type.nil? || @type.assignable?(PAnyType::DEFAULT, guard)
|
903
|
+
# Don't request the iterable_type. Since this Iterable accepts Any element, it is enough that o is iterable.
|
904
|
+
o.iterable?
|
905
|
+
else
|
906
|
+
o = o.iterable_type
|
907
|
+
o.nil? || o.element_type.nil? ? false : @type.assignable?(o.element_type, guard)
|
908
|
+
end
|
909
|
+
end
|
910
|
+
end
|
911
|
+
|
912
|
+
# @api public
|
913
|
+
#
|
914
|
+
class PIteratorType < PTypeWithContainedType
|
915
|
+
def element_type
|
916
|
+
@type
|
917
|
+
end
|
918
|
+
|
919
|
+
def instance?(o)
|
920
|
+
o.is_a?(Iterable) && (@element_type.nil? || @element_type.assignable?(o.element_type))
|
921
|
+
end
|
922
|
+
|
923
|
+
def iterable?(guard = nil)
|
924
|
+
true
|
925
|
+
end
|
926
|
+
|
927
|
+
def iterable_type(guard = nil)
|
928
|
+
@type.nil? ? PIteratbleType::DEFAULT : PIterableType.new(@element_type)
|
929
|
+
end
|
930
|
+
|
931
|
+
DEFAULT = PIteratorType.new(nil)
|
932
|
+
|
933
|
+
protected
|
934
|
+
|
935
|
+
# @api private
|
936
|
+
def _assignable?(o, guard)
|
937
|
+
o.is_a?(PIteratorType) && (@element_type.nil? || @element_type.assignable?(o.element_type, guard))
|
938
|
+
end
|
939
|
+
end
|
940
|
+
|
941
|
+
# @api public
|
942
|
+
#
|
943
|
+
class PStringType < PScalarType
|
944
|
+
attr_reader :size_type, :values
|
945
|
+
|
946
|
+
def initialize(size_type, values = EMPTY_ARRAY)
|
947
|
+
@size_type = size_type
|
948
|
+
@values = values.sort.freeze
|
949
|
+
end
|
950
|
+
|
951
|
+
def accept(visitor, guard)
|
952
|
+
super
|
953
|
+
@size_type.accept(visitor, guard) unless @size_type.nil?
|
954
|
+
end
|
955
|
+
|
956
|
+
def generalize
|
957
|
+
DEFAULT
|
958
|
+
end
|
959
|
+
|
960
|
+
def hash
|
961
|
+
@size_type.hash * 31 + @values.hash
|
962
|
+
end
|
963
|
+
|
964
|
+
def iterable?(guard = nil)
|
965
|
+
true
|
966
|
+
end
|
967
|
+
|
968
|
+
def iterable_type(guard = nil)
|
969
|
+
ITERABLE_TYPE
|
970
|
+
end
|
971
|
+
|
972
|
+
def eql?(o)
|
973
|
+
self.class == o.class && @size_type == o.size_type && @values == o.values
|
974
|
+
end
|
975
|
+
|
976
|
+
def instance?(o)
|
977
|
+
# true if size compliant
|
978
|
+
if o.is_a?(String) && (@size_type.nil? || @size_type.instance?(o.size))
|
979
|
+
@values.empty? || @values.include?(o)
|
980
|
+
else
|
981
|
+
false
|
982
|
+
end
|
983
|
+
end
|
984
|
+
|
985
|
+
DEFAULT = PStringType.new(nil)
|
986
|
+
NON_EMPTY = PStringType.new(PIntegerType.new(1))
|
987
|
+
|
988
|
+
# Iterates over each character of the string
|
989
|
+
ITERABLE_TYPE = PIterableType.new(PStringType.new(PIntegerType.new(1,1)))
|
990
|
+
|
991
|
+
protected
|
992
|
+
|
993
|
+
# @api private
|
994
|
+
def _assignable?(o, guard)
|
995
|
+
if values.empty?
|
996
|
+
# A general string is assignable by any other string or pattern restricted string
|
997
|
+
# if the string has a size constraint it does not match since there is no reasonable way
|
998
|
+
# to compute the min/max length a pattern will match. For enum, it is possible to test that
|
999
|
+
# each enumerator value is within range
|
1000
|
+
case o
|
1001
|
+
when PStringType
|
1002
|
+
# true if size compliant
|
1003
|
+
(@size_type || PCollectionType::DEFAULT_SIZE).assignable?(
|
1004
|
+
o.size_type || PCollectionType::DEFAULT_SIZE, guard)
|
1005
|
+
|
1006
|
+
when PPatternType
|
1007
|
+
# true if size constraint is at least 0 to +Infinity (which is the same as the default)
|
1008
|
+
@size_type.nil? || @size_type.assignable?(PCollectionType::DEFAULT_SIZE, guard)
|
1009
|
+
|
1010
|
+
when PEnumType
|
747
1011
|
if o.values.empty?
|
748
|
-
#
|
749
|
-
#
|
750
|
-
@
|
1012
|
+
# enum represents all enums, and thus all strings, a sized constrained string can thus not
|
1013
|
+
# be assigned any enum (unless it is max size).
|
1014
|
+
@size_type.nil? || @size_type.assignable?(PCollectionType::DEFAULT_SIZE, guard)
|
751
1015
|
else
|
752
|
-
#
|
753
|
-
|
754
|
-
|
755
|
-
|
1016
|
+
# true if all enum values are within range
|
1017
|
+
orange = o.values.map(&:size).minmax
|
1018
|
+
srange = (@size_type || PCollectionType::DEFAULT_SIZE).range
|
1019
|
+
# If o min and max are within the range of t
|
1020
|
+
srange[0] <= orange[0] && srange[1] >= orange[1]
|
756
1021
|
end
|
757
|
-
when PPatternType
|
758
|
-
@patterns.empty?
|
759
1022
|
else
|
1023
|
+
# no other type matches string
|
760
1024
|
false
|
761
|
-
end
|
762
1025
|
end
|
1026
|
+
elsif o.is_a?(PStringType)
|
1027
|
+
# A specific string acts as a set of strings - must have exactly the same strings
|
1028
|
+
# In this case, size does not matter since the definition is very precise anyway
|
1029
|
+
values == o.values
|
1030
|
+
else
|
1031
|
+
# All others are false, since no other type describes the same set of specific strings
|
1032
|
+
false
|
763
1033
|
end
|
1034
|
+
end
|
1035
|
+
end
|
764
1036
|
|
765
|
-
|
766
|
-
|
767
|
-
|
1037
|
+
# @api public
|
1038
|
+
#
|
1039
|
+
class PRegexpType < PScalarType
|
1040
|
+
attr_reader :pattern
|
768
1041
|
|
769
|
-
|
770
|
-
|
771
|
-
|
1042
|
+
def initialize(pattern)
|
1043
|
+
@pattern = pattern
|
1044
|
+
end
|
772
1045
|
|
773
|
-
|
1046
|
+
def regexp
|
1047
|
+
@regexp ||= Regexp.new(@pattern || '')
|
1048
|
+
end
|
774
1049
|
|
775
|
-
|
1050
|
+
def hash
|
1051
|
+
@pattern.hash
|
1052
|
+
end
|
776
1053
|
|
777
|
-
|
778
|
-
|
779
|
-
|
780
|
-
|
781
|
-
|
1054
|
+
def eql?(o)
|
1055
|
+
self.class == o.class && @pattern == o.pattern
|
1056
|
+
end
|
1057
|
+
|
1058
|
+
DEFAULT = PRegexpType.new(nil)
|
1059
|
+
|
1060
|
+
protected
|
1061
|
+
|
1062
|
+
# @api private
|
1063
|
+
#
|
1064
|
+
def _assignable?(o, guard)
|
1065
|
+
o.is_a?(PRegexpType) && (@pattern.nil? || @pattern == o.pattern)
|
1066
|
+
end
|
1067
|
+
end
|
1068
|
+
|
1069
|
+
# Represents a subtype of String that narrows the string to those matching the patterns
|
1070
|
+
# If specified without a pattern it is basically the same as the String type.
|
1071
|
+
#
|
1072
|
+
# @api public
|
1073
|
+
#
|
1074
|
+
class PPatternType < PScalarType
|
1075
|
+
attr_reader :patterns
|
1076
|
+
|
1077
|
+
def initialize(patterns)
|
1078
|
+
@patterns = patterns.freeze
|
1079
|
+
end
|
1080
|
+
|
1081
|
+
def accept(visitor, guard)
|
1082
|
+
super
|
1083
|
+
@patterns.each { |p| p.accept(visitor, guard) }
|
1084
|
+
end
|
1085
|
+
|
1086
|
+
def hash
|
1087
|
+
@patterns.hash
|
1088
|
+
end
|
1089
|
+
|
1090
|
+
def eql?(o)
|
1091
|
+
self.class == o.class && @patterns.size == o.patterns.size && (@patterns - o.patterns).empty?
|
1092
|
+
end
|
1093
|
+
|
1094
|
+
DEFAULT = PPatternType.new(EMPTY_ARRAY)
|
1095
|
+
|
1096
|
+
protected
|
1097
|
+
|
1098
|
+
# @api private
|
1099
|
+
#
|
1100
|
+
def _assignable?(o, guard)
|
1101
|
+
return true if self == o
|
1102
|
+
case o
|
1103
|
+
when PStringType, PEnumType
|
1104
|
+
if o.values.empty?
|
1105
|
+
# Strings / Enums (unknown which ones) cannot all match a pattern, but if there is no pattern it is ok
|
1106
|
+
# (There should really always be a pattern, but better safe than sorry).
|
1107
|
+
@patterns.empty?
|
1108
|
+
else
|
1109
|
+
# all strings in String/Enum type must match one of the patterns in Pattern type,
|
1110
|
+
# or Pattern represents all Patterns == all Strings
|
1111
|
+
regexps = @patterns.map { |p| p.regexp }
|
1112
|
+
regexps.empty? || o.values.all? { |v| regexps.any? {|re| re.match(v) } }
|
1113
|
+
end
|
1114
|
+
when PPatternType
|
1115
|
+
@patterns.empty?
|
1116
|
+
else
|
1117
|
+
false
|
782
1118
|
end
|
1119
|
+
end
|
1120
|
+
end
|
783
1121
|
|
784
|
-
|
785
|
-
|
786
|
-
|
787
|
-
#
|
788
|
-
class PStructElement < TypedModelObject
|
789
|
-
attr_accessor :key_type, :value_type
|
1122
|
+
# @api public
|
1123
|
+
#
|
1124
|
+
class PBooleanType < PScalarType
|
790
1125
|
|
791
|
-
|
792
|
-
|
793
|
-
|
1126
|
+
def instance?(o)
|
1127
|
+
o == true || o == false
|
1128
|
+
end
|
794
1129
|
|
795
|
-
|
796
|
-
k = key_type
|
797
|
-
k = k.optional_type if k.is_a?(POptionalType)
|
798
|
-
k.values[0]
|
799
|
-
end
|
1130
|
+
DEFAULT = PBooleanType.new
|
800
1131
|
|
801
|
-
|
802
|
-
@key_type = key_type
|
803
|
-
@value_type = value_type
|
804
|
-
end
|
1132
|
+
protected
|
805
1133
|
|
806
|
-
|
807
|
-
|
808
|
-
|
1134
|
+
# @api private
|
1135
|
+
#
|
1136
|
+
def _assignable?(o, guard)
|
1137
|
+
o.is_a?(PBooleanType)
|
1138
|
+
end
|
1139
|
+
end
|
809
1140
|
|
810
|
-
|
811
|
-
|
812
|
-
|
1141
|
+
# @api public
|
1142
|
+
#
|
1143
|
+
# @api public
|
1144
|
+
#
|
1145
|
+
class PStructElement < TypedModelObject
|
1146
|
+
attr_accessor :key_type, :value_type
|
813
1147
|
|
814
|
-
|
815
|
-
|
816
|
-
|
1148
|
+
def accept(visitor, guard)
|
1149
|
+
@key_type.accept(visitor, guard)
|
1150
|
+
@value_type.accept(visitor, guard)
|
1151
|
+
end
|
1152
|
+
|
1153
|
+
def hash
|
1154
|
+
value_type.hash * 31 + key_type.hash
|
1155
|
+
end
|
1156
|
+
|
1157
|
+
def name
|
1158
|
+
k = key_type
|
1159
|
+
k = k.optional_type if k.is_a?(POptionalType)
|
1160
|
+
k.values[0]
|
1161
|
+
end
|
1162
|
+
|
1163
|
+
def initialize(key_type, value_type)
|
1164
|
+
@key_type = key_type
|
1165
|
+
@value_type = value_type
|
1166
|
+
end
|
1167
|
+
|
1168
|
+
def generalize
|
1169
|
+
gv_type = @value_type.generalize
|
1170
|
+
@value_type.equal?(gv_type) ? self : PStructElement.new(@key_type, gv_type)
|
1171
|
+
end
|
1172
|
+
|
1173
|
+
def normalize(guard = nil)
|
1174
|
+
nv_type = @value_type.normalize(guard)
|
1175
|
+
@value_type.equal?(nv_type) ? self : PStructElement.new(@key_type, nv_type)
|
1176
|
+
end
|
1177
|
+
|
1178
|
+
def <=>(o)
|
1179
|
+
self.name <=> o.name
|
1180
|
+
end
|
1181
|
+
|
1182
|
+
def eql?(o)
|
1183
|
+
self.class == o.class && value_type == o.value_type && key_type == o.key_type
|
1184
|
+
end
|
1185
|
+
end
|
1186
|
+
|
1187
|
+
# @api public
|
1188
|
+
#
|
1189
|
+
class PStructType < PAnyType
|
1190
|
+
include Enumerable
|
1191
|
+
|
1192
|
+
def initialize(elements)
|
1193
|
+
@elements = elements.sort.freeze
|
1194
|
+
end
|
1195
|
+
|
1196
|
+
def accept(visitor, guard)
|
1197
|
+
super
|
1198
|
+
@elements.each { |elem| elem.accept(visitor, guard) }
|
1199
|
+
end
|
1200
|
+
|
1201
|
+
def each
|
1202
|
+
if block_given?
|
1203
|
+
elements.each { |elem| yield elem }
|
1204
|
+
else
|
1205
|
+
elements.to_enum
|
817
1206
|
end
|
1207
|
+
end
|
818
1208
|
|
819
|
-
|
820
|
-
|
821
|
-
|
822
|
-
|
1209
|
+
def generalize
|
1210
|
+
if @elements.empty?
|
1211
|
+
DEFAULT
|
1212
|
+
else
|
1213
|
+
alter_type_array(@elements, :generalize) { |altered| PStructType.new(altered) }
|
1214
|
+
end
|
1215
|
+
end
|
823
1216
|
|
824
|
-
|
825
|
-
|
826
|
-
|
1217
|
+
def normalize(guard = nil)
|
1218
|
+
if @elements.empty?
|
1219
|
+
DEFAULT
|
1220
|
+
else
|
1221
|
+
alter_type_array(@elements, :normalize, guard) { |altered| PStructType.new(altered) }
|
1222
|
+
end
|
1223
|
+
end
|
827
1224
|
|
828
|
-
|
829
|
-
|
830
|
-
|
831
|
-
else
|
832
|
-
elements.to_enum
|
833
|
-
end
|
834
|
-
end
|
1225
|
+
def hashed_elements
|
1226
|
+
@hashed ||= @elements.reduce({}) {|memo, e| memo[e.name] = e; memo }
|
1227
|
+
end
|
835
1228
|
|
836
|
-
|
837
|
-
|
838
|
-
|
1229
|
+
def hash
|
1230
|
+
@elements.hash
|
1231
|
+
end
|
839
1232
|
|
840
|
-
|
841
|
-
|
842
|
-
|
843
|
-
|
844
|
-
def hash
|
845
|
-
@elements.hash
|
846
|
-
end
|
1233
|
+
def iterable?(guard = nil)
|
1234
|
+
true
|
1235
|
+
end
|
847
1236
|
|
848
|
-
|
849
|
-
|
850
|
-
|
1237
|
+
def iterable_type(guard = nil)
|
1238
|
+
if self == DEFAULT
|
1239
|
+
PIterableType.new(PHashType::DEFAULT_KEY_PAIR_TUPLE)
|
1240
|
+
else
|
1241
|
+
tc = TypeCalculator.singleton
|
1242
|
+
PIterableType.new(
|
1243
|
+
PTupleType.new([
|
1244
|
+
tc.unwrap_single_variant(PVariantType.new(@elements.map {|se| se.key_type })),
|
1245
|
+
tc.unwrap_single_variant(PVariantType.new(@elements.map {|se| se.value_type }))],
|
1246
|
+
PHashType::KEY_PAIR_TUPLE_SIZE))
|
1247
|
+
end
|
1248
|
+
end
|
851
1249
|
|
852
|
-
|
853
|
-
|
854
|
-
|
1250
|
+
def eql?(o)
|
1251
|
+
self.class == o.class && @elements == o.elements
|
1252
|
+
end
|
855
1253
|
|
856
|
-
|
857
|
-
|
858
|
-
|
859
|
-
@elements.all? do |e|
|
860
|
-
key = e.name
|
861
|
-
v = o[key]
|
862
|
-
if v.nil? && !o.include?(key)
|
863
|
-
# Entry is missing. Only OK when key is optional
|
864
|
-
e.key_type.assignable?(PUndefType::DEFAULT)
|
865
|
-
else
|
866
|
-
matched += 1
|
867
|
-
e.value_type.instance?(v)
|
868
|
-
end
|
869
|
-
end && matched == o.size
|
870
|
-
end
|
1254
|
+
def elements
|
1255
|
+
@elements
|
1256
|
+
end
|
871
1257
|
|
872
|
-
|
873
|
-
|
874
|
-
|
875
|
-
|
876
|
-
|
877
|
-
|
878
|
-
|
879
|
-
|
880
|
-
|
881
|
-
|
882
|
-
|
883
|
-
|
884
|
-
|
885
|
-
|
886
|
-
|
887
|
-
|
888
|
-
|
889
|
-
|
890
|
-
|
891
|
-
|
892
|
-
|
893
|
-
|
894
|
-
|
895
|
-
|
896
|
-
|
897
|
-
|
898
|
-
|
899
|
-
|
900
|
-
|
901
|
-
size_o = o.size_type || collection_default_size_t
|
902
|
-
PIntegerType.new(required, elements.size).assignable?(size_o)
|
903
|
-
end
|
1258
|
+
def instance?(o)
|
1259
|
+
return false unless o.is_a?(Hash)
|
1260
|
+
matched = 0
|
1261
|
+
@elements.all? do |e|
|
1262
|
+
key = e.name
|
1263
|
+
v = o[key]
|
1264
|
+
if v.nil? && !o.include?(key)
|
1265
|
+
# Entry is missing. Only OK when key is optional
|
1266
|
+
e.key_type.assignable?(PUndefType::DEFAULT)
|
1267
|
+
else
|
1268
|
+
matched += 1
|
1269
|
+
e.value_type.instance?(v)
|
1270
|
+
end
|
1271
|
+
end && matched == o.size
|
1272
|
+
end
|
1273
|
+
|
1274
|
+
DEFAULT = PStructType.new(EMPTY_ARRAY)
|
1275
|
+
|
1276
|
+
protected
|
1277
|
+
|
1278
|
+
# @api private
|
1279
|
+
def _assignable?(o, guard)
|
1280
|
+
if o.is_a?(Types::PStructType)
|
1281
|
+
h2 = o.hashed_elements
|
1282
|
+
matched = 0
|
1283
|
+
elements.all? do |e1|
|
1284
|
+
e2 = h2[e1.name]
|
1285
|
+
if e2.nil?
|
1286
|
+
e1.key_type.assignable?(PUndefType::DEFAULT, guard)
|
904
1287
|
else
|
905
|
-
|
1288
|
+
matched += 1
|
1289
|
+
e1.key_type.assignable?(e2.key_type, guard) && e1.value_type.assignable?(e2.value_type, guard)
|
1290
|
+
end
|
1291
|
+
end && matched == h2.size
|
1292
|
+
elsif o.is_a?(Types::PHashType)
|
1293
|
+
required = 0
|
1294
|
+
required_elements_assignable = elements.all? do |e|
|
1295
|
+
if e.key_type.assignable?(PUndefType::DEFAULT)
|
1296
|
+
true
|
1297
|
+
else
|
1298
|
+
required += 1
|
1299
|
+
e.value_type.assignable?(o.element_type, guard)
|
906
1300
|
end
|
907
1301
|
end
|
1302
|
+
if required_elements_assignable
|
1303
|
+
size_o = o.size_type || collection_default_size_t
|
1304
|
+
PIntegerType.new(required, elements.size).assignable?(size_o, guard)
|
1305
|
+
end
|
1306
|
+
else
|
1307
|
+
false
|
908
1308
|
end
|
1309
|
+
end
|
1310
|
+
end
|
909
1311
|
|
910
|
-
|
911
|
-
|
912
|
-
|
913
|
-
|
1312
|
+
# @api public
|
1313
|
+
#
|
1314
|
+
class PTupleType < PAnyType
|
1315
|
+
include Enumerable
|
914
1316
|
|
915
|
-
|
916
|
-
|
917
|
-
|
918
|
-
|
1317
|
+
# If set, describes min and max required of the given types - if max > size of
|
1318
|
+
# types, the last type entry repeats
|
1319
|
+
#
|
1320
|
+
attr_reader :size_type
|
919
1321
|
|
920
|
-
|
1322
|
+
attr_reader :types
|
921
1323
|
|
922
|
-
|
923
|
-
|
924
|
-
|
925
|
-
|
926
|
-
|
1324
|
+
def accept(visitor, guard)
|
1325
|
+
super
|
1326
|
+
@size_type.accept(visitor, guard) unless @size_type.nil?
|
1327
|
+
@types.each { |elem| elem.accept(visitor, guard) }
|
1328
|
+
end
|
927
1329
|
|
928
|
-
|
929
|
-
|
930
|
-
|
931
|
-
|
932
|
-
|
933
|
-
# Can't pass a block to a callable that doesn't accept one
|
934
|
-
return false if param_block_t.nil?
|
935
|
-
|
936
|
-
# Check that the block is of the right tyṕe
|
937
|
-
return false unless param_block_t.assignable?(arg_block_t)
|
938
|
-
|
939
|
-
# Check other arguments
|
940
|
-
arg_count = arg_types.size - 1
|
941
|
-
params_size_t = params_tuple.size_type || PIntegerType.new(*params_tuple.size_range)
|
942
|
-
return false unless params_size_t.assignable?(PIntegerType.new(arg_count, arg_count))
|
943
|
-
|
944
|
-
ctypes = params_tuple.types
|
945
|
-
arg_count.times do |index|
|
946
|
-
return false unless (ctypes[index] || ctypes[-1]).assignable?(arg_types[index])
|
947
|
-
end
|
948
|
-
return true
|
949
|
-
end
|
1330
|
+
# @api private
|
1331
|
+
def callable_args?(callable_t, guard)
|
1332
|
+
unless size_type.nil?
|
1333
|
+
raise ArgumentError, 'Callable tuple may not have a size constraint when used as args'
|
1334
|
+
end
|
950
1335
|
|
951
|
-
|
952
|
-
|
953
|
-
|
1336
|
+
params_tuple = callable_t.param_types
|
1337
|
+
param_block_t = callable_t.block_type
|
1338
|
+
arg_types = @types
|
1339
|
+
arg_block_t = arg_types.last
|
1340
|
+
if arg_block_t.kind_of_callable?(true, guard)
|
1341
|
+
# Can't pass a block to a callable that doesn't accept one
|
1342
|
+
return false if param_block_t.nil?
|
954
1343
|
|
955
|
-
|
956
|
-
|
957
|
-
@size_type = size_type.nil? ? nil : size_type.to_size
|
958
|
-
end
|
1344
|
+
# Check that the block is of the right tyṕe
|
1345
|
+
return false unless param_block_t.assignable?(arg_block_t, guard)
|
959
1346
|
|
960
|
-
#
|
961
|
-
|
962
|
-
|
963
|
-
|
964
|
-
types.each { |x| yield x }
|
965
|
-
else
|
966
|
-
types.to_enum
|
967
|
-
end
|
968
|
-
end
|
1347
|
+
# Check other arguments
|
1348
|
+
arg_count = arg_types.size - 1
|
1349
|
+
params_size_t = params_tuple.size_type || PIntegerType.new(*params_tuple.size_range)
|
1350
|
+
return false unless params_size_t.assignable?(PIntegerType.new(arg_count, arg_count), guard)
|
969
1351
|
|
970
|
-
|
971
|
-
|
1352
|
+
ctypes = params_tuple.types
|
1353
|
+
arg_count.times do |index|
|
1354
|
+
return false unless (ctypes[index] || ctypes[-1]).assignable?(arg_types[index], guard)
|
972
1355
|
end
|
1356
|
+
return true
|
1357
|
+
end
|
973
1358
|
|
974
|
-
|
975
|
-
|
976
|
-
|
977
|
-
size_t = size_type || PIntegerType.new(*size_range)
|
1359
|
+
# Check that tuple is assignable and that the block (if declared) is optional
|
1360
|
+
params_tuple.assignable?(self, guard) && (param_block_t.nil? || param_block_t.assignable?(PUndefType::DEFAULT, guard))
|
1361
|
+
end
|
978
1362
|
|
979
|
-
|
980
|
-
|
981
|
-
|
982
|
-
|
983
|
-
true
|
984
|
-
end
|
1363
|
+
def initialize(types, size_type = nil)
|
1364
|
+
@types = types
|
1365
|
+
@size_type = size_type.nil? ? nil : size_type.to_size
|
1366
|
+
end
|
985
1367
|
|
986
|
-
|
987
|
-
|
988
|
-
|
989
|
-
|
990
|
-
|
991
|
-
|
992
|
-
|
993
|
-
|
994
|
-
|
1368
|
+
# Returns Enumerator for the types if no block is given, otherwise, calls the given
|
1369
|
+
# block with each of the types in this tuple
|
1370
|
+
def each
|
1371
|
+
if block_given?
|
1372
|
+
types.each { |x| yield x }
|
1373
|
+
else
|
1374
|
+
types.to_enum
|
1375
|
+
end
|
1376
|
+
end
|
995
1377
|
|
996
|
-
|
997
|
-
|
998
|
-
|
999
|
-
|
1000
|
-
|
1001
|
-
|
1002
|
-
|
1003
|
-
types_size = @types.size
|
1004
|
-
from, to = @size_type.range
|
1005
|
-
min = from - (types_size-1)
|
1006
|
-
min = min <= 0 ? 0 : min
|
1007
|
-
max = to - (types_size-1)
|
1008
|
-
[min, max]
|
1009
|
-
end
|
1378
|
+
def generalize
|
1379
|
+
if self == DEFAULT
|
1380
|
+
DEFAULT
|
1381
|
+
else
|
1382
|
+
alter_type_array(@types, :generalize) { |altered_types| PTupleType.new(altered_types, @size_type) }
|
1383
|
+
end
|
1384
|
+
end
|
1010
1385
|
|
1011
|
-
|
1012
|
-
|
1013
|
-
|
1386
|
+
def normalize(guard = nil)
|
1387
|
+
if self == DEFAULT
|
1388
|
+
DEFAULT
|
1389
|
+
else
|
1390
|
+
alter_type_array(@types, :normalize, guard) { |altered_types| PTupleType.new(altered_types, @size_type) }
|
1391
|
+
end
|
1392
|
+
end
|
1014
1393
|
|
1015
|
-
|
1016
|
-
|
1017
|
-
|
1394
|
+
def instance?(o)
|
1395
|
+
return false unless o.is_a?(Array)
|
1396
|
+
# compute the tuple's min/max size, and check if that size matches
|
1397
|
+
size_t = size_type || PIntegerType.new(*size_range)
|
1018
1398
|
|
1019
|
-
|
1020
|
-
|
1399
|
+
return false unless size_t.instance?(o.size)
|
1400
|
+
o.each_with_index do |element, index|
|
1401
|
+
return false unless (types[index] || types[-1]).instance?(element)
|
1402
|
+
end
|
1403
|
+
true
|
1404
|
+
end
|
1021
1405
|
|
1022
|
-
|
1406
|
+
def iterable?(guard = nil)
|
1407
|
+
true
|
1408
|
+
end
|
1023
1409
|
|
1024
|
-
|
1025
|
-
|
1026
|
-
|
1027
|
-
s_types = types
|
1028
|
-
return true if s_types.empty? && (o.is_a?(PArrayType))
|
1029
|
-
size_s = size_type || PIntegerType.new(*size_range)
|
1410
|
+
def iterable_type(guard = nil)
|
1411
|
+
PIterableType.new(TypeCalculator.singleton.unwrap_single_variant(PVariantType.new(types)))
|
1412
|
+
end
|
1030
1413
|
|
1031
|
-
|
1032
|
-
|
1414
|
+
# Returns the number of elements accepted [min, max] in the tuple
|
1415
|
+
def size_range
|
1416
|
+
if @size_type.nil?
|
1417
|
+
types_size = @types.size
|
1418
|
+
[types_size, types_size]
|
1419
|
+
else
|
1420
|
+
@size_type.range
|
1421
|
+
end
|
1422
|
+
end
|
1033
1423
|
|
1034
|
-
|
1035
|
-
|
1036
|
-
|
1037
|
-
|
1038
|
-
|
1039
|
-
|
1040
|
-
return true
|
1041
|
-
else
|
1042
|
-
return false
|
1043
|
-
end
|
1044
|
-
elsif o.is_a?(PArrayType)
|
1045
|
-
o_entry = o.element_type
|
1046
|
-
# Array of anything can not be assigned (unless tuple is tuple of anything) - this case
|
1047
|
-
# was handled at the top of this method.
|
1048
|
-
#
|
1049
|
-
return false if o_entry.nil?
|
1050
|
-
size_o = o.size_type || PCollectionType::DEFAULT_SIZE
|
1051
|
-
return false unless size_s.assignable?(size_o)
|
1052
|
-
[s_types.size, size_o.range[1]].min.times { |index| return false unless (s_types[index] || s_types[-1]).assignable?(o_entry) }
|
1053
|
-
true
|
1054
|
-
else
|
1055
|
-
false
|
1056
|
-
end
|
1057
|
-
end
|
1424
|
+
# Returns the number of accepted occurrences [min, max] of the last type in the tuple
|
1425
|
+
# The defaults is [1,1]
|
1426
|
+
#
|
1427
|
+
def repeat_last_range
|
1428
|
+
if @size_type.nil?
|
1429
|
+
return [1, 1]
|
1058
1430
|
end
|
1431
|
+
types_size = @types.size
|
1432
|
+
from, to = @size_type.range
|
1433
|
+
min = from - (types_size-1)
|
1434
|
+
min = min <= 0 ? 0 : min
|
1435
|
+
max = to - (types_size-1)
|
1436
|
+
[min, max]
|
1437
|
+
end
|
1059
1438
|
|
1060
|
-
|
1061
|
-
|
1062
|
-
|
1063
|
-
# Types of parameters as a Tuple with required/optional count, or an Integer with min (required), max count
|
1064
|
-
# @return [PTupleType] the tuple representing the parameter types
|
1065
|
-
attr_reader :param_types
|
1066
|
-
|
1067
|
-
# Although being an abstract type reference, only Callable, or all Callables wrapped in
|
1068
|
-
# Optional or Variant are supported
|
1069
|
-
# If not set, the meaning is that block is not supported.
|
1070
|
-
# @return [PAnyType|nil] the block type
|
1071
|
-
attr_reader :block_type
|
1072
|
-
|
1073
|
-
# @param param_types [PTupleType]
|
1074
|
-
# @param block_type [PAnyType|nil]
|
1075
|
-
def initialize(param_types, block_type = nil)
|
1076
|
-
@param_types = param_types
|
1077
|
-
@block_type = block_type
|
1078
|
-
end
|
1439
|
+
def hash
|
1440
|
+
@size_type.hash * 31 + @types.hash
|
1441
|
+
end
|
1079
1442
|
|
1080
|
-
|
1081
|
-
|
1082
|
-
|
1083
|
-
block_t = @block_type.nil? ? nil : @block_type.generalize
|
1084
|
-
PCallableType.new(params_t, block_t)
|
1085
|
-
end
|
1443
|
+
def eql?(o)
|
1444
|
+
self.class == o.class && @types == o.types && @size_type == o.size_type
|
1445
|
+
end
|
1086
1446
|
|
1087
|
-
|
1088
|
-
|
1089
|
-
end
|
1447
|
+
DATA = PTupleType.new([PDataType::DEFAULT], PCollectionType::DEFAULT_SIZE)
|
1448
|
+
DEFAULT = PTupleType.new(EMPTY_ARRAY)
|
1090
1449
|
|
1091
|
-
|
1092
|
-
def callable_args?(required_callable_t)
|
1093
|
-
# If the required callable is euqal or more specific than self, self is acceptable arguments
|
1094
|
-
required_callable_t.assignable?(self)
|
1095
|
-
end
|
1450
|
+
protected
|
1096
1451
|
|
1097
|
-
|
1098
|
-
|
1099
|
-
|
1452
|
+
# @api private
|
1453
|
+
def _assignable?(o, guard)
|
1454
|
+
return true if self == o
|
1455
|
+
s_types = types
|
1456
|
+
return true if s_types.empty? && (o.is_a?(PArrayType))
|
1457
|
+
size_s = size_type || PIntegerType.new(*size_range)
|
1100
1458
|
|
1101
|
-
|
1102
|
-
|
1103
|
-
@param_types.nil? ? nil : @param_types.size_range
|
1104
|
-
end
|
1459
|
+
if o.is_a?(PTupleType)
|
1460
|
+
size_o = o.size_type || PIntegerType.new(*o.size_range)
|
1105
1461
|
|
1106
|
-
#
|
1462
|
+
# not assignable if the number of types in o is outside number of types in t1
|
1463
|
+
if size_s.assignable?(size_o, guard)
|
1464
|
+
o_types = o.types
|
1465
|
+
o_types.size.times do |index|
|
1466
|
+
return false unless (s_types[index] || s_types[-1]).assignable?(o_types[index], guard)
|
1467
|
+
end
|
1468
|
+
return true
|
1469
|
+
else
|
1470
|
+
return false
|
1471
|
+
end
|
1472
|
+
elsif o.is_a?(PArrayType)
|
1473
|
+
o_entry = o.element_type
|
1474
|
+
# Array of anything can not be assigned (unless tuple is tuple of anything) - this case
|
1475
|
+
# was handled at the top of this method.
|
1107
1476
|
#
|
1108
|
-
|
1109
|
-
|
1110
|
-
|
1477
|
+
return false if o_entry.nil?
|
1478
|
+
size_o = o.size_type || PCollectionType::DEFAULT_SIZE
|
1479
|
+
return false unless size_s.assignable?(size_o, guard)
|
1480
|
+
[s_types.size, size_o.range[1]].min.times { |index| return false unless (s_types[index] || s_types[-1]).assignable?(o_entry, guard) }
|
1481
|
+
true
|
1482
|
+
else
|
1483
|
+
false
|
1484
|
+
end
|
1485
|
+
end
|
1486
|
+
end
|
1111
1487
|
|
1112
|
-
|
1113
|
-
|
1114
|
-
|
1115
|
-
|
1116
|
-
|
1117
|
-
|
1118
|
-
|
1119
|
-
|
1120
|
-
|
1121
|
-
|
1122
|
-
|
1123
|
-
|
1488
|
+
# @api public
|
1489
|
+
#
|
1490
|
+
class PCallableType < PAnyType
|
1491
|
+
# Types of parameters as a Tuple with required/optional count, or an Integer with min (required), max count
|
1492
|
+
# @return [PTupleType] the tuple representing the parameter types
|
1493
|
+
attr_reader :param_types
|
1494
|
+
|
1495
|
+
# Although being an abstract type reference, only Callable, or all Callables wrapped in
|
1496
|
+
# Optional or Variant are supported
|
1497
|
+
# If not set, the meaning is that block is not supported.
|
1498
|
+
# @return [PAnyType|nil] the block type
|
1499
|
+
attr_reader :block_type
|
1500
|
+
|
1501
|
+
# @param param_types [PTupleType]
|
1502
|
+
# @param block_type [PAnyType|nil]
|
1503
|
+
def initialize(param_types, block_type = nil)
|
1504
|
+
@param_types = param_types
|
1505
|
+
@block_type = block_type
|
1506
|
+
end
|
1124
1507
|
|
1125
|
-
|
1126
|
-
|
1127
|
-
|
1508
|
+
def accept(visitor, guard)
|
1509
|
+
super
|
1510
|
+
@param_types.accept(visitor, guard) unless @param_types.nil?
|
1511
|
+
@block_type.accept(visitor, guard) unless @block_type.nil?
|
1512
|
+
end
|
1128
1513
|
|
1129
|
-
|
1130
|
-
|
1131
|
-
|
1514
|
+
def generalize
|
1515
|
+
if self == DEFAULT
|
1516
|
+
DEFAULT
|
1517
|
+
else
|
1518
|
+
params_t = @param_types.nil? ? nil : @param_types.generalize
|
1519
|
+
block_t = @block_type.nil? ? nil : @block_type.generalize
|
1520
|
+
@param_types.equal?(params_t) && @block_type.equal?(block_t) ? self : PCallableType.new(params_t, block_t)
|
1521
|
+
end
|
1522
|
+
end
|
1523
|
+
|
1524
|
+
def normalize(guard = nil)
|
1525
|
+
if self == DEFAULT
|
1526
|
+
DEFAULT
|
1527
|
+
else
|
1528
|
+
params_t = @param_types.nil? ? nil : @param_types.normalize(guard)
|
1529
|
+
block_t = @block_type.nil? ? nil : @block_type.normalize(guard)
|
1530
|
+
@param_types.equal?(params_t) && @block_type.equal?(block_t) ? self : PCallableType.new(params_t, block_t)
|
1531
|
+
end
|
1532
|
+
end
|
1533
|
+
|
1534
|
+
def instance?(o)
|
1535
|
+
assignable?(TypeCalculator.infer(o))
|
1536
|
+
end
|
1132
1537
|
|
1133
|
-
|
1538
|
+
# @api private
|
1539
|
+
def callable_args?(required_callable_t, guard)
|
1540
|
+
# If the required callable is euqal or more specific than self, self is acceptable arguments
|
1541
|
+
required_callable_t.assignable?(self, guard)
|
1542
|
+
end
|
1134
1543
|
|
1135
|
-
|
1544
|
+
def kind_of_callable?(optional=true, guard = nil)
|
1545
|
+
true
|
1546
|
+
end
|
1136
1547
|
|
1137
|
-
|
1138
|
-
|
1139
|
-
|
1140
|
-
|
1141
|
-
return true if @param_types.nil?
|
1548
|
+
# Returns the number of accepted arguments [min, max]
|
1549
|
+
def size_range
|
1550
|
+
@param_types.nil? ? nil : @param_types.size_range
|
1551
|
+
end
|
1142
1552
|
|
1143
|
-
|
1144
|
-
|
1145
|
-
|
1146
|
-
|
1147
|
-
|
1148
|
-
|
1149
|
-
|
1150
|
-
|
1151
|
-
|
1553
|
+
# Returns the number of accepted arguments for the last parameter type [min, max]
|
1554
|
+
#
|
1555
|
+
def last_range
|
1556
|
+
@param_types.nil? ? nil : @param_types.repeat_last_range
|
1557
|
+
end
|
1558
|
+
|
1559
|
+
# Range [0,0], [0,1], or [1,1] for the block
|
1560
|
+
#
|
1561
|
+
def block_range
|
1562
|
+
case block_type
|
1563
|
+
when POptionalType
|
1564
|
+
[0,1]
|
1565
|
+
when PVariantType, PCallableType
|
1566
|
+
[1,1]
|
1567
|
+
else
|
1568
|
+
[0,0]
|
1152
1569
|
end
|
1570
|
+
end
|
1153
1571
|
|
1154
|
-
|
1155
|
-
|
1156
|
-
|
1572
|
+
def hash
|
1573
|
+
@param_types.hash * 31 + @block_type.hash
|
1574
|
+
end
|
1157
1575
|
|
1158
|
-
|
1159
|
-
|
1160
|
-
|
1161
|
-
block_t = callable.block_type
|
1162
|
-
# does not support calling with a block, but have to check that callable is ok with missing block
|
1163
|
-
(param_t.nil? || param_t.assignable?(self)) && (block_t.nil? || block_t.assignable(PUndefType::DEFAULT))
|
1164
|
-
end
|
1576
|
+
def eql?(o)
|
1577
|
+
self.class == o.class && @param_types == o.param_types && @block_type == o.block_type
|
1578
|
+
end
|
1165
1579
|
|
1166
|
-
|
1167
|
-
if self == DEFAULT
|
1168
|
-
self
|
1169
|
-
else
|
1170
|
-
PArrayType.new(element_type.nil? ? nil : element_type.generalize)
|
1171
|
-
end
|
1172
|
-
end
|
1580
|
+
DEFAULT = PCallableType.new(nil)
|
1173
1581
|
|
1174
|
-
|
1175
|
-
return false unless o.is_a?(Array)
|
1176
|
-
element_t = element_type
|
1177
|
-
return false unless element_t.nil? || o.all? {|element| element_t.instance?(element) }
|
1178
|
-
size_t = size_type
|
1179
|
-
size_t.nil? || size_t.instance?(o.size)
|
1180
|
-
end
|
1582
|
+
protected
|
1181
1583
|
|
1182
|
-
|
1183
|
-
|
1184
|
-
|
1185
|
-
|
1186
|
-
|
1187
|
-
|
1188
|
-
|
1189
|
-
|
1190
|
-
|
1191
|
-
|
1192
|
-
|
1193
|
-
|
1194
|
-
|
1195
|
-
|
1196
|
-
|
1197
|
-
|
1198
|
-
|
1199
|
-
|
1200
|
-
|
1201
|
-
|
1202
|
-
|
1203
|
-
|
1204
|
-
|
1205
|
-
|
1206
|
-
|
1207
|
-
|
1208
|
-
|
1209
|
-
|
1210
|
-
|
1211
|
-
|
1212
|
-
|
1213
|
-
|
1214
|
-
|
1215
|
-
|
1216
|
-
end
|
1217
|
-
# ... and so must the last, possibly optional (ranged) type
|
1218
|
-
s_entry.assignable?(o_ranged)
|
1219
|
-
elsif o.is_a?(PArrayType)
|
1220
|
-
super && (s_entry.nil? || s_entry.assignable?(o.element_type))
|
1221
|
-
else
|
1222
|
-
false
|
1223
|
-
end
|
1224
|
-
end
|
1584
|
+
# @api private
|
1585
|
+
def _assignable?(o, guard)
|
1586
|
+
return false unless o.is_a?(PCallableType)
|
1587
|
+
# nil param_types means, any other Callable is assignable
|
1588
|
+
return true if @param_types.nil?
|
1589
|
+
|
1590
|
+
# NOTE: these tests are made in reverse as it is calling the callable that is constrained
|
1591
|
+
# (it's lower bound), not its upper bound
|
1592
|
+
return false unless o.param_types.assignable?(@param_types, guard)
|
1593
|
+
# names are ignored, they are just information
|
1594
|
+
# Blocks must be compatible
|
1595
|
+
this_block_t = @block_type || PUndefType::DEFAULT
|
1596
|
+
that_block_t = o.block_type || PUndefType::DEFAULT
|
1597
|
+
that_block_t.assignable?(this_block_t, guard)
|
1598
|
+
end
|
1599
|
+
end
|
1600
|
+
|
1601
|
+
# @api public
|
1602
|
+
#
|
1603
|
+
class PArrayType < PCollectionType
|
1604
|
+
|
1605
|
+
# @api private
|
1606
|
+
def callable_args?(callable, guard = nil)
|
1607
|
+
param_t = callable.param_types
|
1608
|
+
block_t = callable.block_type
|
1609
|
+
# does not support calling with a block, but have to check that callable is ok with missing block
|
1610
|
+
(param_t.nil? || param_t.assignable?(self, guard)) && (block_t.nil? || block_t.assignable(PUndefType::DEFAULT, guard))
|
1611
|
+
end
|
1612
|
+
|
1613
|
+
def generalize
|
1614
|
+
if self == DATA
|
1615
|
+
self
|
1616
|
+
else
|
1617
|
+
super
|
1225
1618
|
end
|
1619
|
+
end
|
1226
1620
|
|
1227
|
-
|
1228
|
-
|
1229
|
-
|
1230
|
-
|
1621
|
+
def normalize(guard = nil)
|
1622
|
+
if self == DATA
|
1623
|
+
self
|
1624
|
+
else
|
1625
|
+
super
|
1626
|
+
end
|
1627
|
+
end
|
1231
1628
|
|
1232
|
-
|
1233
|
-
|
1234
|
-
|
1235
|
-
|
1629
|
+
def instance?(o)
|
1630
|
+
return false unless o.is_a?(Array)
|
1631
|
+
element_t = element_type
|
1632
|
+
return false unless element_t.nil? || o.all? {|element| element_t.instance?(element) }
|
1633
|
+
size_t = size_type
|
1634
|
+
size_t.nil? || size_t.instance?(o.size)
|
1635
|
+
end
|
1236
1636
|
|
1237
|
-
|
1238
|
-
|
1239
|
-
|
1240
|
-
|
1241
|
-
|
1242
|
-
|
1243
|
-
|
1244
|
-
|
1245
|
-
|
1246
|
-
|
1247
|
-
|
1637
|
+
DATA = PArrayType.new(PDataType::DEFAULT, DEFAULT_SIZE)
|
1638
|
+
DEFAULT = PArrayType.new(nil)
|
1639
|
+
EMPTY = PArrayType.new(PUnitType::DEFAULT, ZERO_SIZE)
|
1640
|
+
|
1641
|
+
protected
|
1642
|
+
|
1643
|
+
# Array is assignable if o is an Array and o's element type is assignable, or if o is a Tuple
|
1644
|
+
# @api private
|
1645
|
+
def _assignable?(o, guard)
|
1646
|
+
s_entry = element_type
|
1647
|
+
if o.is_a?(PTupleType)
|
1648
|
+
# If s_entry is nil, this Array type has no opinion on element types. Therefore any
|
1649
|
+
# tuple can be assigned.
|
1650
|
+
return true if s_entry.nil?
|
1651
|
+
|
1652
|
+
return false unless o.types.all? {|o_element_t| s_entry.assignable?(o_element_t, guard) }
|
1653
|
+
o_regular = o.types[0..-2]
|
1654
|
+
o_ranged = o.types[-1]
|
1655
|
+
o_from, o_to = type_to_range(o.size_type)
|
1656
|
+
o_required = o_regular.size + o_from
|
1657
|
+
|
1658
|
+
# array type may be size constrained
|
1659
|
+
size_s = size_type || DEFAULT_SIZE
|
1660
|
+
min, max = size_s.range
|
1661
|
+
# Tuple with fewer min entries can not be assigned
|
1662
|
+
return false if o_required < min
|
1663
|
+
# Tuple with more optionally available entries can not be assigned
|
1664
|
+
return false if o_regular.size + o_to > max
|
1665
|
+
# each tuple type must be assignable to the element type
|
1666
|
+
o_required.times do |index|
|
1667
|
+
o_entry = tuple_entry_at(o, o_from, o_to, index)
|
1668
|
+
return false unless s_entry.assignable?(o_entry, guard)
|
1669
|
+
end
|
1670
|
+
# ... and so must the last, possibly optional (ranged) type
|
1671
|
+
s_entry.assignable?(o_ranged)
|
1672
|
+
elsif o.is_a?(PArrayType)
|
1673
|
+
super && (s_entry.nil? || s_entry.assignable?(o.element_type, guard))
|
1674
|
+
else
|
1675
|
+
false
|
1676
|
+
end
|
1677
|
+
end
|
1678
|
+
end
|
1248
1679
|
|
1249
|
-
|
1250
|
-
|
1251
|
-
|
1680
|
+
# @api public
|
1681
|
+
#
|
1682
|
+
class PHashType < PCollectionType
|
1683
|
+
attr_accessor :key_type
|
1252
1684
|
|
1253
|
-
|
1254
|
-
|
1255
|
-
|
1256
|
-
|
1257
|
-
if (key_t.nil? || o.keys.all? {|key| key_t.instance?(key) }) &&
|
1258
|
-
(element_t.nil? || o.values.all? {|value| element_t.instance?(value) })
|
1259
|
-
size_t = size_type
|
1260
|
-
size_t.nil? || size_t.instance?(o.size)
|
1261
|
-
else
|
1262
|
-
false
|
1263
|
-
end
|
1264
|
-
end
|
1685
|
+
def initialize(key_type, value_type, size_type = nil)
|
1686
|
+
super(value_type, size_type)
|
1687
|
+
@key_type = key_type
|
1688
|
+
end
|
1265
1689
|
|
1266
|
-
|
1267
|
-
|
1268
|
-
|
1690
|
+
def accept(visitor, guard)
|
1691
|
+
super
|
1692
|
+
@key_type.accept(visitor, guard) unless @key_type.nil?
|
1693
|
+
end
|
1269
1694
|
|
1270
|
-
|
1271
|
-
|
1272
|
-
|
1695
|
+
def generalize
|
1696
|
+
if self == DEFAULT || self == DATA || self == EMPTY
|
1697
|
+
self
|
1698
|
+
else
|
1699
|
+
key_t = @key_type
|
1700
|
+
key_t = key_t.generalize unless key_t.nil?
|
1701
|
+
value_t = @element_type
|
1702
|
+
value_t = value_t.generalize unless value_t.nil?
|
1703
|
+
@size_type.nil? && @key_type.equal?(key_t) && @element_type.equal?(value_t) ? self : PHashType.new(key_t, value_t, nil)
|
1704
|
+
end
|
1705
|
+
end
|
1706
|
+
|
1707
|
+
def normalize(guard = nil)
|
1708
|
+
if self == DEFAULT || self == DATA || self == EMPTY
|
1709
|
+
self
|
1710
|
+
else
|
1711
|
+
key_t = @key_type
|
1712
|
+
key_t = key_t.normalize(guard) unless key_t.nil?
|
1713
|
+
value_t = @element_type
|
1714
|
+
value_t = value_t.normalize(guard) unless value_t.nil?
|
1715
|
+
@size_type.nil? && @key_type.equal?(key_t) && @element_type.equal?(value_t) ? self : PHashType.new(key_t, value_t, nil)
|
1716
|
+
end
|
1717
|
+
end
|
1718
|
+
|
1719
|
+
def hash
|
1720
|
+
@key_type.hash * 31 + super
|
1721
|
+
end
|
1722
|
+
|
1723
|
+
def instance?(o)
|
1724
|
+
return false unless o.is_a?(Hash)
|
1725
|
+
key_t = key_type
|
1726
|
+
element_t = element_type
|
1727
|
+
if (key_t.nil? || o.keys.all? {|key| key_t.instance?(key) }) &&
|
1728
|
+
(element_t.nil? || o.values.all? {|value| element_t.instance?(value) })
|
1729
|
+
size_t = size_type
|
1730
|
+
size_t.nil? || size_t.instance?(o.size)
|
1731
|
+
else
|
1732
|
+
false
|
1733
|
+
end
|
1734
|
+
end
|
1735
|
+
|
1736
|
+
def iterable?(guard = nil)
|
1737
|
+
true
|
1738
|
+
end
|
1739
|
+
|
1740
|
+
def iterable_type(guard = nil)
|
1741
|
+
if self == DEFAULT || self == EMPTY
|
1742
|
+
PIterableType.new(DEFAULT_KEY_PAIR_TUPLE)
|
1743
|
+
else
|
1744
|
+
PIterableType.new(PTupleType.new([@key_type, @element_type], KEY_PAIR_TUPLE_SIZE))
|
1745
|
+
end
|
1746
|
+
end
|
1747
|
+
|
1748
|
+
def eql?(o)
|
1749
|
+
super && @key_type == o.key_type
|
1750
|
+
end
|
1751
|
+
|
1752
|
+
def is_the_empty_hash?
|
1753
|
+
self == EMPTY
|
1754
|
+
end
|
1755
|
+
|
1756
|
+
DEFAULT = PHashType.new(nil, nil)
|
1757
|
+
KEY_PAIR_TUPLE_SIZE = PIntegerType.new(2,2)
|
1758
|
+
DEFAULT_KEY_PAIR_TUPLE = PTupleType.new([PUnitType::DEFAULT, PUnitType::DEFAULT], KEY_PAIR_TUPLE_SIZE)
|
1759
|
+
DATA = PHashType.new(PScalarType::DEFAULT, PDataType::DEFAULT, DEFAULT_SIZE)
|
1760
|
+
EMPTY = PHashType.new(PUnitType::DEFAULT, PUnitType::DEFAULT, PIntegerType.new(0, 0))
|
1761
|
+
|
1762
|
+
protected
|
1763
|
+
|
1764
|
+
# Hash is assignable if o is a Hash and o's key and element types are assignable
|
1765
|
+
# @api private
|
1766
|
+
def _assignable?(o, guard)
|
1767
|
+
case o
|
1768
|
+
when PHashType
|
1769
|
+
size_s = size_type
|
1770
|
+
return true if (size_s.nil? || size_s.from == 0) && o.is_the_empty_hash?
|
1771
|
+
return false unless (key_type.nil? || key_type.assignable?(o.key_type, guard)) && (element_type.nil? || element_type.assignable?(o.element_type, guard))
|
1772
|
+
super
|
1773
|
+
when PStructType
|
1774
|
+
# hash must accept String as key type
|
1775
|
+
# hash must accept all value types
|
1776
|
+
# hash must accept the size of the struct
|
1777
|
+
o_elements = o.elements
|
1778
|
+
(size_type || DEFAULT_SIZE).instance?(o_elements.size) &&
|
1779
|
+
o_elements.all? {|e| (key_type.nil? || key_type.instance?(e.name)) && (element_type.nil? || element_type.assignable?(e.value_type, guard)) }
|
1780
|
+
else
|
1781
|
+
false
|
1782
|
+
end
|
1783
|
+
end
|
1784
|
+
end
|
1785
|
+
|
1786
|
+
# A flexible type describing an any? of other types
|
1787
|
+
# @api public
|
1788
|
+
#
|
1789
|
+
class PVariantType < PAnyType
|
1790
|
+
include Enumerable
|
1273
1791
|
|
1274
|
-
|
1275
|
-
|
1276
|
-
|
1277
|
-
|
1278
|
-
|
1279
|
-
|
1280
|
-
|
1281
|
-
|
1282
|
-
|
1283
|
-
|
1284
|
-
|
1285
|
-
|
1286
|
-
|
1287
|
-
|
1288
|
-
|
1289
|
-
|
1290
|
-
|
1291
|
-
|
1292
|
-
|
1293
|
-
|
1294
|
-
|
1295
|
-
|
1792
|
+
attr_reader :types
|
1793
|
+
|
1794
|
+
# @param types [Array[PAnyType]] the variants
|
1795
|
+
def initialize(types)
|
1796
|
+
@types = types.uniq.freeze
|
1797
|
+
end
|
1798
|
+
|
1799
|
+
def accept(visitor, guard)
|
1800
|
+
super
|
1801
|
+
@types.each { |t| t.accept(visitor, guard) }
|
1802
|
+
end
|
1803
|
+
|
1804
|
+
def each
|
1805
|
+
if block_given?
|
1806
|
+
types.each { |t| yield t }
|
1807
|
+
else
|
1808
|
+
types.to_enum
|
1809
|
+
end
|
1810
|
+
end
|
1811
|
+
|
1812
|
+
def generalize
|
1813
|
+
if self == DEFAULT || self == DATA
|
1814
|
+
self
|
1815
|
+
else
|
1816
|
+
alter_type_array(@types, :generalize) { |altered| PVariantType.new(mod_types) }
|
1817
|
+
end
|
1818
|
+
end
|
1819
|
+
|
1820
|
+
def normalize(guard = nil)
|
1821
|
+
if self == DEFAULT || self == DATA || @types.empty?
|
1822
|
+
self
|
1823
|
+
else
|
1824
|
+
# Normalize all contained types
|
1825
|
+
modified = false
|
1826
|
+
types = alter_type_array(@types, :normalize, guard)
|
1827
|
+
if types == self
|
1828
|
+
types = @types
|
1829
|
+
else
|
1830
|
+
modified = true
|
1831
|
+
end
|
1832
|
+
|
1833
|
+
if types.size == 1
|
1834
|
+
types[0]
|
1835
|
+
elsif types.any? { |t| t.is_a?(PUndefType) }
|
1836
|
+
# Undef entry present. Use an OptionalType with a normalized Variant of all types that are not Undef
|
1837
|
+
POptionalType.new(PVariantType.new(types.reject { |ot| ot.is_a?(PUndefType) }).normalize(guard)).normalize(guard)
|
1838
|
+
else
|
1839
|
+
# Merge all variants into this one
|
1840
|
+
types = types.map do |t|
|
1841
|
+
if t.is_a?(PVariantType)
|
1842
|
+
modified = true
|
1843
|
+
t.types
|
1296
1844
|
else
|
1297
|
-
|
1845
|
+
t
|
1846
|
+
end
|
1847
|
+
end
|
1848
|
+
types.flatten! if modified
|
1849
|
+
size_before_merge = types.size
|
1850
|
+
|
1851
|
+
types = swap_not_undefs(types)
|
1852
|
+
types = swap_optionals(types)
|
1853
|
+
types = merge_enums(types)
|
1854
|
+
types = merge_patterns(types)
|
1855
|
+
types = merge_int_ranges(types)
|
1856
|
+
types = merge_float_ranges(types)
|
1857
|
+
|
1858
|
+
if types.size == 1
|
1859
|
+
types[0]
|
1860
|
+
else
|
1861
|
+
modified || types.size != size_before_merge ? PVariantType.new(types) : self
|
1298
1862
|
end
|
1299
1863
|
end
|
1300
1864
|
end
|
1865
|
+
end
|
1301
1866
|
|
1302
|
-
|
1303
|
-
|
1304
|
-
|
1305
|
-
class PVariantType < PAnyType
|
1306
|
-
include Enumerable
|
1867
|
+
def hash
|
1868
|
+
@types.hash
|
1869
|
+
end
|
1307
1870
|
|
1308
|
-
|
1871
|
+
def instance?(o)
|
1872
|
+
# instance of variant if o is instance? of any of variant's types
|
1873
|
+
@types.any? { |type| type.instance?(o) }
|
1874
|
+
end
|
1309
1875
|
|
1310
|
-
|
1311
|
-
|
1312
|
-
|
1876
|
+
def kind_of_callable?(optional = true, guard = nil)
|
1877
|
+
@types.all? { |type| type.kind_of_callable?(optional, guard) }
|
1878
|
+
end
|
1879
|
+
|
1880
|
+
def resolved?
|
1881
|
+
@types.all? { |type| type.resolved? }
|
1882
|
+
end
|
1883
|
+
|
1884
|
+
def eql?(o)
|
1885
|
+
o = DATA if o.is_a?(PDataType)
|
1886
|
+
self.class == o.class && @types.size == o.types.size && (@types - o.types).empty?
|
1887
|
+
end
|
1313
1888
|
|
1314
|
-
|
1315
|
-
|
1316
|
-
|
1889
|
+
# Variant compatible with the Data type.
|
1890
|
+
DATA = PVariantType.new([PHashType::DATA, PArrayType::DATA, PScalarType::DEFAULT, PUndefType::DEFAULT, PTupleType::DATA])
|
1891
|
+
|
1892
|
+
DEFAULT = PVariantType.new(EMPTY_ARRAY)
|
1893
|
+
|
1894
|
+
protected
|
1895
|
+
|
1896
|
+
# @api private
|
1897
|
+
def _assignable?(o, guard)
|
1898
|
+
# Data is a specific variant
|
1899
|
+
o = DATA if o.is_a?(PDataType)
|
1900
|
+
if o.is_a?(PVariantType)
|
1901
|
+
# A variant is assignable if all of its options are assignable to one of this type's options
|
1902
|
+
return true if self == o
|
1903
|
+
o.types.all? do |other|
|
1904
|
+
# if the other is a Variant, all of its options, but be assignable to one of this type's options
|
1905
|
+
other = other.is_a?(PDataType) ? DATA : other
|
1906
|
+
if other.is_a?(PVariantType)
|
1907
|
+
assignable?(other, guard)
|
1317
1908
|
else
|
1318
|
-
types.
|
1909
|
+
types.any? {|option_t| option_t.assignable?(other, guard) }
|
1319
1910
|
end
|
1320
1911
|
end
|
1912
|
+
else
|
1913
|
+
# A variant is assignable if o is assignable to any of its types
|
1914
|
+
types.any? { |option_t| option_t.assignable?(o, guard) }
|
1915
|
+
end
|
1916
|
+
end
|
1321
1917
|
|
1322
|
-
|
1323
|
-
|
1918
|
+
# @api private
|
1919
|
+
def swap_optionals(array)
|
1920
|
+
if array.size > 1
|
1921
|
+
parts = array.partition {|t| t.is_a?(POptionalType) }
|
1922
|
+
optionals = parts[0]
|
1923
|
+
if optionals.size > 1
|
1924
|
+
others = parts[1]
|
1925
|
+
others << POptionalType.new(PVariantType.new(optionals.map { |optional| optional.type }).normalize)
|
1926
|
+
array = others
|
1324
1927
|
end
|
1928
|
+
end
|
1929
|
+
array
|
1930
|
+
end
|
1325
1931
|
|
1326
|
-
|
1327
|
-
|
1932
|
+
# @api private
|
1933
|
+
def swap_not_undefs(array)
|
1934
|
+
if array.size > 1
|
1935
|
+
parts = array.partition {|t| t.is_a?(PNotUndefType) }
|
1936
|
+
not_undefs = parts[0]
|
1937
|
+
if not_undefs.size > 1
|
1938
|
+
others = parts[1]
|
1939
|
+
others << PNotUndefType.new(PVariantType.new(not_undefs.map { |not_undef| not_undef.type }).normalize)
|
1940
|
+
array = others
|
1328
1941
|
end
|
1942
|
+
end
|
1943
|
+
array
|
1944
|
+
end
|
1329
1945
|
|
1330
|
-
|
1331
|
-
|
1332
|
-
|
1946
|
+
# @api private
|
1947
|
+
def merge_enums(array)
|
1948
|
+
if array.size > 1
|
1949
|
+
parts = array.partition {|t| t.is_a?(PEnumType) || t.is_a?(PStringType) && !t.values.empty? }
|
1950
|
+
enums = parts[0]
|
1951
|
+
if enums.size > 1
|
1952
|
+
others = parts[1]
|
1953
|
+
others << PEnumType.new(enums.map { |enum| enum.values }.flatten.uniq)
|
1954
|
+
array = others
|
1333
1955
|
end
|
1956
|
+
end
|
1957
|
+
array
|
1958
|
+
end
|
1334
1959
|
|
1335
|
-
|
1336
|
-
|
1960
|
+
# @api private
|
1961
|
+
def merge_patterns(array)
|
1962
|
+
if array.size > 1
|
1963
|
+
parts = array.partition {|t| t.is_a?(PPatternType) }
|
1964
|
+
patterns = parts[0]
|
1965
|
+
if patterns.size > 1
|
1966
|
+
others = parts[1]
|
1967
|
+
others << PPatternType.new(patterns.map { |pattern| pattern.patterns }.flatten.uniq)
|
1968
|
+
array = others
|
1337
1969
|
end
|
1970
|
+
end
|
1971
|
+
array
|
1972
|
+
end
|
1338
1973
|
|
1339
|
-
|
1340
|
-
|
1341
|
-
|
1342
|
-
|
1343
|
-
|
1974
|
+
# @api private
|
1975
|
+
def merge_int_ranges(array)
|
1976
|
+
if array.size > 1
|
1977
|
+
parts = array.partition {|t| t.is_a?(PIntegerType) }
|
1978
|
+
ranges = parts[0]
|
1979
|
+
array = merge_ranges(ranges) + parts[1] if ranges.size > 1
|
1980
|
+
end
|
1981
|
+
array
|
1982
|
+
end
|
1344
1983
|
|
1345
|
-
|
1346
|
-
|
1347
|
-
|
1348
|
-
|
1349
|
-
|
1350
|
-
|
1351
|
-
|
1352
|
-
|
1353
|
-
|
1354
|
-
|
1355
|
-
|
1356
|
-
|
1357
|
-
|
1358
|
-
|
1359
|
-
|
1360
|
-
|
1361
|
-
|
1362
|
-
|
1363
|
-
|
1364
|
-
else
|
1365
|
-
types.any? {|option_t| option_t.assignable?(other) }
|
1366
|
-
end
|
1367
|
-
end
|
1984
|
+
def merge_float_ranges(array)
|
1985
|
+
if array.size > 1
|
1986
|
+
parts = array.partition {|t| t.is_a?(PFloatType) }
|
1987
|
+
ranges = parts[0]
|
1988
|
+
array = merge_ranges(ranges) + parts[1] if ranges.size > 1
|
1989
|
+
end
|
1990
|
+
array
|
1991
|
+
end
|
1992
|
+
|
1993
|
+
# @api private
|
1994
|
+
def merge_ranges(ranges)
|
1995
|
+
result = []
|
1996
|
+
while !ranges.empty?
|
1997
|
+
unmerged = []
|
1998
|
+
x = ranges.pop
|
1999
|
+
result << ranges.inject(x) do |memo, y|
|
2000
|
+
merged = memo.merge(y)
|
2001
|
+
if merged.nil?
|
2002
|
+
unmerged << y
|
1368
2003
|
else
|
1369
|
-
|
1370
|
-
types.any? { |option_t| option_t.assignable?(o) }
|
2004
|
+
memo = merged
|
1371
2005
|
end
|
2006
|
+
memo
|
1372
2007
|
end
|
2008
|
+
ranges = unmerged
|
1373
2009
|
end
|
2010
|
+
result
|
2011
|
+
end
|
2012
|
+
end
|
1374
2013
|
|
1375
|
-
|
1376
|
-
|
1377
|
-
|
1378
|
-
|
2014
|
+
# @api public
|
2015
|
+
#
|
2016
|
+
class PRuntimeType < PAnyType
|
2017
|
+
attr_reader :runtime, :runtime_type_name
|
1379
2018
|
|
1380
|
-
|
1381
|
-
|
1382
|
-
|
1383
|
-
|
2019
|
+
def initialize(runtime, runtime_type_name = nil)
|
2020
|
+
@runtime = runtime
|
2021
|
+
@runtime_type_name = runtime_type_name
|
2022
|
+
end
|
1384
2023
|
|
1385
|
-
|
1386
|
-
|
1387
|
-
|
2024
|
+
def hash
|
2025
|
+
@runtime.hash * 31 + @runtime_type_name.hash
|
2026
|
+
end
|
1388
2027
|
|
1389
|
-
|
1390
|
-
|
1391
|
-
|
2028
|
+
def eql?(o)
|
2029
|
+
self.class == o.class && @runtime == o.runtime && @runtime_type_name == o.runtime_type_name
|
2030
|
+
end
|
1392
2031
|
|
1393
|
-
|
1394
|
-
|
1395
|
-
|
2032
|
+
def instance?(o)
|
2033
|
+
assignable?(TypeCalculator.infer(o))
|
2034
|
+
end
|
1396
2035
|
|
1397
|
-
|
2036
|
+
def iterable?(guard = nil)
|
2037
|
+
c = class_from_string(@runtime_type_name)
|
2038
|
+
c.nil? ? false : c < Iterable
|
2039
|
+
end
|
1398
2040
|
|
1399
|
-
|
2041
|
+
def iterable_type(guard = nil)
|
2042
|
+
iterable?(guard) ? PIterableType.new(self) : nil
|
2043
|
+
end
|
1400
2044
|
|
1401
|
-
|
1402
|
-
# a class that is the same or subclass of t1's resolved runtime type name
|
1403
|
-
# @api private
|
1404
|
-
def _assignable?(o)
|
1405
|
-
return false unless o.is_a?(PRuntimeType)
|
1406
|
-
return false unless @runtime == o.runtime
|
1407
|
-
return true if @runtime_type_name.nil? # t1 is wider
|
1408
|
-
return false if o.runtime_type_name.nil? # t1 not nil, so o can not be wider
|
2045
|
+
DEFAULT = PRuntimeType.new(nil)
|
1409
2046
|
|
1410
|
-
|
1411
|
-
c1 = class_from_string(@runtime_type_name)
|
1412
|
-
c2 = class_from_string(o.runtime_type_name)
|
1413
|
-
return false unless c1.is_a?(Module) && c2.is_a?(Module)
|
1414
|
-
!!(c2 <= c1)
|
1415
|
-
end
|
1416
|
-
end
|
2047
|
+
protected
|
1417
2048
|
|
1418
|
-
|
1419
|
-
|
1420
|
-
|
1421
|
-
|
2049
|
+
# Assignable if o's has the same runtime and the runtime name resolves to
|
2050
|
+
# a class that is the same or subclass of t1's resolved runtime type name
|
2051
|
+
# @api private
|
2052
|
+
def _assignable?(o, guard)
|
2053
|
+
return false unless o.is_a?(PRuntimeType)
|
2054
|
+
return false unless @runtime == o.runtime
|
2055
|
+
return true if @runtime_type_name.nil? # t1 is wider
|
2056
|
+
return false if o.runtime_type_name.nil? # t1 not nil, so o can not be wider
|
1422
2057
|
|
1423
|
-
|
2058
|
+
# NOTE: This only supports Ruby, must change when/if the set of runtimes is expanded
|
2059
|
+
c1 = class_from_string(@runtime_type_name)
|
2060
|
+
c2 = class_from_string(o.runtime_type_name)
|
2061
|
+
return false unless c1.is_a?(Module) && c2.is_a?(Module)
|
2062
|
+
!!(c2 <= c1)
|
2063
|
+
end
|
2064
|
+
end
|
1424
2065
|
|
1425
|
-
|
1426
|
-
|
1427
|
-
|
2066
|
+
# Abstract representation of a type that can be placed in a Catalog.
|
2067
|
+
# @api public
|
2068
|
+
#
|
2069
|
+
class PCatalogEntryType < PAnyType
|
1428
2070
|
|
1429
|
-
|
1430
|
-
# @api private
|
1431
|
-
def _assignable?(o)
|
1432
|
-
o.is_a?(PCatalogEntryType)
|
1433
|
-
end
|
1434
|
-
end
|
2071
|
+
DEFAULT = PCatalogEntryType.new
|
1435
2072
|
|
1436
|
-
|
1437
|
-
|
1438
|
-
|
1439
|
-
class PHostClassType < PCatalogEntryType
|
1440
|
-
attr_reader :class_name
|
2073
|
+
def instance?(o)
|
2074
|
+
assignable?(TypeCalculator.infer(o))
|
2075
|
+
end
|
1441
2076
|
|
1442
|
-
|
1443
|
-
|
1444
|
-
|
2077
|
+
protected
|
2078
|
+
# @api private
|
2079
|
+
def _assignable?(o, guard)
|
2080
|
+
o.is_a?(PCatalogEntryType)
|
2081
|
+
end
|
2082
|
+
end
|
1445
2083
|
|
1446
|
-
|
1447
|
-
|
1448
|
-
|
1449
|
-
|
1450
|
-
|
1451
|
-
end
|
2084
|
+
# Represents a (host-) class in the Puppet Language.
|
2085
|
+
# @api public
|
2086
|
+
#
|
2087
|
+
class PHostClassType < PCatalogEntryType
|
2088
|
+
attr_reader :class_name
|
1452
2089
|
|
1453
|
-
|
2090
|
+
def initialize(class_name)
|
2091
|
+
@class_name = class_name
|
2092
|
+
end
|
1454
2093
|
|
1455
|
-
|
2094
|
+
def hash
|
2095
|
+
11 * @class_name.hash
|
2096
|
+
end
|
2097
|
+
def eql?(o)
|
2098
|
+
self.class == o.class && @class_name == o.class_name
|
2099
|
+
end
|
1456
2100
|
|
1457
|
-
|
1458
|
-
def _assignable?(o)
|
1459
|
-
return false unless o.is_a?(PHostClassType)
|
1460
|
-
# Class = Class[name}, Class[name] != Class
|
1461
|
-
return true if @class_name.nil?
|
1462
|
-
# Class[name] = Class[name]
|
1463
|
-
@class_name == o.class_name
|
1464
|
-
end
|
1465
|
-
end
|
2101
|
+
DEFAULT = PHostClassType.new(nil)
|
1466
2102
|
|
1467
|
-
|
1468
|
-
# @api public
|
1469
|
-
#
|
1470
|
-
class PResourceType < PCatalogEntryType
|
1471
|
-
attr_reader :type_name, :title
|
2103
|
+
protected
|
1472
2104
|
|
1473
|
-
|
1474
|
-
|
1475
|
-
|
1476
|
-
|
2105
|
+
# @api private
|
2106
|
+
def _assignable?(o, guard)
|
2107
|
+
return false unless o.is_a?(PHostClassType)
|
2108
|
+
# Class = Class[name}, Class[name] != Class
|
2109
|
+
return true if @class_name.nil?
|
2110
|
+
# Class[name] = Class[name]
|
2111
|
+
@class_name == o.class_name
|
2112
|
+
end
|
2113
|
+
end
|
1477
2114
|
|
1478
|
-
|
1479
|
-
|
1480
|
-
|
2115
|
+
# Represents a Resource Type in the Puppet Language
|
2116
|
+
# @api public
|
2117
|
+
#
|
2118
|
+
class PResourceType < PCatalogEntryType
|
2119
|
+
attr_reader :type_name, :title
|
1481
2120
|
|
1482
|
-
|
1483
|
-
|
1484
|
-
|
2121
|
+
def initialize(type_name, title = nil)
|
2122
|
+
@type_name = type_name
|
2123
|
+
@title = title
|
2124
|
+
end
|
2125
|
+
|
2126
|
+
def hash
|
2127
|
+
@type_name.hash * 31 + @title.hash
|
2128
|
+
end
|
2129
|
+
|
2130
|
+
def eql?(o)
|
2131
|
+
self.class == o.class && @type_name == o.type_name && @title == o.title
|
2132
|
+
end
|
2133
|
+
|
2134
|
+
DEFAULT = PResourceType.new(nil)
|
2135
|
+
|
2136
|
+
protected
|
2137
|
+
|
2138
|
+
# @api private
|
2139
|
+
def _assignable?(o, guard)
|
2140
|
+
return false unless o.is_a?(PResourceType)
|
2141
|
+
return true if @type_name.nil?
|
2142
|
+
return false if @type_name != o.type_name
|
2143
|
+
return true if @title.nil?
|
2144
|
+
@title == o.title
|
2145
|
+
end
|
2146
|
+
end
|
2147
|
+
|
2148
|
+
# Represents a type that accept PUndefType instead of the type parameter
|
2149
|
+
# required_type - is a short hand for Variant[T, Undef]
|
2150
|
+
# @api public
|
2151
|
+
#
|
2152
|
+
class POptionalType < PTypeWithContainedType
|
2153
|
+
def optional_type
|
2154
|
+
@type
|
2155
|
+
end
|
1485
2156
|
|
1486
|
-
|
2157
|
+
def kind_of_callable?(optional=true, guard = nil)
|
2158
|
+
optional && !@type.nil? && @type.kind_of_callable?(optional, guard)
|
2159
|
+
end
|
1487
2160
|
|
1488
|
-
|
2161
|
+
def instance?(o)
|
2162
|
+
PUndefType::DEFAULT.instance?(o) || (!@type.nil? && @type.instance?(o))
|
2163
|
+
end
|
1489
2164
|
|
1490
|
-
|
1491
|
-
|
1492
|
-
|
1493
|
-
|
1494
|
-
|
1495
|
-
|
1496
|
-
|
2165
|
+
def normalize(guard = nil)
|
2166
|
+
n = super
|
2167
|
+
if n.type.nil?
|
2168
|
+
n
|
2169
|
+
else
|
2170
|
+
if n.type.is_a?(PNotUndefType)
|
2171
|
+
# No point in having an NotUndef in an Optional
|
2172
|
+
POptionalType.new(n.type.type).normalize
|
2173
|
+
elsif n.type.assignable?(PUndefType::DEFAULT)
|
2174
|
+
# THe type is Optional anyway, so it can be stripped of
|
2175
|
+
n.type
|
2176
|
+
else
|
2177
|
+
n
|
1497
2178
|
end
|
1498
2179
|
end
|
2180
|
+
end
|
1499
2181
|
|
1500
|
-
|
1501
|
-
# required_type - is a short hand for Variant[T, Undef]
|
1502
|
-
# @api public
|
1503
|
-
#
|
1504
|
-
class POptionalType < PAnyType
|
1505
|
-
attr_reader :optional_type
|
2182
|
+
DEFAULT = POptionalType.new(nil)
|
1506
2183
|
|
1507
|
-
|
1508
|
-
@optional_type = optional_type
|
1509
|
-
end
|
2184
|
+
protected
|
1510
2185
|
|
1511
|
-
|
1512
|
-
|
1513
|
-
|
2186
|
+
# @api private
|
2187
|
+
def _assignable?(o, guard)
|
2188
|
+
return true if o.is_a?(PUndefType)
|
2189
|
+
return true if @type.nil?
|
2190
|
+
if o.is_a?(POptionalType)
|
2191
|
+
@type.assignable?(o.optional_type, guard)
|
2192
|
+
else
|
2193
|
+
@type.assignable?(o, guard)
|
2194
|
+
end
|
2195
|
+
end
|
2196
|
+
end
|
1514
2197
|
|
1515
|
-
|
1516
|
-
|
1517
|
-
end
|
2198
|
+
class PTypeReferenceType < PAnyType
|
2199
|
+
attr_reader :name, :parameters
|
1518
2200
|
|
1519
|
-
|
1520
|
-
|
1521
|
-
|
2201
|
+
def initialize(name, parameters = nil)
|
2202
|
+
@name = name
|
2203
|
+
@parameters = parameters.nil? ? EMPTY_ARRAY : parameters
|
2204
|
+
end
|
1522
2205
|
|
1523
|
-
|
1524
|
-
|
1525
|
-
|
2206
|
+
def callable?(args)
|
2207
|
+
false
|
2208
|
+
end
|
1526
2209
|
|
1527
|
-
|
1528
|
-
|
1529
|
-
|
2210
|
+
def instance?(o)
|
2211
|
+
false
|
2212
|
+
end
|
2213
|
+
|
2214
|
+
def hash
|
2215
|
+
@name.hash * 31 + @parameters.hash
|
2216
|
+
end
|
1530
2217
|
|
1531
|
-
|
2218
|
+
def eql?(o)
|
2219
|
+
super && o.name == @name && o.parameters == @parameters
|
2220
|
+
end
|
1532
2221
|
|
1533
|
-
|
2222
|
+
protected
|
1534
2223
|
|
1535
|
-
|
1536
|
-
|
1537
|
-
|
1538
|
-
|
1539
|
-
|
1540
|
-
|
1541
|
-
|
1542
|
-
|
1543
|
-
|
2224
|
+
def _assignable?(o, guard)
|
2225
|
+
# A type must be assignable to itself or a lot of unit tests will break
|
2226
|
+
o == self
|
2227
|
+
end
|
2228
|
+
|
2229
|
+
DEFAULT = PTypeReferenceType.new('UnresolvedReference')
|
2230
|
+
end
|
2231
|
+
|
2232
|
+
# Describes a named alias for another Type.
|
2233
|
+
# The alias is created with a name and an unresolved type expression. The type expression may
|
2234
|
+
# in turn contain other aliases (including the alias that contains it) which means that an alias
|
2235
|
+
# might contain self recursion. Whether or not that is the case is computed and remembered when the alias
|
2236
|
+
# is resolved since guarding against self recursive constructs is relatively expensive.
|
2237
|
+
#
|
2238
|
+
class PTypeAliasType < PAnyType
|
2239
|
+
attr_reader :name
|
2240
|
+
|
2241
|
+
# @param name [String] The name of the type
|
2242
|
+
# @param type_expr [Model::PopsObject] The expression that describes the aliased type
|
2243
|
+
# @param resolved_type [PAnyType] the resolve type (only used for the DEFAULT initialization)
|
2244
|
+
def initialize(name, type_expr, resolved_type = nil)
|
2245
|
+
@name = name
|
2246
|
+
@type_expr = type_expr
|
2247
|
+
@resolved_type = resolved_type
|
2248
|
+
@self_recursion = false
|
2249
|
+
end
|
2250
|
+
|
2251
|
+
# Returns the resolved type. The type must have been resolved by a call prior to calls to this
|
2252
|
+
# method or an error will be raised.
|
2253
|
+
#
|
2254
|
+
# @return [PAnyType] The resolved type of this alias.
|
2255
|
+
# @raise [Puppet::Error] unless the type has been resolved prior to calling this method
|
2256
|
+
def resolved_type
|
2257
|
+
raise Puppet::Error, "Reference to unresolved type #{@name}" unless @resolved_type
|
2258
|
+
@resolved_type
|
2259
|
+
end
|
2260
|
+
|
2261
|
+
def callable_args?(callable, guard)
|
2262
|
+
guarded_recursion(guard, false) { |g| resolved_type.callable_args?(callable, g) }
|
2263
|
+
end
|
2264
|
+
|
2265
|
+
def kind_of_callable?(optional=true, guard = nil)
|
2266
|
+
guarded_recursion(guard, false) { |g| resolved_type.kind_of_callable?(optional, g) }
|
2267
|
+
end
|
2268
|
+
|
2269
|
+
def instance?(o)
|
2270
|
+
# No value can ever be recursive so no guard is needed here
|
2271
|
+
resolved_type.instance?(o)
|
2272
|
+
end
|
2273
|
+
|
2274
|
+
def iterable?(guard = nil)
|
2275
|
+
guarded_recursion(guard, false) { |g| resolved_type.iterable?(g) }
|
2276
|
+
end
|
2277
|
+
|
2278
|
+
def iterable_type(guard = nil)
|
2279
|
+
guarded_recursion(guard, nil) { |g| resolved_type.iterable_type(g) }
|
2280
|
+
end
|
2281
|
+
|
2282
|
+
def hash
|
2283
|
+
@name.hash
|
2284
|
+
end
|
2285
|
+
|
2286
|
+
# Called from the TypeParser once it has found a type using the Loader. The TypeParser will
|
2287
|
+
# interpret the contained expression and the resolved type is remembered. This method also
|
2288
|
+
# checks and remembers if the resolve type contains self recursion.
|
2289
|
+
#
|
2290
|
+
# @param type_parser [TypeParser] type parser that will interpret the type expression
|
2291
|
+
# @param loader [Loader::Loader] loader to use when loading type aliases
|
2292
|
+
# @return [PTypeAliasType] the receiver of the call, i.e. `self`
|
2293
|
+
# @api private
|
2294
|
+
def resolve(type_parser, loader)
|
2295
|
+
if @resolved_type.nil?
|
2296
|
+
# resolved to PTypeReferenceType::DEFAULT during resolve to avoid endless recursion
|
2297
|
+
@resolved_type = PTypeReferenceType::DEFAULT
|
2298
|
+
@self_recursion = true # assumed while it being found out below
|
2299
|
+
begin
|
2300
|
+
@resolved_type = type_parser.interpret(@type_expr, loader).normalize
|
2301
|
+
|
2302
|
+
# Find out if this type is recursive. A recursive type has performance implications
|
2303
|
+
# on several methods and this knowledge is used to avoid that for non-recursive
|
2304
|
+
# types.
|
2305
|
+
guard = RecursionGuard.new
|
2306
|
+
accept(NoopTypeAcceptor::INSTANCE, guard)
|
2307
|
+
@self_recursion = guard.recursive_this?(self)
|
2308
|
+
rescue
|
2309
|
+
@resolved_type = nil
|
2310
|
+
raise
|
1544
2311
|
end
|
1545
2312
|
end
|
2313
|
+
self
|
2314
|
+
end
|
2315
|
+
|
2316
|
+
def eql?(o)
|
2317
|
+
super && o.name == @name
|
1546
2318
|
end
|
1547
|
-
end
|
1548
2319
|
|
2320
|
+
def accept(visitor, guard)
|
2321
|
+
guarded_recursion(guard, nil) do |g|
|
2322
|
+
super
|
2323
|
+
resolved_type.accept(visitor, g)
|
2324
|
+
end
|
2325
|
+
end
|
2326
|
+
|
2327
|
+
def self_recursion?
|
2328
|
+
@self_recursion
|
2329
|
+
end
|
2330
|
+
|
2331
|
+
protected
|
2332
|
+
|
2333
|
+
def _assignable?(o, guard)
|
2334
|
+
guard ||= RecursionGuard.new
|
2335
|
+
if guard.add_this(self) == RecursionGuard::SELF_RECURSION_IN_BOTH
|
2336
|
+
# Recursion detected both in self and other. This means that other is assignable
|
2337
|
+
# to self. This point would not have been reached otherwise
|
2338
|
+
true
|
2339
|
+
else
|
2340
|
+
resolved_type.assignable?(o, guard)
|
2341
|
+
end
|
2342
|
+
end
|
2343
|
+
|
2344
|
+
private
|
2345
|
+
|
2346
|
+
def guarded_recursion(guard, dflt)
|
2347
|
+
if @self_recursion
|
2348
|
+
guard ||= RecursionGuard.new
|
2349
|
+
(guard.add_this(self) & RecursionGuard::SELF_RECURSION_IN_THIS) == 0 ? yield(guard) : dflt
|
2350
|
+
else
|
2351
|
+
yield(guard)
|
2352
|
+
end
|
2353
|
+
end
|
2354
|
+
|
2355
|
+
DEFAULT = PTypeAliasType.new('UnresolvedAlias', nil, PTypeReferenceType::DEFAULT)
|
2356
|
+
end
|
2357
|
+
end
|
2358
|
+
end
|