chef 11.14.6-x86-mingw32 → 11.16.0-x86-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.
- checksums.yaml +4 -4
 - data/lib/chef/exceptions.rb +4 -0
 - data/lib/chef/mixin/windows_architecture_helper.rb +16 -0
 - data/lib/chef/platform/query_helpers.rb +5 -1
 - data/lib/chef/provider/dsc_script.rb +148 -0
 - data/lib/chef/provider/user/dscl.rb +32 -28
 - data/lib/chef/providers.rb +1 -0
 - data/lib/chef/resource/dsc_script.rb +140 -0
 - data/lib/chef/resources.rb +1 -0
 - data/lib/chef/util/dsc/configuration_generator.rb +115 -0
 - data/lib/chef/util/dsc/lcm_output_parser.rb +133 -0
 - data/lib/chef/util/dsc/local_configuration_manager.rb +137 -0
 - data/lib/chef/util/dsc/resource_info.rb +26 -0
 - data/lib/chef/util/path_helper.rb +2 -2
 - data/lib/chef/util/powershell/cmdlet.rb +136 -0
 - data/lib/chef/util/powershell/cmdlet_result.rb +46 -0
 - data/lib/chef/version.rb +1 -1
 - data/spec/functional/resource/dsc_script_spec.rb +337 -0
 - data/spec/functional/resource/group_spec.rb +5 -1
 - data/spec/functional/util/powershell/cmdlet_spec.rb +114 -0
 - data/spec/spec_helper.rb +3 -0
 - data/spec/support/platform_helpers.rb +24 -0
 - data/spec/unit/platform/query_helpers_spec.rb +23 -0
 - data/spec/unit/provider/dsc_script_spec.rb +145 -0
 - data/spec/unit/provider/user/dscl_spec.rb +2 -1
 - data/spec/unit/resource/dsc_script_spec.rb +127 -0
 - data/spec/unit/util/dsc/configuration_generator_spec.rb +171 -0
 - data/spec/unit/util/dsc/lcm_output_parser_spec.rb +169 -0
 - data/spec/unit/util/dsc/local_configuration_manager_spec.rb +134 -0
 - data/spec/unit/util/powershell/cmdlet_spec.rb +106 -0
 - metadata +20 -4
 
    
        checksums.yaml
    CHANGED
    
    | 
         @@ -1,7 +1,7 @@ 
     | 
|
| 
       1 
1 
     | 
    
         
             
            ---
         
     | 
| 
       2 
2 
     | 
    
         
             
            SHA1:
         
     | 
| 
       3 
     | 
    
         
            -
              metadata.gz:  
     | 
| 
       4 
     | 
    
         
            -
              data.tar.gz:  
     | 
| 
      
 3 
     | 
    
         
            +
              metadata.gz: 49b6fe3b7b3cc9463d7faa2c42c063805a8c0622
         
     | 
| 
      
 4 
     | 
    
         
            +
              data.tar.gz: bf2edeaa5447924dc2f5d6c4f21409f99e268f03
         
     | 
| 
       5 
5 
     | 
    
         
             
            SHA512:
         
     | 
| 
       6 
     | 
    
         
            -
              metadata.gz:  
     | 
| 
       7 
     | 
    
         
            -
              data.tar.gz:  
     | 
| 
      
 6 
     | 
    
         
            +
              metadata.gz: edd6c14fb435fc1c2ee3f19ae061221c3f895b650b77f44f5142c5ba6b0af671e7205535c76a90f23d24c4e96fbd5e45fb1eeb5f7e0a754733274d9e770b8057
         
     | 
| 
      
 7 
     | 
    
         
            +
              data.tar.gz: 8f044ddd62016822cc9fdbfd3d80c9d1598b34aff1af8c0274bdfcd5339eebe61bdca9d0bd25c53cb114eded9a62fd45d412752eec5ff6064c2516326e71e986
         
     | 
    
        data/lib/chef/exceptions.rb
    CHANGED
    
    | 
         @@ -117,6 +117,8 @@ class Chef 
     | 
|
| 
       117 
117 
     | 
    
         
             
                class ObsoleteDependencySyntax < ArgumentError; end
         
     | 
| 
       118 
118 
     | 
    
         
             
                class InvalidDataBagPath < ArgumentError; end
         
     | 
| 
       119 
119 
     | 
    
         | 
| 
      
 120 
     | 
    
         
            +
                class PowershellCmdletException < RuntimeError; end
         
     | 
| 
      
 121 
     | 
    
         
            +
             
     | 
| 
       120 
122 
     | 
    
         
             
                # A different version of a cookbook was added to a
         
     | 
| 
       121 
123 
     | 
    
         
             
                # VersionedRecipeList than the one already there.
         
     | 
| 
       122 
124 
     | 
    
         
             
                class CookbookVersionConflict < ArgumentError ; end
         
     | 
| 
         @@ -176,6 +178,8 @@ class Chef 
     | 
|
| 
       176 
178 
     | 
    
         | 
| 
       177 
179 
     | 
    
         
             
                class ChildConvergeError < RuntimeError; end
         
     | 
| 
       178 
180 
     | 
    
         | 
| 
      
 181 
     | 
    
         
            +
                class NoProviderAvailable < RuntimeError; end
         
     | 
| 
      
 182 
     | 
    
         
            +
             
     | 
| 
       179 
183 
     | 
    
         
             
                class MissingRole < RuntimeError
         
     | 
| 
       180 
184 
     | 
    
         
             
                  NULL = Object.new
         
     | 
| 
       181 
185 
     | 
    
         | 
| 
         @@ -41,6 +41,22 @@ class Chef 
     | 
|
| 
       41 
