mirror of
https://github.com/ansible-collections/community.general.git
synced 2024-09-14 20:13:21 +02:00
Ansible.Basic - make choices validate case insensitive (#51203)
* Ansible.Basic - make choices validate case insensitive * fix win_certificate_store tests
This commit is contained in:
parent
c34f85c788
commit
a259b810ad
3 changed files with 159 additions and 57 deletions
|
@ -553,9 +553,7 @@ namespace Ansible.Basic
|
||||||
{
|
{
|
||||||
string msg = String.Format("argument spec entry contains an invalid key '{0}', valid keys: {1}",
|
string msg = String.Format("argument spec entry contains an invalid key '{0}', valid keys: {1}",
|
||||||
key, String.Join(", ", specDefaults.Keys));
|
key, String.Join(", ", specDefaults.Keys));
|
||||||
if (optionsContext.Count > 0)
|
throw new ArgumentException(FormatOptionsContext(msg, " - "));
|
||||||
msg += String.Format(" - found in {0}.", String.Join(" -> ", optionsContext));
|
|
||||||
throw new ArgumentException(msg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ensure the value is casted to the type we expect
|
// ensure the value is casted to the type we expect
|
||||||
|
@ -597,9 +595,7 @@ namespace Ansible.Basic
|
||||||
{
|
{
|
||||||
string msg = String.Format("argument spec for '{0}' did not match expected type {1}: actual type {2}",
|
string msg = String.Format("argument spec for '{0}' did not match expected type {1}: actual type {2}",
|
||||||
key, optionType.FullName, actualType.FullName);
|
key, optionType.FullName, actualType.FullName);
|
||||||
if (optionsContext.Count > 0)
|
throw new ArgumentException(FormatOptionsContext(msg, " - "));
|
||||||
msg += String.Format(" - found in {0}.", String.Join(" -> ", optionsContext));
|
|
||||||
throw new ArgumentException(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -626,18 +622,14 @@ namespace Ansible.Basic
|
||||||
if (!optionTypes.ContainsKey(typeValue))
|
if (!optionTypes.ContainsKey(typeValue))
|
||||||
{
|
{
|
||||||
string msg = String.Format("{0} '{1}' is unsupported", key, typeValue);
|
string msg = String.Format("{0} '{1}' is unsupported", key, typeValue);
|
||||||
if (optionsContext.Count > 0)
|
msg = String.Format("{0}. Valid types are: {1}", FormatOptionsContext(msg, " - "), String.Join(", ", optionTypes.Keys));
|
||||||
msg += String.Format(" - found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
msg += String.Format(". Valid types are: {0}", String.Join(", ", optionTypes.Keys));
|
|
||||||
throw new ArgumentException(msg);
|
throw new ArgumentException(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
else if (!(entry.Value is Delegate))
|
else if (!(entry.Value is Delegate))
|
||||||
{
|
{
|
||||||
string msg = String.Format("{0} must either be a string or delegate, was: {1}", key, valueType.FullName);
|
string msg = String.Format("{0} must either be a string or delegate, was: {1}", key, valueType.FullName);
|
||||||
if (optionsContext.Count > 0)
|
throw new ArgumentException(FormatOptionsContext(msg, " - "));
|
||||||
msg += String.Format(" - found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
throw new ArgumentException(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -755,34 +747,43 @@ namespace Ansible.Basic
|
||||||
{
|
{
|
||||||
string msg = String.Format("argument for {0} is of type {1} and we were unable to convert to {2}: {3}",
|
string msg = String.Format("argument for {0} is of type {1} and we were unable to convert to {2}: {3}",
|
||||||
k, value.GetType(), type, e.InnerException.Message);
|
k, value.GetType(), type, e.InnerException.Message);
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// ensure it matches the choices if there are choices set
|
// ensure it matches the choices if there are choices set
|
||||||
List<object> choices = (List<object>)v["choices"];
|
List<string> choices = ((List<object>)v["choices"]).Select(x => x.ToString()).Cast<string>().ToList();
|
||||||
if (choices.Count > 0)
|
if (choices.Count > 0)
|
||||||
{
|
{
|
||||||
// allow one or more when type="list" param with choices
|
List<string> values;
|
||||||
|
string choiceMsg;
|
||||||
if (type == "list")
|
if (type == "list")
|
||||||
{
|
{
|
||||||
var diffList = ((List<object>)value).Except(choices).ToList();
|
values = ((List<object>)value).Select(x => x.ToString()).Cast<string>().ToList();
|
||||||
if (diffList.Count > 0)
|
choiceMsg = "one or more of";
|
||||||
{
|
|
||||||
string msg = String.Format("value of {0} must be one or more of: {1}. Got no match for: {2}",
|
|
||||||
k, String.Join(", ", choices), String.Join(", ", diffList));
|
|
||||||
if (optionsContext.Count > 0)
|
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else if (!choices.Contains(value))
|
else
|
||||||
{
|
{
|
||||||
string msg = String.Format("value of {0} must be one of: {1}, got: {2}", k, String.Join(", ", choices), value);
|
values = new List<string>() { value.ToString() };
|
||||||
if (optionsContext.Count > 0)
|
choiceMsg = "one of";
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
}
|
||||||
FailJson(msg);
|
|
||||||
|
List<string> diffList = values.Except(choices, StringComparer.OrdinalIgnoreCase).ToList();
|
||||||
|
List<string> caseDiffList = values.Except(choices).ToList();
|
||||||
|
if (diffList.Count > 0)
|
||||||
|
{
|
||||||
|
string msg = String.Format("value of {0} must be {1}: {2}. Got no match for: {3}",
|
||||||
|
k, choiceMsg, String.Join(", ", choices), String.Join(", ", diffList));
|
||||||
|
FailJson(FormatOptionsContext(msg));
|
||||||
|
}
|
||||||
|
else if (caseDiffList.Count > 0)
|
||||||
|
{
|
||||||
|
// For backwards compatibility with Legacy.psm1 we need to be matching choices that are not case sensitive.
|
||||||
|
// We will warn the user it was case insensitive and tell them this will become case sensitive in the future.
|
||||||
|
string msg = String.Format(
|
||||||
|
"value of {0} was a case insensitive match of {1}: {2}. Checking of choices will be case sensitive in a future Ansible release. Case insensitive matches were: {3}",
|
||||||
|
k, choiceMsg, String.Join(", ", choices), String.Join(", ", caseDiffList.Select(x => RemoveNoLogValues(x, noLogValues)))
|
||||||
|
);
|
||||||
|
Warn(FormatOptionsContext(msg));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -848,9 +849,7 @@ namespace Ansible.Basic
|
||||||
{
|
{
|
||||||
legalInputs.RemoveAll(x => passVars.Keys.Contains(x.Replace("_ansible_", "")));
|
legalInputs.RemoveAll(x => passVars.Keys.Contains(x.Replace("_ansible_", "")));
|
||||||
string msg = String.Format("Unsupported parameters for ({0}) module: {1}", ModuleName, String.Join(", ", unsupportedParameters));
|
string msg = String.Format("Unsupported parameters for ({0}) module: {1}", ModuleName, String.Join(", ", unsupportedParameters));
|
||||||
if (optionsContext.Count > 0)
|
msg = String.Format("{0}. Supported parameters include: {1}", FormatOptionsContext(msg), String.Join(", ", legalInputs));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
msg += String.Format(". Supported parameters include: {0}", String.Join(", ", legalInputs));
|
|
||||||
FailJson(msg);
|
FailJson(msg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -871,9 +870,7 @@ namespace Ansible.Basic
|
||||||
if (count > 1)
|
if (count > 1)
|
||||||
{
|
{
|
||||||
string msg = String.Format("parameters are mutually exclusive: {0}", String.Join(", ", mutualCheck));
|
string msg = String.Format("parameters are mutually exclusive: {0}", String.Join(", ", mutualCheck));
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -899,9 +896,7 @@ namespace Ansible.Basic
|
||||||
if (missing.Count > 0)
|
if (missing.Count > 0)
|
||||||
{
|
{
|
||||||
string msg = String.Format("missing required arguments: {0}", String.Join(", ", missing));
|
string msg = String.Format("missing required arguments: {0}", String.Join(", ", missing));
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -923,9 +918,7 @@ namespace Ansible.Basic
|
||||||
if (found.Contains(true) && found.Contains(false))
|
if (found.Contains(true) && found.Contains(false))
|
||||||
{
|
{
|
||||||
string msg = String.Format("parameters are required together: {0}", String.Join(", ", requiredCheck));
|
string msg = String.Format("parameters are required together: {0}", String.Join(", ", requiredCheck));
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -946,9 +939,7 @@ namespace Ansible.Basic
|
||||||
if (count == 0)
|
if (count == 0)
|
||||||
{
|
{
|
||||||
string msg = String.Format("one of the following is required: {0}", String.Join(", ", requiredCheck));
|
string msg = String.Format("one of the following is required: {0}", String.Join(", ", requiredCheck));
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -993,9 +984,7 @@ namespace Ansible.Basic
|
||||||
{
|
{
|
||||||
string msg = String.Format("{0} is {1} but {2} of the following are missing: {3}",
|
string msg = String.Format("{0} is {1} but {2} of the following are missing: {3}",
|
||||||
key, val.ToString(), term, String.Join(", ", missing));
|
key, val.ToString(), term, String.Join(", ", missing));
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1036,9 +1025,7 @@ namespace Ansible.Basic
|
||||||
{
|
{
|
||||||
string msg = String.Format("argument for list entry {0} is of type {1} and we were unable to convert to {2}: {3}",
|
string msg = String.Format("argument for list entry {0} is of type {1} and we were unable to convert to {2}: {3}",
|
||||||
key, element.GetType(), elements, e.Message);
|
key, element.GetType(), elements, e.Message);
|
||||||
if (optionsContext.Count > 0)
|
FailJson(FormatOptionsContext(msg));
|
||||||
msg += String.Format(" found in {0}", String.Join(" -> ", optionsContext));
|
|
||||||
FailJson(msg);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1216,6 +1203,13 @@ namespace Ansible.Basic
|
||||||
cleanupFiles = new List<string>();
|
cleanupFiles = new List<string>();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private string FormatOptionsContext(string msg, string prefix = " ")
|
||||||
|
{
|
||||||
|
if (optionsContext.Count > 0)
|
||||||
|
msg += String.Format("{0}found in {1}", prefix, String.Join(" -> ", optionsContext));
|
||||||
|
return msg;
|
||||||
|
}
|
||||||
|
|
||||||
[DllImport("kernel32.dll")]
|
[DllImport("kernel32.dll")]
|
||||||
private static extern IntPtr GetConsoleWindow();
|
private static extern IntPtr GetConsoleWindow();
|
||||||
|
|
||||||
|
|
|
@ -5,7 +5,7 @@
|
||||||
path: '{{win_cert_dir}}\subj-cert.pem'
|
path: '{{win_cert_dir}}\subj-cert.pem'
|
||||||
store_location: FakeLocation
|
store_location: FakeLocation
|
||||||
register: fail_fake_location
|
register: fail_fake_location
|
||||||
failed_when: "fail_fake_location.msg != 'value of store_location must be one of: CurrentUser, LocalMachine, got: FakeLocation'"
|
failed_when: "fail_fake_location.msg != 'value of store_location must be one of: CurrentUser, LocalMachine. Got no match for: FakeLocation'"
|
||||||
|
|
||||||
- name: fail with invalid store name
|
- name: fail with invalid store name
|
||||||
win_certificate_store:
|
win_certificate_store:
|
||||||
|
@ -13,7 +13,7 @@
|
||||||
path: '{{win_cert_dir}}\subj-cert.pem'
|
path: '{{win_cert_dir}}\subj-cert.pem'
|
||||||
store_name: FakeName
|
store_name: FakeName
|
||||||
register: fail_fake_name
|
register: fail_fake_name
|
||||||
failed_when: "fail_fake_name.msg != 'value of store_name must be one of: AddressBook, AuthRoot, CertificateAuthority, Disallowed, My, Root, TrustedPeople, TrustedPublisher, got: FakeName'"
|
failed_when: "fail_fake_name.msg != 'value of store_name must be one of: AddressBook, AuthRoot, CertificateAuthority, Disallowed, My, Root, TrustedPeople, TrustedPublisher. Got no match for: FakeName'"
|
||||||
|
|
||||||
- name: fail when state=present and no path is set
|
- name: fail when state=present and no path is set
|
||||||
win_certificate_store:
|
win_certificate_store:
|
||||||
|
|
|
@ -1172,7 +1172,7 @@ test_no_log - Invoked with:
|
||||||
$expected_msg = "internal error: argument spec entry contains an invalid key 'invalid', valid keys: apply_defaults, "
|
$expected_msg = "internal error: argument spec entry contains an invalid key 'invalid', valid keys: apply_defaults, "
|
||||||
$expected_msg += "aliases, choices, default, elements, mutually_exclusive, no_log, options, removed_in_version, "
|
$expected_msg += "aliases, choices, default, elements, mutually_exclusive, no_log, options, removed_in_version, "
|
||||||
$expected_msg += "required, required_if, required_one_of, required_together, supports_check_mode, type - "
|
$expected_msg += "required, required_if, required_one_of, required_together, supports_check_mode, type - "
|
||||||
$expected_msg += "found in option_key -> sub_option_key."
|
$expected_msg += "found in option_key -> sub_option_key"
|
||||||
|
|
||||||
$actual.Keys.Count | Assert-Equals -Expected 3
|
$actual.Keys.Count | Assert-Equals -Expected 3
|
||||||
$actual.failed | Assert-Equals -Expected $true
|
$actual.failed | Assert-Equals -Expected $true
|
||||||
|
@ -1446,6 +1446,114 @@ test_no_log - Invoked with:
|
||||||
$actual.invocation | Assert-DictionaryEquals -Expected @{module_args = $complex_args}
|
$actual.invocation | Assert-DictionaryEquals -Expected @{module_args = $complex_args}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
"Numeric choices" = {
|
||||||
|
$spec = @{
|
||||||
|
options = @{
|
||||||
|
option_key = @{
|
||||||
|
choices = 1, 2, 3
|
||||||
|
type = "int"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$complex_args = @{
|
||||||
|
option_key = "2"
|
||||||
|
}
|
||||||
|
|
||||||
|
$m = [Ansible.Basic.AnsibleModule]::Create(@(), $spec)
|
||||||
|
try {
|
||||||
|
$m.ExitJson()
|
||||||
|
} catch [System.Management.Automation.RuntimeException] {
|
||||||
|
$output = [Ansible.Basic.AnsibleModule]::FromJson($_test_out)
|
||||||
|
}
|
||||||
|
$output.Keys.Count | Assert-Equals -Expected 2
|
||||||
|
$output.changed | Assert-Equals -Expected $false
|
||||||
|
$output.invocation | Assert-DictionaryEquals -Expected @{module_args = @{option_key = 2}}
|
||||||
|
}
|
||||||
|
|
||||||
|
"Case insensitive choice" = {
|
||||||
|
$spec = @{
|
||||||
|
options = @{
|
||||||
|
option_key = @{
|
||||||
|
choices = "abc", "def"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$complex_args = @{
|
||||||
|
option_key = "ABC"
|
||||||
|
}
|
||||||
|
|
||||||
|
$m = [Ansible.Basic.AnsibleModule]::Create(@(), $spec)
|
||||||
|
try {
|
||||||
|
$m.ExitJson()
|
||||||
|
} catch [System.Management.Automation.RuntimeException] {
|
||||||
|
$output = [Ansible.Basic.AnsibleModule]::FromJson($_test_out)
|
||||||
|
}
|
||||||
|
$expected_warning = "value of option_key was a case insensitive match of one of: abc, def. "
|
||||||
|
$expected_warning += "Checking of choices will be case sensitive in a future Ansible release. "
|
||||||
|
$expected_warning += "Case insensitive matches were: ABC"
|
||||||
|
|
||||||
|
$output.invocation | Assert-DictionaryEquals -Expected @{module_args = @{option_key = "ABC"}}
|
||||||
|
$output.warnings.Count | Assert-Equals -Expected 1
|
||||||
|
$output.warnings[0] | Assert-Equals -Expected $expected_warning
|
||||||
|
}
|
||||||
|
|
||||||
|
"Case insensitive choice no_log" = {
|
||||||
|
$spec = @{
|
||||||
|
options = @{
|
||||||
|
option_key = @{
|
||||||
|
choices = "abc", "def"
|
||||||
|
no_log = $true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$complex_args = @{
|
||||||
|
option_key = "ABC"
|
||||||
|
}
|
||||||
|
|
||||||
|
$m = [Ansible.Basic.AnsibleModule]::Create(@(), $spec)
|
||||||
|
try {
|
||||||
|
$m.ExitJson()
|
||||||
|
} catch [System.Management.Automation.RuntimeException] {
|
||||||
|
$output = [Ansible.Basic.AnsibleModule]::FromJson($_test_out)
|
||||||
|
}
|
||||||
|
$expected_warning = "value of option_key was a case insensitive match of one of: abc, def. "
|
||||||
|
$expected_warning += "Checking of choices will be case sensitive in a future Ansible release. "
|
||||||
|
$expected_warning += "Case insensitive matches were: VALUE_SPECIFIED_IN_NO_LOG_PARAMETER"
|
||||||
|
|
||||||
|
$output.invocation | Assert-DictionaryEquals -Expected @{module_args = @{option_key = "VALUE_SPECIFIED_IN_NO_LOG_PARAMETER"}}
|
||||||
|
$output.warnings.Count | Assert-Equals -Expected 1
|
||||||
|
$output.warnings[0] | Assert-Equals -Expected $expected_warning
|
||||||
|
}
|
||||||
|
|
||||||
|
"Case insentitive choice as list" = {
|
||||||
|
$spec = @{
|
||||||
|
options = @{
|
||||||
|
option_key = @{
|
||||||
|
choices = "abc", "def", "ghi", "JKL"
|
||||||
|
type = "list"
|
||||||
|
elements = "str"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$complex_args = @{
|
||||||
|
option_key = "AbC", "ghi", "jkl"
|
||||||
|
}
|
||||||
|
|
||||||
|
$m = [Ansible.Basic.AnsibleModule]::Create(@(), $spec)
|
||||||
|
try {
|
||||||
|
$m.ExitJson()
|
||||||
|
} catch [System.Management.Automation.RuntimeException] {
|
||||||
|
$output = [Ansible.Basic.AnsibleModule]::FromJson($_test_out)
|
||||||
|
}
|
||||||
|
$expected_warning = "value of option_key was a case insensitive match of one or more of: abc, def, ghi, JKL. "
|
||||||
|
$expected_warning += "Checking of choices will be case sensitive in a future Ansible release. "
|
||||||
|
$expected_warning += "Case insensitive matches were: AbC, jkl"
|
||||||
|
|
||||||
|
$output.invocation | Assert-DictionaryEquals -Expected @{module_args = $complex_args}
|
||||||
|
$output.warnings.Count | Assert-Equals -Expected 1
|
||||||
|
$output.warnings[0] | Assert-Equals -Expected $expected_warning
|
||||||
|
}
|
||||||
|
|
||||||
"Invalid choice" = {
|
"Invalid choice" = {
|
||||||
$spec = @{
|
$spec = @{
|
||||||
options = @{
|
options = @{
|
||||||
|
@ -1468,7 +1576,7 @@ test_no_log - Invoked with:
|
||||||
}
|
}
|
||||||
$failed | Assert-Equals -Expected $true
|
$failed | Assert-Equals -Expected $true
|
||||||
|
|
||||||
$expected_msg = "value of option_key must be one of: a, b, got: c"
|
$expected_msg = "value of option_key must be one of: a, b. Got no match for: c"
|
||||||
|
|
||||||
$actual.Keys.Count | Assert-Equals -Expected 4
|
$actual.Keys.Count | Assert-Equals -Expected 4
|
||||||
$actual.changed | Assert-Equals -Expected $false
|
$actual.changed | Assert-Equals -Expected $false
|
||||||
|
@ -1500,7 +1608,7 @@ test_no_log - Invoked with:
|
||||||
}
|
}
|
||||||
$failed | Assert-Equals -Expected $true
|
$failed | Assert-Equals -Expected $true
|
||||||
|
|
||||||
$expected_msg = "value of option_key must be one of: a, b, got: ***"
|
$expected_msg = "value of option_key must be one of: a, b. Got no match for: ***"
|
||||||
|
|
||||||
$actual.Keys.Count | Assert-Equals -Expected 4
|
$actual.Keys.Count | Assert-Equals -Expected 4
|
||||||
$actual.changed | Assert-Equals -Expected $false
|
$actual.changed | Assert-Equals -Expected $false
|
||||||
|
|
Loading…
Reference in a new issue