chef 12.4.0.rc.2-universal-mingw32 → 12.4.0-universal-mingw32

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (49) hide show
  1. checksums.yaml +4 -4
  2. data/distro/powershell/chef/chef.psm1 +327 -0
  3. data/lib/chef/chef_class.rb +4 -4
  4. data/lib/chef/client.rb +12 -6
  5. data/lib/chef/node_map.rb +63 -38
  6. data/lib/chef/platform/priority_map.rb +54 -0
  7. data/lib/chef/platform/provider_mapping.rb +2 -2
  8. data/lib/chef/platform/provider_priority_map.rb +3 -21
  9. data/lib/chef/platform/resource_priority_map.rb +5 -22
  10. data/lib/chef/provider.rb +1 -1
  11. data/lib/chef/provider/package/rpm.rb +2 -2
  12. data/lib/chef/provider/service/debian.rb +0 -2
  13. data/lib/chef/provider/service/insserv.rb +0 -2
  14. data/lib/chef/provider/service/invokercd.rb +0 -2
  15. data/lib/chef/provider/service/redhat.rb +0 -2
  16. data/lib/chef/provider/service/upstart.rb +0 -2
  17. data/lib/chef/provider/user.rb +0 -2
  18. data/lib/chef/resource.rb +23 -24
  19. data/lib/chef/resource/lwrp_base.rb +2 -1
  20. data/lib/chef/resource/macports_package.rb +2 -1
  21. data/lib/chef/resource/package.rb +0 -5
  22. data/lib/chef/resource_resolver.rb +1 -0
  23. data/lib/chef/version.rb +1 -1
  24. data/spec/integration/recipes/lwrp_spec.rb +2 -6
  25. data/spec/integration/recipes/recipe_dsl_spec.rb +254 -39
  26. data/spec/support/shared/shared_examples.rb +1 -1
  27. data/spec/unit/api_client_spec.rb +1 -1
  28. data/spec/unit/client_spec.rb +35 -19
  29. data/spec/unit/cookbook_version_spec.rb +1 -1
  30. data/spec/unit/data_bag_item_spec.rb +1 -1
  31. data/spec/unit/data_bag_spec.rb +1 -1
  32. data/spec/unit/environment_spec.rb +1 -1
  33. data/spec/unit/exceptions_spec.rb +1 -1
  34. data/spec/unit/json_compat_spec.rb +1 -1
  35. data/spec/unit/lwrp_spec.rb +43 -4
  36. data/spec/unit/node_spec.rb +1 -1
  37. data/spec/unit/osc_user_spec.rb +1 -1
  38. data/spec/unit/provider/package/rpm_spec.rb +335 -124
  39. data/spec/unit/provider_resolver_spec.rb +0 -1
  40. data/spec/unit/recipe_spec.rb +12 -8
  41. data/spec/unit/resource_collection_spec.rb +1 -1
  42. data/spec/unit/resource_resolver_spec.rb +49 -0
  43. data/spec/unit/resource_spec.rb +19 -4
  44. data/spec/unit/role_spec.rb +1 -1
  45. data/spec/unit/run_list_spec.rb +1 -1
  46. data/spec/unit/runner_spec.rb +2 -2
  47. data/spec/unit/user_spec.rb +1 -1
  48. metadata +10 -8
  49. data/spec/support/pedant/Gemfile.lock +0 -67
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA1:
3
- metadata.gz: 3c86d339c90f91e7424eb62914aa079c88147bbf
4
- data.tar.gz: e46377e3e8d217dde44e76fcdad337c5dc905363
3
+ metadata.gz: bcb980c409f1addd0865425c6e6a2f2ca023392a
4
+ data.tar.gz: 397ce0b10766fd3d13482f68cd6949d9cc33caf3
5
5
  SHA512:
6
- metadata.gz: 10a7da378604081a93c7ce8b70cffe9c9e93379ffdbc1dcef15e7ffb0f5069c3b001db3de5c0063f8b1120622292a05a29daec093e207604c1f56b79b2d70579
7
- data.tar.gz: d2430bf003d81db3206c0f9bf74695a9b42118ce733c3a13a129dd53b7221fabd5c75b645dfaa7da5083a5536b156a824cca0535d5054212454971377af27df5
6
+ metadata.gz: e9fa764de5bd4b56e3b1faf5887e2dadd1b1b9a0a8fc590f2f451d8cffe535661f03634364c83686c60d8a8a925b593167ccd9169dfe7af9aed5ffd471264b9c
7
+ data.tar.gz: 4452f6a6cf351c1efa9d9f1e96bf913bc77893f0eca035921b78aa64a0535c03705002363b1910a2bad65a09d3ceb57f205194ba72bda5f625f6e0ae9d69c955
@@ -0,0 +1,327 @@
1
+ 
2
+ function Load-Win32Bindings {
3
+ Add-Type -TypeDefinition @"
4
+ using System;
5
+ using System.Diagnostics;
6
+ using System.Runtime.InteropServices;
7
+
8
+ namespace Chef
9
+ {
10
+
11
+ [StructLayout(LayoutKind.Sequential)]
12
+ public struct PROCESS_INFORMATION
13
+ {
14
+ public IntPtr hProcess;
15
+ public IntPtr hThread;
16
+ public uint dwProcessId;
17
+ public uint dwThreadId;
18
+ }
19
+
20
+ [StructLayout(LayoutKind.Sequential, CharSet = CharSet.Unicode)]
21
+ public struct STARTUPINFO
22
+ {
23
+ public uint cb;
24
+ public string lpReserved;
25
+ public string lpDesktop;
26
+ public string lpTitle;
27
+ public uint dwX;
28
+ public uint dwY;
29
+ public uint dwXSize;
30
+ public uint dwYSize;
31
+ public uint dwXCountChars;
32
+ public uint dwYCountChars;
33
+ public uint dwFillAttribute;
34
+ public STARTF dwFlags;
35
+ public ShowWindow wShowWindow;
36
+ public short cbReserved2;
37
+ public IntPtr lpReserved2;
38
+ public IntPtr hStdInput;
39
+ public IntPtr hStdOutput;
40
+ public IntPtr hStdError;
41
+ }
42
+
43
+ [StructLayout(LayoutKind.Sequential)]
44
+ public struct SECURITY_ATTRIBUTES
45
+ {
46
+ public int length;
47
+ public IntPtr lpSecurityDescriptor;
48
+ public bool bInheritHandle;
49
+ }
50
+
51
+ [Flags]
52
+ public enum CreationFlags : int
53
+ {
54
+ NONE = 0,
55
+ DEBUG_PROCESS = 0x00000001,
56
+ DEBUG_ONLY_THIS_PROCESS = 0x00000002,
57
+ CREATE_SUSPENDED = 0x00000004,
58
+ DETACHED_PROCESS = 0x00000008,
59
+ CREATE_NEW_CONSOLE = 0x00000010,
60
+ CREATE_NEW_PROCESS_GROUP = 0x00000200,
61
+ CREATE_UNICODE_ENVIRONMENT = 0x00000400,
62
+ CREATE_SEPARATE_WOW_VDM = 0x00000800,
63
+ CREATE_SHARED_WOW_VDM = 0x00001000,
64
+ CREATE_PROTECTED_PROCESS = 0x00040000,
65
+ EXTENDED_STARTUPINFO_PRESENT = 0x00080000,
66
+ CREATE_BREAKAWAY_FROM_JOB = 0x01000000,
67
+ CREATE_PRESERVE_CODE_AUTHZ_LEVEL = 0x02000000,
68
+ CREATE_DEFAULT_ERROR_MODE = 0x04000000,
69
+ CREATE_NO_WINDOW = 0x08000000,
70
+ }
71
+
72
+ [Flags]
73
+ public enum STARTF : uint
74
+ {
75
+ STARTF_USESHOWWINDOW = 0x00000001,
76
+ STARTF_USESIZE = 0x00000002,
77
+ STARTF_USEPOSITION = 0x00000004,
78
+ STARTF_USECOUNTCHARS = 0x00000008,
79
+ STARTF_USEFILLATTRIBUTE = 0x00000010,
80
+ STARTF_RUNFULLSCREEN = 0x00000020, // ignored for non-x86 platforms
81
+ STARTF_FORCEONFEEDBACK = 0x00000040,
82
+ STARTF_FORCEOFFFEEDBACK = 0x00000080,
83
+ STARTF_USESTDHANDLES = 0x00000100,
84
+ }
85
+
86
+ public enum ShowWindow : short
87
+ {
88
+ SW_HIDE = 0,
89
+ SW_SHOWNORMAL = 1,
90
+ SW_NORMAL = 1,
91
+ SW_SHOWMINIMIZED = 2,
92
+ SW_SHOWMAXIMIZED = 3,
93
+ SW_MAXIMIZE = 3,
94
+ SW_SHOWNOACTIVATE = 4,
95
+ SW_SHOW = 5,
96
+ SW_MINIMIZE = 6,
97
+ SW_SHOWMINNOACTIVE = 7,
98
+ SW_SHOWNA = 8,
99
+ SW_RESTORE = 9,
100
+ SW_SHOWDEFAULT = 10,
101
+ SW_FORCEMINIMIZE = 11,
102
+ SW_MAX = 11
103
+ }
104
+
105
+ public enum StandardHandle : int
106
+ {
107
+ Input = -10,
108
+ Output = -11,
109
+ Error = -12
110
+ }
111
+
112
+ public static class Kernel32
113
+ {
114
+ [DllImport("kernel32.dll", SetLastError=true)]
115
+ [return: MarshalAs(UnmanagedType.Bool)]
116
+ public static extern bool CreateProcess(
117
+ string lpApplicationName,
118
+ string lpCommandLine,
119
+ ref SECURITY_ATTRIBUTES lpProcessAttributes,
120
+ ref SECURITY_ATTRIBUTES lpThreadAttributes,
121
+ [MarshalAs(UnmanagedType.Bool)] bool bInheritHandles,
122
+ CreationFlags dwCreationFlags,
123
+ IntPtr lpEnvironment,
124
+ string lpCurrentDirectory,
125
+ ref STARTUPINFO lpStartupInfo,
126
+ out PROCESS_INFORMATION lpProcessInformation);
127
+
128
+ [DllImport("kernel32.dll", SetLastError=true)]
129
+ public static extern IntPtr GetStdHandle(
130
+ StandardHandle nStdHandle);
131
+
132
+ [DllImport("kernel32", SetLastError=true)]
133
+ public static extern int WaitForSingleObject(
134
+ IntPtr hHandle,
135
+ int dwMilliseconds);
136
+
137
+ [DllImport("kernel32", SetLastError=true)]
138
+ [return: MarshalAs(UnmanagedType.Bool)]
139
+ public static extern bool CloseHandle(
140
+ IntPtr hObject);
141
+
142
+ [DllImport("kernel32", SetLastError=true)]
143
+ [return: MarshalAs(UnmanagedType.Bool)]
144
+ public static extern bool GetExitCodeProcess(
145
+ IntPtr hProcess,
146
+ out int lpExitCode);
147
+ }
148
+ }
149
+ "@
150
+ }
151
+
152
+ function Run-ExecutableAndWait($AppPath, $ArgumentString) {
153
+ # Use the Win32 API to create a new process and wait for it to terminate.
154
+ $null = Load-Win32Bindings
155
+
156
+ $si = New-Object Chef.STARTUPINFO
157
+ $pi = New-Object Chef.PROCESS_INFORMATION
158
+
159
+ $si.cb = [System.Runtime.InteropServices.Marshal]::SizeOf($si)
160
+ $si.wShowWindow = [Chef.ShowWindow]::SW_SHOW
161
+ $si.dwFlags = [Chef.STARTF]::STARTF_USESTDHANDLES
162
+ $si.hStdError = [Chef.Kernel32]::GetStdHandle([Chef.StandardHandle]::Error)
163
+ $si.hStdOutput = [Chef.Kernel32]::GetStdHandle([Chef.StandardHandle]::Output)
164
+ $si.hStdInput = [Chef.Kernel32]::GetStdHandle([Chef.StandardHandle]::Input)
165
+
166
+ $pSec = New-Object Chef.SECURITY_ATTRIBUTES
167
+ $pSec.Length = [System.Runtime.InteropServices.Marshal]::SizeOf($pSec)
168
+ $pSec.bInheritHandle = $true
169
+ $tSec = New-Object Chef.SECURITY_ATTRIBUTES
170
+ $tSec.Length = [System.Runtime.InteropServices.Marshal]::SizeOf($tSec)
171
+ $tSec.bInheritHandle = $true
172
+
173
+ $success = [Chef.Kernel32]::CreateProcess($AppPath, $ArgumentString, [ref] $pSec, [ref] $tSec, $true, [Chef.CreationFlags]::NONE, [IntPtr]::Zero, $pwd, [ref] $si, [ref] $pi)
174
+ if (-Not $success) {
175
+ $reason = [System.Runtime.InteropServices.Marshal]::GetLastWin32Error()
176
+ throw "Unable to create process [$ArgumentString]. Error code $reason."
177
+ }
178
+ $waitReason = [Chef.Kernel32]::WaitForSingleObject($pi.hProcess, -1)
179
+ if ($waitReason -ne 0) {
180
+ if ($waitReason -eq -1) {
181
+ $reason = [System.Runtime.InteropServices.Marshal]::GetLastWin32Error()
182
+ throw "Could not wait for process to terminate. Error code $reason."
183
+ } else {
184
+ throw "WaitForSingleObject failed with return code $waitReason - it's impossible!"
185
+ }
186
+ }
187
+ $success = [Chef.Kernel32]::GetExitCodeProcess($pi.hProcess, [ref] $global:LASTEXITCODE)
188
+ if (-Not $success) {
189
+ $reason = [System.Runtime.InteropServices.Marshal]::GetLastWin32Error()
190
+ throw "Process exit code unavailable. Error code $reason."
191
+ }
192
+ $success = [Chef.Kernel32]::CloseHandle($pi.hProcess)
193
+ if (-Not $success) {
194
+ $reason = [System.Runtime.InteropServices.Marshal]::GetLastWin32Error()
195
+ throw "Unable to release process handle. Error code $reason."
196
+ }
197
+ $success = [Chef.Kernel32]::CloseHandle($pi.hThread)
198
+ if (-Not $success) {
199
+ $reason = [System.Runtime.InteropServices.Marshal]::GetLastWin32Error()
200
+ throw "Unable to release thread handle. Error code $reason."
201
+ }
202
+ }
203
+
204
+ function Get-ScriptDirectory {
205
+ if (!$PSScriptRoot) {
206
+ $Invocation = (Get-Variable MyInvocation -Scope 1).Value
207
+ $PSScriptRoot = Split-Path $Invocation.MyCommand.Path
208
+ }
209
+ $PSScriptRoot
210
+ }
211
+
212
+ function Run-RubyCommand($command, $argList) {
213
+ # This method exists to take the given list of arguments and get it past ruby's command-line
214
+ # interpreter unscathed and untampered. See https://github.com/ruby/ruby/blob/trunk/win32/win32.c#L1582
215
+ # for a list of transformations that ruby attempts to perform with your command-line arguments
216
+ # before passing it onto a script. The most important task is to defeat the globbing
217
+ # and wild-card expansion that ruby performs. Note that ruby does not use MSVCRT's argc/argv
218
+ # and deliberately reparses the raw command-line instead.
219
+ #
220
+ # To stop ruby from interpreting command-line arguments as globs, they need to be enclosed in '
221
+ # Ruby doesn't allow any escape characters inside '. This unfortunately prevents us from sending
222
+ # any strings which themselves contain '. Ruby does allow multi-fragment arguments though.
223
+ # "foo bar"'baz qux'123"foo" is interpreted as 1 argument because there are no un-escaped
224
+ # whitespace there. The argument would be interpreted as the string "foo barbaz qux123foo".
225
+ # This lets us escape ' characters by exiting the ' quoted string, injecting a "'" fragment and
226
+ # then resuming the ' quoted string again.
227
+ #
228
+ # In the process of defeating ruby, one must also defeat the helpfulness of powershell.
229
+ # When arguments come into this method, the standard PS rules for interpreting cmdlet arguments
230
+ # apply. When using & (call operator) and providing an array of arguments, powershell (verified
231
+ # on PS 4.0 on Windows Server 2012R2) will not evaluate them but (contrary to documentation),
232
+ # it will still marginally interpret them. The behaviour of PS 5.0 seems to be different but
233
+ # ignore that for now. If any of the provided arguments has a space in it, powershell checks
234
+ # the first and last character to ensure that they are " characters (and that's all it checks).
235
+ # If they are not, it will blindly surround that argument with " characters. It won't do this
236
+ # operation if no space is present, even if other special characters are present. If it notices
237
+ # leading and trailing " characters, it won't actually check to see if there are other "
238
+ # characters in the string. Since PS 5.0 changes this behavior, we could consider using the --%
239
+ # "stop screwing up my arguments" operator, which is available since PS 3.0. When encountered
240
+ # --% indicates that the rest of line is to be sent literally... except if the parser encounters
241
+ # %FOO% cmd style environment variables. Because reasons. And there is no way to escape the
242
+ # % character in *any* waym shape or form.
243
+ # https://connect.microsoft.com/PowerShell/feedback/details/376207/executing-commands-which-require-quotes-and-variables-is-practically-impossible
244
+ #
245
+ # In case you think that you're either reading this incorrectly or that I'm full of shit, here
246
+ # are some examples. These use EchoArgs.exe from the PowerShell Community Extensions package.
247
+ # I have not included the argument parsing output from EchoArgs.exe to prevent confusing you with
248
+ # more details about MSVCRT's parsing algorithm.
249
+ #
250
+ # $x = "foo '' bar `"baz`""
251
+ # & EchoArgs @($x, $x)
252
+ # Command line:
253
+ # "C:\Program Files (x86)\PowerShell Community Extensions\Pscx3\Pscx\Apps\EchoArgs.exe" "foo '' bar "baz"" "foo '' bar "baz""
254
+ #
255
+ # $x = "abc'123'nospace`"lulz`"!!!"
256
+ # & EchoArgs @($x, $x)
257
+ # Command line:
258
+ # "C:\Program Files (x86)\PowerShell Community Extensions\Pscx3\Pscx\Apps\EchoArgs.exe" abc'123'nospace"lulz"!!! abc'123'nospace"lulz"!!!
259
+ #
260
+ # $x = "`"`"Look ma! Tonnes of spaces! 'foo' 'bar'`"`""
261
+ # & EchoArgs @($x, $x)
262
+ # Command line:
263
+ # "C:\Program Files (x86)\PowerShell Community Extensions\Pscx3\Pscx\Apps\EchoArgs.exe" ""Look ma! Tonnes of spaces! 'foo' 'bar'"" ""Look ma! Tonnes of spaces! 'foo' 'bar'""
264
+ #
265
+ # Given all this, we can now device a strategy to work around all these immensely helpful, well
266
+ # documented and useful tools by looking at each incoming argument, escaping any ' characters
267
+ # with a '"'"' sequence, surrounding each argument with ' & joining them with a space separating
268
+ # them.
269
+ # There is another bug (https://bugs.ruby-lang.org/issues/11142) that causes ruby to mangle any
270
+ # "" two-character double quote sequence but since we always emit our strings inside ' except for
271
+ # ' characters, this should be ok. Just remember that an argument '' should get translated to
272
+ # ''"'"''"'"'' on the command line. If those intervening empty ''s are not present, the presence
273
+ # of "" will cause ruby to mangle that argument.
274
+ $transformedList = $argList | foreach { "'" + ( $_ -replace "'","'`"'`"'" ) + "'" }
275
+ $fortifiedArgString = $transformedList -join ' '
276
+
277
+ # Use the correct embedded ruby path. We'll be deployed at a path that looks like
278
+ # [C:\opscode or some other prefix]\chef\modules\chef
279
+ $ruby = Join-Path (Get-ScriptDirectory) "..\..\embedded\bin\ruby.exe"
280
+ $commandPath = Join-Path (Get-ScriptDirectory) "..\..\bin\$command"
281
+
282
+ Run-ExecutableAndWait $ruby """$ruby"" '$commandPath' $fortifiedArgString"
283
+ }
284
+
285
+
286
+ function chef-apply {
287
+ Run-RubyCommand 'chef-apply' $args
288
+ }
289
+
290
+ function chef-client {
291
+ Run-RubyCommand 'chef-client' $args
292
+ }
293
+
294
+ function chef-service-manager {
295
+ Run-RubyCommand 'chef-service-manager' $args
296
+ }
297
+
298
+ function chef-shell {
299
+ Run-RubyCommand 'chef-shell' $args
300
+ }
301
+
302
+ function chef-solo {
303
+ Run-RubyCommand 'chef-solo' $args
304
+ }
305
+
306
+ function chef-windows-service {
307
+ Run-RubyCommand 'chef-windows-service' $args
308
+ }
309
+
310
+ function knife {
311
+ Run-RubyCommand 'knife' $args
312
+ }
313
+
314
+ Export-ModuleMember -function chef-apply
315
+ Export-ModuleMember -function chef-client
316
+ Export-ModuleMember -function chef-service-manager
317
+ Export-ModuleMember -function chef-shell
318
+ Export-ModuleMember -function chef-solo
319
+ Export-ModuleMember -function chef-windows-service
320
+ Export-ModuleMember -function knife
321
+
322
+ # To debug this module, uncomment the line below and then run the following.
323
+ # Export-ModuleMember -function Run-RubyCommand
324
+ # Remove-Module chef
325
+ # Import-Module chef
326
+ # "puts ARGV" | Out-File C:\opscode\chef\bin\puts_args
327
+ # Run-RubyCommand puts_args 'Here' "are" some '"very interesting"' 'arguments[to]' "`"try out`""
@@ -58,7 +58,7 @@ class Chef
58
58
  # @return [Array<Class>] Priority Array of Provider Classes to use for the resource_name on the node
59
59
  #
60
60
  def get_provider_priority_array(resource_name)
61
- result = provider_priority_map.get_priority_array(node, resource_name)
61
+ result = provider_priority_map.get_priority_array(node, resource_name.to_sym)
62
62
  result = result.dup if result
63
63
  result
64
64
  end
@@ -71,7 +71,7 @@ class Chef
71
71
  # @return [Array<Class>] Priority Array of Resource Classes to use for the resource_name on the node
72
72
  #
73
73
  def get_resource_priority_array(resource_name)
74
- result = resource_priority_map.get_priority_array(node, resource_name)
74
+ result = resource_priority_map.get_priority_array(node, resource_name.to_sym)
75
75
  result = result.dup if result
76
76
  result
77
77
  end
@@ -86,7 +86,7 @@ class Chef
86
86
  # @return [Array<Class>] Modified Priority Array of Provider Classes to use for the resource_name on the node
87
87
  #
88
88
  def set_provider_priority_array(resource_name, priority_array, *filter, &block)
89
- result = provider_priority_map.set_priority_array(resource_name, priority_array, *filter, &block)
89
+ result = provider_priority_map.set_priority_array(resource_name.to_sym, priority_array, *filter, &block)
90
90
  result = result.dup if result
91
91
  result
92
92
  end
@@ -101,7 +101,7 @@ class Chef
101
101
  # @return [Array<Class>] Modified Priority Array of Resource Classes to use for the resource_name on the node
102
102
  #
103
103
  def set_resource_priority_array(resource_name, priority_array, *filter, &block)
104
- result = resource_priority_map.set_priority_array(resource_name, priority_array, *filter, &block)
104
+ result = resource_priority_map.set_priority_array(resource_name.to_sym, priority_array, *filter, &block)
105
105
  result = result.dup if result
106
106
  result
107
107
  end
data/lib/chef/client.rb CHANGED
@@ -309,12 +309,18 @@ class Chef
309
309
  # with the proper exit status code and everything gets raised
310
310
  # as a RunFailedWrappingError
311
311
  if run_error || converge_error || audit_error
312
- error = if run_error == converge_error
313
- Chef::Exceptions::RunFailedWrappingError.new(converge_error, audit_error)
314
- else
315
- Chef::Exceptions::RunFailedWrappingError.new(run_error, converge_error, audit_error)
316
- end
317
- error.fill_backtrace
312
+ error = if Chef::Config[:audit_mode] == :disabled
313
+ run_error || converge_error
314
+ else
315
+ e = if run_error == converge_error
316
+ Chef::Exceptions::RunFailedWrappingError.new(converge_error, audit_error)
317
+ else
318
+ Chef::Exceptions::RunFailedWrappingError.new(run_error, converge_error, audit_error)
319
+ end
320
+ e.fill_backtrace
321
+ e
322
+ end
323
+
318
324
  Chef::Application.debug_stacktrace(error)
319
325
  raise error
320
326
  end
data/lib/chef/node_map.rb CHANGED
@@ -38,30 +38,35 @@ class Chef
38
38
  #
39
39
  # @return [NodeMap] Returns self for possible chaining
40
40
  #
41
- def set(key, value, platform: nil, platform_version: nil, platform_family: nil, os: nil, on_platform: nil, on_platforms: nil, canonical: nil, &block)
41
+ def set(key, value, platform: nil, platform_version: nil, platform_family: nil, os: nil, on_platform: nil, on_platforms: nil, canonical: nil, override: nil, &block)
42
42
  Chef::Log.deprecation "The on_platform option to node_map has been deprecated" if on_platform
43
43
  Chef::Log.deprecation "The on_platforms option to node_map has been deprecated" if on_platforms
44
44
  platform ||= on_platform || on_platforms
45
- filters = { platform: platform, platform_version: platform_version, platform_family: platform_family, os: os }
46
- new_matcher = { filters: filters, block: block, value: value, canonical: canonical }
47
- @map[key] ||= []
48
- # Decide where to insert the matcher; the new value is preferred over
49
- # anything more specific (see `priority_of`) and is preferred over older
50
- # values of the same specificity. (So all other things being equal,
51
- # newest wins.)
45
+ filters = {}
46
+ filters[:platform] = platform if platform
47
+ filters[:platform_version] = platform_version if platform_version
48
+ filters[:platform_family] = platform_family if platform_family
49
+ filters[:os] = os if os
50
+ new_matcher = { value: value, filters: filters }
51
+ new_matcher[:block] = block if block
52
+ new_matcher[:canonical] = canonical if canonical
53
+ new_matcher[:override] = override if override
54
+
55
+ # The map is sorted in order of preference already; we just need to find
56
+ # our place in it (just before the first value with the same preference level).
52
57
  insert_at = nil
53
- @map[key].each_with_index do |matcher, index|
54
- if specificity(new_matcher) >= specificity(matcher)
55
- insert_at = index
56
- break
57
- end
58
+ @map[key] ||= []
59
+ @map[key].each_with_index do |matcher,index|
60
+ cmp = compare_matchers(key, new_matcher, matcher)
61
+ insert_at ||= index if cmp && cmp <= 0
58
62
  end
59
63
  if insert_at
60
64
  @map[key].insert(insert_at, new_matcher)
61
65
  else
62
66
  @map[key] << new_matcher
63
67
  end
64
- self
68
+ insert_at ||= @map[key].size - 1
69
+ @map
65
70
  end
66
71
 
67
72
  #
@@ -116,30 +121,7 @@ class Chef
116
121
  remaining
117
122
  end
118
123
 
119
- private
120
-
121
- #
122
- # Gives a value for "how specific" the matcher is.
123
- # Things which specify more specific filters get a higher number
124
- # (platform_version > platform > platform_family > os); things
125
- # with a block have higher specificity than similar things without
126
- # a block.
127
- #
128
- def specificity(matcher)
129
- if matcher[:filters][:platform_version]
130
- specificity = 8
131
- elsif matcher[:filters][:platform]
132
- specificity = 6
133
- elsif matcher[:filters][:platform_family]
134
- specificity = 4
135
- elsif matcher[:filters][:os]
136
- specificity = 2
137
- else
138
- specificity = 0
139
- end
140
- specificity += 1 if matcher[:block]
141
- specificity
142
- end
124
+ protected
143
125
 
144
126
  #
145
127
  # Succeeds if:
@@ -197,5 +179,48 @@ class Chef
197
179
  return true if canonical.nil?
198
180
  !!canonical == !!matcher[:canonical]
199
181
  end
182
+
183
+ def compare_matchers(key, new_matcher, matcher)
184
+ cmp = compare_matcher_properties(new_matcher, matcher) { |m| m[:filters][:block] }
185
+ return cmp if cmp != 0
186
+ cmp = compare_matcher_properties(new_matcher, matcher) { |m| m[:filters][:platform_version] }
187
+ return cmp if cmp != 0
188
+ cmp = compare_matcher_properties(new_matcher, matcher) { |m| m[:filters][:platform] }
189
+ return cmp if cmp != 0
190
+ cmp = compare_matcher_properties(new_matcher, matcher) { |m| m[:filters][:platform_family] }
191
+ return cmp if cmp != 0
192
+ cmp = compare_matcher_properties(new_matcher, matcher) { |m| m[:filters][:os] }
193
+ return cmp if cmp != 0
194
+ cmp = compare_matcher_properties(new_matcher, matcher) { |m| m[:override] }
195
+ return cmp if cmp != 0
196
+ # If all things are identical, return 0
197
+ 0
198
+ end
199
+
200
+ def compare_matcher_properties(new_matcher, matcher)
201
+ a = yield(new_matcher)
202
+ b = yield(matcher)
203
+
204
+ # Check for blcacklists ('!windows'). Those always come *after* positive
205
+ # whitelists.
206
+ a_negated = Array(a).any? { |f| f.is_a?(String) && f.start_with?('!') }
207
+ b_negated = Array(b).any? { |f| f.is_a?(String) && f.start_with?('!') }
208
+ if a_negated != b_negated
209
+ return 1 if a_negated
210
+ return -1 if b_negated
211
+ end
212
+
213
+ # We treat false / true and nil / not-nil with the same comparison
214
+ a = nil if a == false
215
+ b = nil if b == false
216
+ cmp = a <=> b
217
+ # This is the case where one is non-nil, and one is nil. The one that is
218
+ # nil is "greater" (i.e. it should come last).
219
+ if cmp.nil?
220
+ return 1 if a.nil?
221
+ return -1 if b.nil?
222
+ end
223
+ cmp
224
+ end
200
225
  end
201
226
  end