41 
     | 
    
         
             
                      is_i386_process_on_x86_64_windows?
         
     | 
| 
       42 
42 
     | 
    
         
             
                  end
         
     | 
| 
       43 
43 
     | 
    
         | 
| 
      
 44 
     | 
    
         
            +
                  def with_os_architecture(node)
         
     | 
| 
      
 45 
     | 
    
         
            +
                    wow64_redirection_state = nil
         
     | 
| 
      
 46 
     | 
    
         
            +
             
     | 
| 
      
 47 
     | 
    
         
            +
                    if wow64_architecture_override_required?(node, node_windows_architecture(node))
         
     | 
| 
      
 48 
     | 
    
         
            +
                      wow64_redirection_state = disable_wow64_file_redirection(node)
         
     | 
| 
      
 49 
     | 
    
         
            +
                    end
         
     | 
| 
      
 50 
     | 
    
         
            +
             
     | 
| 
      
 51 
     | 
    
         
            +
                    begin
         
     | 
| 
      
 52 
     | 
    
         
            +
                      yield
         
     | 
| 
      
 53 
     | 
    
         
            +
                    ensure
         
     | 
| 
      
 54 
     | 
    
         
            +
                      if wow64_redirection_state
         
     | 
| 
      
 55 
     | 
    
         
            +
                        restore_wow64_file_redirection(node, wow64_redirection_state)
         
     | 
| 
      
 56 
     | 
    
         
            +
                      end
         
     | 
| 
      
 57 
     | 
    
         
            +
                    end
         
     | 
| 
      
 58 
     | 
    
         
            +
                  end
         
     | 
| 
      
 59 
     | 
    
         
            +
             
     | 
| 
       44 
60 
     | 
    
         
             
                  def node_supports_windows_architecture?(node, desired_architecture)
         
     | 
| 
       45 
61 
     | 
    
         
             
                    assert_valid_windows_architecture!(desired_architecture)
         
     | 
| 
       46 
62 
     | 
    
         
             
                    return (node_windows_architecture(node) == :x86_64 ||
         
     | 
| 
         @@ -0,0 +1,148 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            #
         
     | 
| 
      
 2 
     | 
    
         
            +
            # Author:: Adam Edwards (<adamed@getchef.com>)
         
     | 
| 
      
 3 
     | 
    
         
            +
            #
         
     | 
| 
      
 4 
     | 
    
         
            +
            # Copyright:: 2014, Chef Software, Inc.
         
     | 
| 
      
 5 
     | 
    
         
            +
            #
         
     | 
| 
      
 6 
     | 
    
         
            +
            # Licensed under the Apache License, Version 2.0 (the "License");
         
     | 
| 
      
 7 
     | 
    
         
            +
            # you may not use this file except in compliance with the License.
         
     | 
| 
      
 8 
     | 
    
         
            +
            # You may obtain a copy of the License at
         
     | 
| 
      
 9 
     | 
    
         
            +
            #
         
     | 
| 
      
 10 
     | 
    
         
            +
            #     http://www.apache.org/licenses/LICENSE-2.0
         
     | 
| 
      
 11 
     | 
    
         
            +
            #
         
     | 
| 
      
 12 
     | 
    
         
            +
            # Unless required by applicable law or agreed to in writing, software
         
     | 
| 
      
 13 
     | 
    
         
            +
            # distributed under the License is distributed on an "AS IS" BASIS,
         
     | 
| 
      
 14 
     | 
    
         
            +
            # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
         
     | 
| 
      
 15 
     | 
    
         
            +
            # See the License for the specific language governing permissions and
         
     | 
| 
      
 16 
     | 
    
         
            +
            # limitations under the License.
         
     | 
| 
      
 17 
     | 
    
         
            +
            #
         
     | 
| 
      
 18 
     | 
    
         
            +
             
     | 
| 
      
 19 
     | 
    
         
            +
            require 'chef/util/powershell/cmdlet'
         
     | 
| 
      
 20 
     | 
    
         
            +
            require 'chef/util/dsc/configuration_generator'
         
     | 
| 
      
 21 
     | 
    
         
            +
            require 'chef/util/dsc/local_configuration_manager'
         
     | 
| 
      
 22 
     | 
    
         
            +
             
     | 
| 
      
 23 
     | 
    
         
            +
            class Chef
         
     | 
| 
      
 24 
     | 
    
         
            +
              class Provider
         
     | 
| 
      
 25 
     | 
    
         
            +
                class DscScript < Chef::Provider
         
     | 
| 
      
 26 
     | 
    
         
            +
                  def initialize(dsc_resource, run_context)
         
     | 
| 
      
 27 
     | 
    
         
            +
                    super(dsc_resource, run_context)
         
     | 
| 
      
 28 
     | 
    
         
            +
                    @dsc_resource = dsc_resource
         
     | 
| 
      
 29 
     | 
    
         
            +
                    @resource_converged = false
         
     | 
| 
      
 30 
     | 
    
         
            +
                    @operations = {
         
     | 
| 
      
 31 
     | 
    
         
            +
                      :set => Proc.new { |config_manager, document|
         
     | 
| 
      
 32 
     | 
    
         
            +
                        config_manager.set_configuration(document)
         
     | 
| 
      
 33 
     | 
    
         
            +
                      },
         
     | 
| 
      
 34 
     | 
    
         
            +
                      :test => Proc.new { |config_manager, document|
         
     | 
| 
      
 35 
     | 
    
         
            +
                        config_manager.test_configuration(document)
         
     | 
| 
      
 36 
     | 
    
         
            +
                      }}
         
     | 
| 
      
 37 
     | 
    
         
            +
                  end
         
     | 
| 
      
 38 
     | 
    
         
            +
             
     | 
| 
      
 39 
     | 
    
         
            +
                  def action_run
         
     | 
| 
      
 40 
     | 
    
         
            +
                    if ! @resource_converged
         
     | 
| 
      
 41 
     | 
    
         
            +
                      converge_by(generate_description) do
         
     | 
| 
      
 42 
     | 
    
         
            +
                        run_configuration(:set)
         
     | 
| 
      
 43 
     | 
    
         
            +
                        Chef::Log.info("DSC resource configuration completed successfully")
         
     | 
| 
      
 44 
     | 
    
         
            +
                      end
         
     | 
| 
      
 45 
     | 
    
         
            +
                    end
         
     | 
| 
      
 46 
     | 
    
         
            +
                  end
         
     | 
| 
      
 47 
     | 
    
         
            +
             
     | 
| 
      
 48 
     | 
    
         
            +
                  def load_current_resource
         
     | 
| 
      
 49 
     | 
    
         
            +
                    @dsc_resources_info = run_configuration(:test)
         
     | 
| 
      
 50 
     | 
    
         
            +
                    @resource_converged = @dsc_resources_info.all? do |resource|
         
     | 
| 
      
 51 
     | 
    
         
            +
                      !resource.changes_state?
         
     | 
| 
      
 52 
     | 
    
         
            +
                    end
         
     | 
| 
      
 53 
     | 
    
         
            +
                  end
         
     | 
| 
      
 54 
     | 
    
         
            +
             
     | 
| 
      
 55 
     | 
    
         
            +
                  def whyrun_supported?
         
     | 
| 
      
 56 
     | 
    
         
            +
                    true
         
     | 
| 
      
 57 
     | 
    
         
            +
                  end
         
     | 
| 
      
 58 
     | 
    
         
            +
             
     | 
| 
      
 59 
     | 
    
         
            +
                  protected
         
     | 
| 
      
 60 
     | 
    
         
            +
             
     | 
| 
      
 61 
     | 
    
         
            +
                  def run_configuration(operation)
         
     | 
| 
      
 62 
     | 
    
         
            +
                    config_directory = ::Dir.mktmpdir("chef-dsc-script")
         
     | 
| 
      
 63 
     | 
    
         
            +
                    configuration_data_path = get_configuration_data_path(config_directory)
         
     | 
| 
      
 64 
     | 
    
         
            +
                    configuration_flags = get_augmented_configuration_flags(configuration_data_path)
         
     | 
| 
      
 65 
     | 
    
         
            +
             
     | 
| 
      
 66 
     | 
    
         
            +
                    config_manager = Chef::Util::DSC::LocalConfigurationManager.new(@run_context.node, config_directory)
         
     | 
| 
      
 67 
     | 
    
         
            +
             
     | 
| 
      
 68 
     | 
    
         
            +
                    begin
         
     | 
| 
      
 69 
     | 
    
         
            +
                      configuration_document = generate_configuration_document(config_directory, configuration_flags)
         
     | 
| 
      
 70 
     | 
    
         
            +
                      @operations[operation].call(config_manager, configuration_document)
         
     | 
| 
      
 71 
     | 
    
         
            +
                    rescue Exception => e
         
     | 
| 
      
 72 
     | 
    
         
            +
                      Chef::Log.error("DSC operation failed: #{e.message.to_s}")
         
     | 
| 
      
 73 
     | 
    
         
            +
                      raise e
         
     | 
| 
      
 74 
     | 
    
         
            +
                    ensure
         
     | 
| 
      
 75 
     | 
    
         
            +
                      ::FileUtils.rm_rf(config_directory)
         
     | 
| 
      
 76 
     | 
    
         
            +
                    end
         
     | 
| 
      
 77 
     | 
    
         
            +
                  end
         
     | 
| 
      
 78 
     | 
    
         
            +
             
     | 
| 
      
 79 
     | 
    
         
            +
                  def get_augmented_configuration_flags(configuration_data_path)
         
     | 
| 
      
 80 
     | 
    
         
            +
                    updated_flags = nil
         
     | 
| 
      
 81 
     | 
    
         
            +
                    if configuration_data_path
         
     | 
| 
      
 82 
     | 
    
         
            +
                      updated_flags = @dsc_resource.flags.nil? ? {} : @dsc_resource.flags.dup
         
     | 
| 
      
 83 
     | 
    
         
            +
                      Chef::Util::PathHelper.validate_path(configuration_data_path)
         
     | 
| 
      
 84 
     | 
    
         
            +
                      updated_flags[:configurationdata] = configuration_data_path
         
     | 
| 
      
 85 
     | 
    
         
            +
                    end
         
     | 
| 
      
 86 
     | 
    
         
            +
                    updated_flags
         
     | 
| 
      
 87 
     | 
    
         
            +
                  end
         
     | 
| 
      
 88 
     | 
    
         
            +
             
     | 
| 
      
 89 
     | 
    
         
            +
                  def generate_configuration_document(config_directory, configuration_flags)
         
     | 
| 
      
 90 
     | 
    
         
            +
                    shellout_flags = {
         
     | 
| 
      
 91 
     | 
    
         
            +
                      :cwd => @dsc_resource.cwd,
         
     | 
| 
      
 92 
     | 
    
         
            +
                      :environment => @dsc_resource.environment,
         
     | 
| 
      
 93 
     | 
    
         
            +
                      :timeout => @dsc_resource.timeout
         
     | 
| 
      
 94 
     | 
    
         
            +
                    }
         
     | 
| 
      
 95 
     | 
    
         
            +
             
     | 
| 
      
 96 
     | 
    
         
            +
                    generator = Chef::Util::DSC::ConfigurationGenerator.new(@run_context.node, config_directory)
         
     | 
| 
      
 97 
     | 
    
         
            +
             
     | 
| 
      
 98 
     | 
    
         
            +
                    if @dsc_resource.command
         
     | 
| 
      
 99 
     | 
    
         
            +
                      generator.configuration_document_from_script_path(@dsc_resource.command, configuration_name, configuration_flags, shellout_flags)
         
     | 
| 
      
 100 
     | 
    
         
            +
                    else
         
     | 
| 
      
 101 
     | 
    
         
            +
                      # If code is also not provided, we mimic what the other script resources do (execute nothing)
         
     | 
| 
      
 102 
     | 
    
         
            +
                      Chef::Log.warn("Neither code or command were provided for dsc_resource[#{@dsc_resource.name}].") unless @dsc_resource.code
         
     | 
| 
      
 103 
     | 
    
         
            +
                      generator.configuration_document_from_script_code(@dsc_resource.code || '', configuration_flags, shellout_flags)
         
     | 
| 
      
 104 
     | 
    
         
            +
                    end
         
     | 
| 
      
 105 
     | 
    
         
            +
                  end
         
     | 
| 
      
 106 
     | 
    
         
            +
             
     | 
| 
      
 107 
     | 
    
         
            +
                  def get_configuration_data_path(config_directory)
         
     | 
| 
      
 108 
     | 
    
         
            +
                    if @dsc_resource.configuration_data_script
         
     | 
| 
      
 109 
     | 
    
         
            +
                      @dsc_resource.configuration_data_script
         
     | 
| 
      
 110 
     | 
    
         
            +
                    elsif @dsc_resource.configuration_data
         
     | 
| 
      
 111 
     | 
    
         
            +
                      configuration_data_path = "#{config_directory}/chef_dsc_config_data.psd1"
         
     | 
| 
      
 112 
     | 
    
         
            +
                      ::File.open(configuration_data_path, 'wt') do | script |
         
     | 
| 
      
 113 
     | 
    
         
            +
                        script.write(@dsc_resource.configuration_data)
         
     | 
| 
      
 114 
     | 
    
         
            +
                      end
         
     | 
| 
      
 115 
     | 
    
         
            +
                      configuration_data_path
         
     | 
| 
      
 116 
     | 
    
         
            +
                    end
         
     | 
| 
      
 117 
     | 
    
         
            +
                  end
         
     | 
| 
      
 118 
     | 
    
         
            +
             
     | 
| 
      
 119 
     | 
    
         
            +
                  def configuration_name
         
     | 
| 
      
 120 
     | 
    
         
            +
                    @dsc_resource.configuration_name || @dsc_resource.name
         
     | 
| 
      
 121 
     | 
    
         
            +
                  end
         
     | 
| 
      
 122 
     | 
    
         
            +
             
     | 
| 
      
 123 
     | 
    
         
            +
                  def configuration_friendly_name
         
     | 
| 
      
 124 
     | 
    
         
            +
                    if @dsc_resource.code
         
     | 
| 
      
 125 
     | 
    
         
            +
                      @dsc_resource.name
         
     | 
| 
      
 126 
     | 
    
         
            +
                    else
         
     | 
| 
      
 127 
     | 
    
         
            +
                      configuration_name
         
     | 
| 
      
 128 
     | 
    
         
            +
                    end
         
     | 
| 
      
 129 
     | 
    
         
            +
                  end
         
     | 
| 
      
 130 
     | 
    
         
            +
             
     | 
| 
      
 131 
     | 
    
         
            +
                  private
         
     | 
| 
      
 132 
     | 
    
         
            +
             
     | 
| 
      
 133 
     | 
    
         
            +
                  def generate_description
         
     | 
| 
      
 134 
     | 
    
         
            +
                    ["converge DSC configuration '#{configuration_friendly_name}'"] +
         
     | 
| 
      
 135 
     | 
    
         
            +
                      @dsc_resources_info.map do |resource|
         
     | 
| 
      
 136 
     | 
    
         
            +
                        if resource.changes_state?
         
     | 
| 
      
 137 
     | 
    
         
            +
                          # We ignore the last log message because it only contains the time it took, which looks weird
         
     | 
| 
      
 138 
     | 
    
         
            +
                          cleaned_messages = resource.change_log[0..-2].map { |c| c.sub(/^#{Regexp.escape(resource.name)}/, '').strip }
         
     | 
| 
      
 139 
     | 
    
         
            +
                          "converge DSC resource #{resource.name} by #{cleaned_messages.find_all{ |c| c != ''}.join("\n")}"
         
     | 
| 
      
 140 
     | 
    
         
            +
                        else
         
     | 
| 
      
 141 
     | 
    
         
            +
                          # This is needed because a dsc script can have resouces that are both converged and not
         
     | 
| 
      
 142 
     | 
    
         
            +
                          "converge DSC resource #{resource.name} by doing nothing because it is already converged"
         
     | 
| 
      
 143 
     | 
    
         
            +
                        end
         
     | 
| 
      
 144 
     | 
    
         
            +
                      end
         
     | 
| 
      
 145 
     | 
    
         
            +
                  end
         
     | 
| 
      
 146 
     | 
    
         
            +
                end
         
     | 
| 
      
 147 
     | 
    
         
            +
              end
         
     | 
| 
      
 148 
     | 
    
         
            +
            end
         
     | 
| 
         @@ -110,18 +110,18 @@ user password using shadow hash.") 
     | 
|
| 
       110 
110 
     | 
    
         
             
                      @current_resource = Chef::Resource::User.new(@new_resource.username)
         
     | 
| 
       111 
111 
     | 
    
         
             
                      @current_resource.username(@new_resource.username)
         
     | 
| 
       112 
112 
     | 
    
         | 
| 
       113 
     | 
    
         
            -
                      user_info = read_user_info
         
     | 
| 
       114 
     | 
    
         
            -
                      if user_info
         
     | 
| 
       115 
     | 
    
         
            -
                        @current_resource.uid(dscl_get(user_info, :uid))
         
     | 
| 
       116 
     | 
    
         
            -
                        @current_resource.gid(dscl_get(user_info, :gid))
         
     | 
| 
       117 
     | 
    
         
            -
                        @current_resource.home(dscl_get(user_info, :home))
         
     | 
| 
       118 
     | 
    
         
            -
                        @current_resource.shell(dscl_get(user_info, :shell))
         
     | 
| 
       119 
     | 
    
         
            -
                        @current_resource.comment(dscl_get(user_info, :comment))
         
     | 
| 
       120 
     | 
    
         
            -
                        @authentication_authority = dscl_get(user_info, :auth_authority)
         
     | 
| 
       121 
     | 
    
         
            -
             
     | 
| 
       122 
     | 
    
         
            -
                        if @new_resource.password && dscl_get(user_info, :password) == "********"
         
     | 
| 
      
 113 
     | 
    
         
            +
                      @user_info = read_user_info
         
     | 
| 
      
 114 
     | 
    
         
            +
                      if @user_info
         
     | 
| 
      
 115 
     | 
    
         
            +
                        @current_resource.uid(dscl_get(@user_info, :uid))
         
     | 
| 
      
 116 
     | 
    
         
            +
                        @current_resource.gid(dscl_get(@user_info, :gid))
         
     | 
| 
      
 117 
     | 
    
         
            +
                        @current_resource.home(dscl_get(@user_info, :home))
         
     | 
| 
      
 118 
     | 
    
         
            +
                        @current_resource.shell(dscl_get(@user_info, :shell))
         
     | 
| 
      
 119 
     | 
    
         
            +
                        @current_resource.comment(dscl_get(@user_info, :comment))
         
     | 
| 
      
 120 
     | 
    
         
            +
                        @authentication_authority = dscl_get(@user_info, :auth_authority)
         
     | 
| 
      
 121 
     | 
    
         
            +
             
     | 
| 
      
 122 
     | 
    
         
            +
                        if @new_resource.password && dscl_get(@user_info, :password) == "********"
         
     | 
| 
       123 
123 
     | 
    
         
             
                          # A password is set. Let's get the password information from shadow file
         
     | 
| 
       124 
     | 
    
         
            -
                          shadow_hash_binary = dscl_get(user_info, :shadow_hash)
         
     | 
| 
      
 124 
     | 
    
         
            +
                          shadow_hash_binary = dscl_get(@user_info, :shadow_hash)
         
     | 
| 
       125 
125 
     | 
    
         | 
| 
       126 
126 
     | 
    
         
             
                          # Calling shell_out directly since we want to give an input stream
         
     | 
| 
       127 
127 
     | 
    
         
             
                          shadow_hash_xml = convert_binary_plist_to_xml(shadow_hash_binary.string)
         
     | 
| 
         @@ -158,22 +158,26 @@ user password using shadow hash.") 
     | 
|
| 
       158 
158 
     | 
    
         | 
| 
       159 
159 
     | 
    
         
             
                    def create_user
         
     | 
| 
       160 
160 
     | 
    
         
             
                      dscl_create_user
         
     | 
| 
      
 161 
     | 
    
         
            +
                      # set_password modifies the plist file of the user directly. So update
         
     | 
| 
      
 162 
     | 
    
         
            +
                      # the password first before making any modifications to the user.
         
     | 
| 
      
 163 
     | 
    
         
            +
                      set_password
         
     | 
| 
       161 
164 
     | 
    
         
             
                      dscl_create_comment
         
     | 
| 
       162 
165 
     | 
    
         
             
                      dscl_set_uid
         
     | 
| 
       163 
166 
     | 
    
         
             
                      dscl_set_gid
         
     | 
| 
       164 
167 
     | 
    
         
             
                      dscl_set_home
         
     | 
| 
       165 
168 
     | 
    
         
             
                      dscl_set_shell
         
     | 
| 
       166 
     | 
    
         
            -
                      set_password
         
     | 
| 
       167 
169 
     | 
    
         
             
                    end
         
     | 
| 
       168 
170 
     | 
    
         | 
| 
       169 
171 
     | 
    
         
             
                    def manage_user
         
     | 
| 
      
 172 
     | 
    
         
            +
                      # set_password modifies the plist file of the user directly. So update
         
     | 
| 
      
 173 
     | 
    
         
            +
                      # the password first before making any modifications to the user.
         
     | 
| 
      
 174 
     | 
    
         
            +
                      set_password        if diverged_password?
         
     | 
| 
       170 
175 
     | 
    
         
             
                      dscl_create_user    if diverged?(:username)
         
     | 
| 
       171 
176 
     | 
    
         
             
                      dscl_create_comment if diverged?(:comment)
         
     | 
| 
       172 
177 
     | 
    
         
             
                      dscl_set_uid        if diverged?(:uid)
         
     | 
| 
       173 
178 
     | 
    
         
             
                      dscl_set_gid        if diverged?(:gid)
         
     | 
| 
       174 
179 
     | 
    
         
             
                      dscl_set_home       if diverged?(:home)
         
     | 
| 
       175 
180 
     | 
    
         
             
                      dscl_set_shell      if diverged?(:shell)
         
     | 
| 
       176 
     | 
    
         
            -
                      set_password        if diverged_password?
         
     | 
| 
       177 
181 
     | 
    
         
             
                    end
         
     | 
| 
       178 
182 
     | 
    
         | 
| 
       179 
183 
     | 
    
         
             
                    #
         
     | 
| 
         @@ -339,22 +343,18 @@ user password using shadow hash.") 
     | 
|
| 
       339 
343 
     | 
    
         
             
                        :input => shadow_info.to_plist, :live_stream => shadow_info_binary)
         
     | 
| 
       340 
344 
     | 
    
         
             
                      command.run_command
         
     | 
| 
       341 
345 
     | 
    
         | 
| 
      
 346 
     | 
    
         
            +
                      if @user_info.nil?
         
     | 
| 
      
 347 
     | 
    
         
            +
                        # User is  just created. read_user_info() will read the fresh information
         
     | 
| 
      
 348 
     | 
    
         
            +
                        # for the user with a cache flush. However with experimentation we've seen
         
     | 
| 
      
 349 
     | 
    
         
            +
                        # that dscl cache is not immediately updated after the creation of the user
         
     | 
| 
      
 350 
     | 
    
         
            +
                        # This is odd and needs to be investigated further.
         
     | 
| 
      
 351 
     | 
    
         
            +
                        sleep 3
         
     | 
| 
      
 352 
     | 
    
         
            +
                        @user_info = read_user_info
         
     | 
| 
      
 353 
     | 
    
         
            +
                      end
         
     | 
| 
      
 354 
     | 
    
         
            +
             
     | 
| 
       342 
355 
     | 
    
         
             
                      # Replace the shadow info in user's plist
         
     | 
| 
       343 
     | 
    
         
            -
                      user_info  
     | 
| 
       344 
     | 
    
         
            -
                       
     | 
| 
       345 
     | 
    
         
            -
             
     | 
| 
       346 
     | 
    
         
            -
                      #
         
     | 
| 
       347 
     | 
    
         
            -
                      # Before saving the user's plist file we need to wait for dscl to
         
     | 
| 
       348 
     | 
    
         
            -
                      # update its caches and flush them to disk. In order to achieve this
         
     | 
| 
       349 
     | 
    
         
            -
                      # we need to wait first for our changes to get into the dscl cache
         
     | 
| 
       350 
     | 
    
         
            -
                      # and then flush the cache to disk before saving password into the
         
     | 
| 
       351 
     | 
    
         
            -
                      # plist file. 3 seconds is the minimum experimental value for dscl
         
     | 
| 
       352 
     | 
    
         
            -
                      # cache to be updated. We can get rid of this sleep when we find a
         
     | 
| 
       353 
     | 
    
         
            -
                      # trigger to update dscl cache.
         
     | 
| 
       354 
     | 
    
         
            -
                      #
         
     | 
| 
       355 
     | 
    
         
            -
                      sleep 3
         
     | 
| 
       356 
     | 
    
         
            -
                      shell_out("dscacheutil '-flushcache'")
         
     | 
| 
       357 
     | 
    
         
            -
                      save_user_info(user_info)
         
     | 
| 
      
 356 
     | 
    
         
            +
                      dscl_set(@user_info, :shadow_hash, shadow_info_binary)
         
     | 
| 
      
 357 
     | 
    
         
            +
                      save_user_info(@user_info)
         
     | 
| 
       358 
358 
     | 
    
         
             
                    end
         
     | 
| 
       359 
359 
     | 
    
         | 
| 
       360 
360 
     | 
    
         
             
                    #
         
     | 
| 
         @@ -555,6 +555,10 @@ user password using shadow hash.") 
     | 
|
| 
       555 
555 
     | 
    
         
             
                    def read_user_info
         
     | 
| 
       556 
556 
     | 
    
         
             
                      user_info = nil
         
     | 
| 
       557 
557 
     | 
    
         | 
| 
      
 558 
     | 
    
         
            +
                      # We flush the cache here in order to make sure that we read fresh information
         
     | 
| 
      
 559 
     | 
    
         
            +
                      # for the user. 
         
     | 
| 
      
 560 
     | 
    
         
            +
                      shell_out("dscacheutil '-flushcache'")
         
     | 
| 
      
 561 
     | 
    
         
            +
             
     | 
| 
       558 
562 
     | 
    
         
             
                      begin
         
     | 
| 
       559 
563 
     | 
    
         
             
                        user_plist_file = "#{USER_PLIST_DIRECTORY}/#{@new_resource.username}.plist"
         
     | 
| 
       560 
564 
     | 
    
         
             
                        user_plist_info = run_plutil("convert xml1 -o - #{user_plist_file}")
         
     | 
    
        data/lib/chef/providers.rb
    CHANGED
    
    | 
         @@ -24,6 +24,7 @@ require 'chef/provider/cron/solaris' 
     | 
|
| 
       24 
24 
     | 
    
         
             
            require 'chef/provider/cron/aix'
         
     | 
| 
       25 
25 
     | 
    
         
             
            require 'chef/provider/deploy'
         
     | 
| 
       26 
26 
     | 
    
         
             
            require 'chef/provider/directory'
         
     | 
| 
      
 27 
     | 
    
         
            +
            require 'chef/provider/dsc_script'
         
     | 
| 
       27 
28 
     | 
    
         
             
            require 'chef/provider/env'
         
     | 
| 
       28 
29 
     | 
    
         
             
            require 'chef/provider/erl_call'
         
     | 
| 
       29 
30 
     | 
    
         
             
            require 'chef/provider/execute'
         
     | 
| 
         @@ -0,0 +1,140 @@ 
     | 
|
| 
      
 1 
     | 
    
         
            +
            #
         
     | 
| 
      
 2 
     | 
    
         
            +
            # Author:: Adam Edwards (<adamed@getchef.com>)
         
     | 
| 
      
 3 
     | 
    
         
            +
            # Copyright:: Copyright (c) 2014 Chef Software, Inc.
         
     | 
| 
      
 4 
     | 
    
         
            +
            # License:: Apache License, Version 2.0
         
     | 
| 
      
 5 
     | 
    
         
            +
            #
         
     | 
| 
      
 6 
     | 
    
         
            +
            # Licensed under the Apache License, Version 2.0 (the "License");
         
     | 
| 
      
 7 
     | 
    
         
            +
            # you may not use this file except in compliance with the License.
         
     | 
| 
      
 8 
     | 
    
         
            +
            # You may obtain a copy of the License at
         
     | 
| 
      
 9 
     | 
    
         
            +
            #
         
     | 
| 
      
 10 
     | 
    
         
            +
            #     http://www.apache.org/licenses/LICENSE-2.0
         
     | 
| 
      
 11 
     | 
    
         
            +
            #
         
     | 
| 
      
 12 
     | 
    
         
            +
            # Unless required by applicable law or agreed to in writing, software
         
     | 
| 
      
 13 
     | 
    
         
            +
            # distributed under the License is distributed on an "AS IS" BASIS,
         
     | 
| 
      
 14 
     | 
    
         
            +
            # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
         
     | 
| 
      
 15 
     | 
    
         
            +
            # See the License for the specific language governing permissions and
         
     | 
| 
      
 16 
     | 
    
         
            +
            # limitations under the License.
         
     | 
| 
      
 17 
     | 
    
         
            +
            #
         
     | 
| 
      
 18 
     | 
    
         
            +
             
     | 
| 
      
 19 
     | 
    
         
            +
            require 'chef/exceptions'
         
     | 
| 
      
 20 
     | 
    
         
            +
             
     | 
| 
      
 21 
     | 
    
         
            +
            class Chef
         
     | 
| 
      
 22 
     | 
    
         
            +
              class Resource
         
     | 
| 
      
 23 
     | 
    
         
            +
                class DscScript < Chef::Resource
         
     | 
| 
      
 24 
     | 
    
         
            +
             
     | 
| 
      
 25 
     | 
    
         
            +
                  provides :dsc_script, :on_platforms => ["windows"]
         
     | 
| 
      
 26 
     | 
    
         
            +
             
     | 
| 
      
 27 
     | 
    
         
            +
                  def initialize(name, run_context=nil)
         
     | 
| 
      
 28 
     | 
    
         
            +
                    super
         
     | 
| 
      
 29 
     | 
    
         
            +
                    @allowed_actions.push(:run)
         
     | 
| 
      
 30 
     | 
    
         
            +
                    @action = :run
         
     | 
| 
      
 31 
     | 
    
         
            +
                    if(run_context && Chef::Platform.supports_dsc?(run_context.node))
         
     | 
| 
      
 32 
     | 
    
         
            +
                      @provider = Chef::Provider::DscScript
         
     | 
| 
      
 33 
     | 
    
         
            +
                    else
         
     | 
| 
      
 34 
     | 
    
         
            +
                      raise Chef::Exceptions::NoProviderAvailable,
         
     | 
| 
      
 35 
     | 
    
         
            +
                        "#{powershell_info_str(run_context)}\nPowershell 4.0 or higher was not detected on your system and is required to use the dsc_script resource."
         
     | 
| 
      
 36 
     | 
    
         
            +
                    end
         
     | 
| 
      
 37 
     | 
    
         
            +
                  end
         
     | 
| 
      
 38 
     | 
    
         
            +
             
     | 
| 
      
 39 
     | 
    
         
            +
                  def code(arg=nil)
         
     | 
| 
      
 40 
     | 
    
         
            +
                    if arg && command
         
     | 
| 
      
 41 
     | 
    
         
            +
                      raise ArgumentError, "Only one of 'code' and 'command' attributes may be specified"
         
     | 
| 
      
 42 
     | 
    
         
            +
                    end
         
     | 
| 
      
 43 
     | 
    
         
            +
                    if arg && configuration_name
         
     | 
| 
      
 44 
     | 
    
         
            +
                      raise ArgumentError, "The 'code' and 'command' attributes may not be used together"
         
     | 
| 
      
 45 
     | 
    
         
            +
                    end
         
     | 
| 
      
 46 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 47 
     | 
    
         
            +
                      :code,
         
     | 
| 
      
 48 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 49 
     | 
    
         
            +
                      :kind_of => [ String ]
         
     | 
| 
      
 50 
     | 
    
         
            +
                    )
         
     | 
| 
      
 51 
     | 
    
         
            +
                  end
         
     | 
| 
      
 52 
     | 
    
         
            +
             
     | 
| 
      
 53 
     | 
    
         
            +
                  def configuration_name(arg=nil)
         
     | 
| 
      
 54 
     | 
    
         
            +
                    if arg && code
         
     | 
| 
      
 55 
     | 
    
         
            +
                      raise ArgumentError, "Attribute `configuration_name` may not be set if `code` is set"
         
     | 
| 
      
 56 
     | 
    
         
            +
                    end
         
     | 
| 
      
 57 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 58 
     | 
    
         
            +
                      :configuration_name,
         
     | 
| 
      
 59 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 60 
     | 
    
         
            +
                      :kind_of => [ String ]
         
     | 
| 
      
 61 
     | 
    
         
            +
                    )
         
     | 
| 
      
 62 
     | 
    
         
            +
                  end
         
     | 
| 
      
 63 
     | 
    
         
            +
             
     | 
| 
      
 64 
     | 
    
         
            +
                  def command(arg=nil)
         
     | 
| 
      
 65 
     | 
    
         
            +
                    if arg && code
         
     | 
| 
      
 66 
     | 
    
         
            +
                      raise ArgumentError, "The 'code' and 'command' attributes may not be used together"
         
     | 
| 
      
 67 
     | 
    
         
            +
                    end
         
     | 
| 
      
 68 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 69 
     | 
    
         
            +
                      :command,
         
     | 
| 
      
 70 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 71 
     | 
    
         
            +
                      :kind_of => [ String ]
         
     | 
| 
      
 72 
     | 
    
         
            +
                    )
         
     | 
| 
      
 73 
     | 
    
         
            +
                  end
         
     | 
| 
      
 74 
     | 
    
         
            +
             
     | 
| 
      
 75 
     | 
    
         
            +
                  def configuration_data(arg=nil)
         
     | 
| 
      
 76 
     | 
    
         
            +
                    if arg && configuration_data_script
         
     | 
| 
      
 77 
     | 
    
         
            +
                      raise ArgumentError, "The 'configuration_data' and 'configuration_data_script' attributes may not be used together"
         
     | 
| 
      
 78 
     | 
    
         
            +
                    end
         
     | 
| 
      
 79 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 80 
     | 
    
         
            +
                      :configuration_data,
         
     | 
| 
      
 81 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 82 
     | 
    
         
            +
                      :kind_of => [ String ]
         
     | 
| 
      
 83 
     | 
    
         
            +
                    )
         
     | 
| 
      
 84 
     | 
    
         
            +
                  end
         
     | 
| 
      
 85 
     | 
    
         
            +
             
     | 
| 
      
 86 
     | 
    
         
            +
                  def configuration_data_script(arg=nil)
         
     | 
| 
      
 87 
     | 
    
         
            +
                    if arg && configuration_data
         
     | 
| 
      
 88 
     | 
    
         
            +
                      raise ArgumentError, "The 'configuration_data' and 'configuration_data_script' attributes may not be used together"
         
     | 
| 
      
 89 
     | 
    
         
            +
                    end
         
     | 
| 
      
 90 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 91 
     | 
    
         
            +
                      :configuration_data_script,
         
     | 
| 
      
 92 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 93 
     | 
    
         
            +
                      :kind_of => [ String ]
         
     | 
| 
      
 94 
     | 
    
         
            +
                    )
         
     | 
| 
      
 95 
     | 
    
         
            +
                  end
         
     | 
| 
      
 96 
     | 
    
         
            +
             
     | 
| 
      
 97 
     | 
    
         
            +
                  def flags(arg=nil)
         
     | 
| 
      
 98 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 99 
     | 
    
         
            +
                      :flags,
         
     | 
| 
      
 100 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 101 
     | 
    
         
            +
                      :kind_of => [ Hash ]
         
     | 
| 
      
 102 
     | 
    
         
            +
                    )
         
     | 
| 
      
 103 
     | 
    
         
            +
                  end
         
     | 
| 
      
 104 
     | 
    
         
            +
             
     | 
| 
      
 105 
     | 
    
         
            +
                  def cwd(arg=nil)
         
     | 
| 
      
 106 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 107 
     | 
    
         
            +
                      :cwd,
         
     | 
| 
      
 108 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 109 
     | 
    
         
            +
                      :kind_of => [ String ]
         
     | 
| 
      
 110 
     | 
    
         
            +
                    )
         
     | 
| 
      
 111 
     | 
    
         
            +
                  end
         
     | 
| 
      
 112 
     | 
    
         
            +
             
     | 
| 
      
 113 
     | 
    
         
            +
                  def environment(arg=nil)
         
     | 
| 
      
 114 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 115 
     | 
    
         
            +
                      :environment,
         
     | 
| 
      
 116 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 117 
     | 
    
         
            +
                      :kind_of => [ Hash ]
         
     | 
| 
      
 118 
     | 
    
         
            +
                    )
         
     | 
| 
      
 119 
     | 
    
         
            +
                  end
         
     | 
| 
      
 120 
     | 
    
         
            +
             
     | 
| 
      
 121 
     | 
    
         
            +
                  def timeout(arg=nil)
         
     | 
| 
      
 122 
     | 
    
         
            +
                    set_or_return(
         
     | 
| 
      
 123 
     | 
    
         
            +
                      :timeout,
         
     | 
| 
      
 124 
     | 
    
         
            +
                      arg,
         
     | 
| 
      
 125 
     | 
    
         
            +
                      :kind_of => [ Integer ]
         
     | 
| 
      
 126 
     | 
    
         
            +
                    )
         
     | 
| 
      
 127 
     | 
    
         
            +
                  end
         
     | 
| 
      
 128 
     | 
    
         
            +
             
     | 
| 
      
 129 
     | 
    
         
            +
                  private
         
     | 
| 
      
 130 
     | 
    
         
            +
             
     | 
| 
      
 131 
     | 
    
         
            +
                  def powershell_info_str(run_context)
         
     | 
| 
      
 132 
     | 
    
         
            +
                    if run_context && run_context.node[:languages] && run_context.node[:languages][:powershell]
         
     | 
| 
      
 133 
     | 
    
         
            +
                        install_info = "Powershell #{run_context.node[:languages][:powershell][:version]} was found on the system."
         
     | 
| 
      
 134 
     | 
    
         
            +
                      else
         
     | 
| 
      
 135 
     | 
    
         
            +
                        install_info = 'Powershell was not found.'
         
     | 
| 
      
 136 
     | 
    
         
            +
                      end
         
     | 
| 
      
 137 
     | 
    
         
            +
                  end
         
     | 
| 
      
 138 
     | 
    
         
            +
                end
         
     | 
| 
      
 139 
     | 
    
         
            +
              end
         
     | 
| 
      
 140 
     | 
    
         
            +
            end
         
     |