├── .gitignore ├── .kitchen.yml ├── LICENSE.txt ├── README.md ├── bin ├── chefdk-update-app.bat ├── chefdk-update-app.rb └── chefdk-update-app.sh └── test └── integration ├── bootstrap.ps1 └── bootstrap.sh /.gitignore: -------------------------------------------------------------------------------- 1 | *.gem 2 | *.rbc 3 | .bundle 4 | Gemfile 5 | Gemfile.lock 6 | tmp 7 | .rvmrc 8 | .rbenv-version 9 | .ruby-version 10 | .DS_Store 11 | .kitchen/ 12 | .kitchen.local.yml 13 | -------------------------------------------------------------------------------- /.kitchen.yml: -------------------------------------------------------------------------------- 1 | --- 2 | driver: 3 | name: vagrant 4 | 5 | provisioner: 6 | name: shell 7 | data_path: bin 8 | 9 | platforms: 10 | - name: ubuntu-14.04 11 | - name: windows-2012r2-core 12 | 13 | suites: 14 | - name: default 15 | -------------------------------------------------------------------------------- /LICENSE.txt: -------------------------------------------------------------------------------- 1 | 2 | Apache License 3 | Version 2.0, January 2004 4 | http://www.apache.org/licenses/ 5 | 6 | TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION 7 | 8 | 1. Definitions. 9 | 10 | "License" shall mean the terms and conditions for use, reproduction, 11 | and distribution as defined by Sections 1 through 9 of this document. 12 | 13 | "Licensor" shall mean the copyright owner or entity authorized by 14 | the copyright owner that is granting the License. 15 | 16 | "Legal Entity" shall mean the union of the acting entity and all 17 | other entities that control, are controlled by, or are under common 18 | control with that entity. For the purposes of this definition, 19 | "control" means (i) the power, direct or indirect, to cause the 20 | direction or management of such entity, whether by contract or 21 | otherwise, or (ii) ownership of fifty percent (50%) or more of the 22 | outstanding shares, or (iii) beneficial ownership of such entity. 23 | 24 | "You" (or "Your") shall mean an individual or Legal Entity 25 | exercising permissions granted by this License. 26 | 27 | "Source" form shall mean the preferred form for making modifications, 28 | including but not limited to software source code, documentation 29 | source, and configuration files. 30 | 31 | "Object" form shall mean any form resulting from mechanical 32 | transformation or translation of a Source form, including but 33 | not limited to compiled object code, generated documentation, 34 | and conversions to other media types. 35 | 36 | "Work" shall mean the work of authorship, whether in Source or 37 | Object form, made available under the License, as indicated by a 38 | copyright notice that is included in or attached to the work 39 | (an example is provided in the Appendix below). 40 | 41 | "Derivative Works" shall mean any work, whether in Source or Object 42 | form, that is based on (or derived from) the Work and for which the 43 | editorial revisions, annotations, elaborations, or other modifications 44 | represent, as a whole, an original work of authorship. For the purposes 45 | of this License, Derivative Works shall not include works that remain 46 | separable from, or merely link (or bind by name) to the interfaces of, 47 | the Work and Derivative Works thereof. 48 | 49 | "Contribution" shall mean any work of authorship, including 50 | the original version of the Work and any modifications or additions 51 | to that Work or Derivative Works thereof, that is intentionally 52 | submitted to Licensor for inclusion in the Work by the copyright owner 53 | or by an individual or Legal Entity authorized to submit on behalf of 54 | the copyright owner. For the purposes of this definition, "submitted" 55 | means any form of electronic, verbal, or written communication sent 56 | to the Licensor or its representatives, including but not limited to 57 | communication on electronic mailing lists, source code control systems, 58 | and issue tracking systems that are managed by, or on behalf of, the 59 | Licensor for the purpose of discussing and improving the Work, but 60 | excluding communication that is conspicuously marked or otherwise 61 | designated in writing by the copyright owner as "Not a Contribution." 62 | 63 | "Contributor" shall mean Licensor and any individual or Legal Entity 64 | on behalf of whom a Contribution has been received by Licensor and 65 | subsequently incorporated within the Work. 66 | 67 | 2. Grant of Copyright License. Subject to the terms and conditions of 68 | this License, each Contributor hereby grants to You a perpetual, 69 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 70 | copyright license to reproduce, prepare Derivative Works of, 71 | publicly display, publicly perform, sublicense, and distribute the 72 | Work and such Derivative Works in Source or Object form. 73 | 74 | 3. Grant of Patent License. Subject to the terms and conditions of 75 | this License, each Contributor hereby grants to You a perpetual, 76 | worldwide, non-exclusive, no-charge, royalty-free, irrevocable 77 | (except as stated in this section) patent license to make, have made, 78 | use, offer to sell, sell, import, and otherwise transfer the Work, 79 | where such license applies only to those patent claims licensable 80 | by such Contributor that are necessarily infringed by their 81 | Contribution(s) alone or by combination of their Contribution(s) 82 | with the Work to which such Contribution(s) was submitted. If You 83 | institute patent litigation against any entity (including a 84 | cross-claim or counterclaim in a lawsuit) alleging that the Work 85 | or a Contribution incorporated within the Work constitutes direct 86 | or contributory patent infringement, then any patent licenses 87 | granted to You under this License for that Work shall terminate 88 | as of the date such litigation is filed. 89 | 90 | 4. Redistribution. You may reproduce and distribute copies of the 91 | Work or Derivative Works thereof in any medium, with or without 92 | modifications, and in Source or Object form, provided that You 93 | meet the following conditions: 94 | 95 | (a) You must give any other recipients of the Work or 96 | Derivative Works a copy of this License; and 97 | 98 | (b) You must cause any modified files to carry prominent notices 99 | stating that You changed the files; and 100 | 101 | (c) You must retain, in the Source form of any Derivative Works 102 | that You distribute, all copyright, patent, trademark, and 103 | attribution notices from the Source form of the Work, 104 | excluding those notices that do not pertain to any part of 105 | the Derivative Works; and 106 | 107 | (d) If the Work includes a "NOTICE" text file as part of its 108 | distribution, then any Derivative Works that You distribute must 109 | include a readable copy of the attribution notices contained 110 | within such NOTICE file, excluding those notices that do not 111 | pertain to any part of the Derivative Works, in at least one 112 | of the following places: within a NOTICE text file distributed 113 | as part of the Derivative Works; within the Source form or 114 | documentation, if provided along with the Derivative Works; or, 115 | within a display generated by the Derivative Works, if and 116 | wherever such third-party notices normally appear. The contents 117 | of the NOTICE file are for informational purposes only and 118 | do not modify the License. You may add Your own attribution 119 | notices within Derivative Works that You distribute, alongside 120 | or as an addendum to the NOTICE text from the Work, provided 121 | that such additional attribution notices cannot be construed 122 | as modifying the License. 123 | 124 | You may add Your own copyright statement to Your modifications and 125 | may provide additional or different license terms and conditions 126 | for use, reproduction, or distribution of Your modifications, or 127 | for any such Derivative Works as a whole, provided Your use, 128 | reproduction, and distribution of the Work otherwise complies with 129 | the conditions stated in this License. 130 | 131 | 5. Submission of Contributions. Unless You explicitly state otherwise, 132 | any Contribution intentionally submitted for inclusion in the Work 133 | by You to the Licensor shall be under the terms and conditions of 134 | this License, without any additional terms or conditions. 135 | Notwithstanding the above, nothing herein shall supersede or modify 136 | the terms of any separate license agreement you may have executed 137 | with Licensor regarding such Contributions. 138 | 139 | 6. Trademarks. This License does not grant permission to use the trade 140 | names, trademarks, service marks, or product names of the Licensor, 141 | except as required for reasonable and customary use in describing the 142 | origin of the Work and reproducing the content of the NOTICE file. 143 | 144 | 7. Disclaimer of Warranty. Unless required by applicable law or 145 | agreed to in writing, Licensor provides the Work (and each 146 | Contributor provides its Contributions) on an "AS IS" BASIS, 147 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or 148 | implied, including, without limitation, any warranties or conditions 149 | of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A 150 | PARTICULAR PURPOSE. You are solely responsible for determining the 151 | appropriateness of using or redistributing the Work and assume any 152 | risks associated with Your exercise of permissions under this License. 153 | 154 | 8. Limitation of Liability. In no event and under no legal theory, 155 | whether in tort (including negligence), contract, or otherwise, 156 | unless required by applicable law (such as deliberate and grossly 157 | negligent acts) or agreed to in writing, shall any Contributor be 158 | liable to You for damages, including any direct, indirect, special, 159 | incidental, or consequential damages of any character arising as a 160 | result of this License or out of the use or inability to use the 161 | Work (including but not limited to damages for loss of goodwill, 162 | work stoppage, computer failure or malfunction, or any and all 163 | other commercial damages or losses), even if such Contributor 164 | has been advised of the possibility of such damages. 165 | 166 | 9. Accepting Warranty or Additional Liability. While redistributing 167 | the Work or Derivative Works thereof, You may choose to offer, 168 | and charge a fee for, acceptance of support, warranty, indemnity, 169 | or other liability obligations and/or rights consistent with this 170 | License. However, in accepting such obligations, You may act only 171 | on Your own behalf and on Your sole responsibility, not on behalf 172 | of any other Contributor, and only if You agree to indemnify, 173 | defend, and hold each Contributor harmless for any liability 174 | incurred by, or claims asserted against, such Contributor by reason 175 | of your accepting any such warranty or additional liability. 176 | 177 | END OF TERMS AND CONDITIONS 178 | 179 | APPENDIX: How to apply the Apache License to your work. 180 | 181 | To apply the Apache License to your work, attach the following 182 | boilerplate notice, with the fields enclosed by brackets "[]" 183 | replaced with your own identifying information. (Don't include 184 | the brackets!) The text should be enclosed in the appropriate 185 | comment syntax for the file format. We also recommend that a 186 | file or class name and description of purpose be included on the 187 | same "printed page" as the copyright notice for easier 188 | identification within third-party archives. 189 | 190 | Copyright [yyyy] [name of copyright owner] 191 | 192 | Licensed under the Apache License, Version 2.0 (the "License"); 193 | you may not use this file except in compliance with the License. 194 | You may obtain a copy of the License at 195 | 196 | http://www.apache.org/licenses/LICENSE-2.0 197 | 198 | Unless required by applicable law or agreed to in writing, software 199 | distributed under the License is distributed on an "AS IS" BASIS, 200 | WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 201 | See the License for the specific language governing permissions and 202 | limitations under the License. 203 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # chefdk-update-app 2 | 3 | A little help when you want to update an appbundled project inside [ChefDK](https://downloads.chef.io/chef-dk/). 4 | 5 | ## Requirements 6 | 7 | * You need to have a pre-existing ChefDK installation in either `/opt/chefdk` (for Unix-like distros) or `%SYSTEMDRIVE%\opscode\chefdk` (for Windows distros). 8 | * You need to have the `git` command in your PATH. 9 | * You need patience if this doesn't work out of the box every time :) 10 | 11 | ## Usage 12 | 13 | Clone this repository with and enter the project directory: 14 | 15 | ```sh 16 | git clone https://github.com/fnichol/chefdk-update-app.git 17 | cd chefdk-update-app 18 | ``` 19 | 20 | Choose a ChefDK "app" to update. If you don't need a list of options, run the help command. 21 | 22 | For Unix distros: 23 | 24 | ```sh 25 | ./bin/chefdk-update-app.sh --help 26 | ``` 27 | 28 | For Windows distros from PowerShell: 29 | 30 | ```powershell 31 | & bin\chefdk-update-app.bat --help 32 | ``` 33 | 34 | Choose a Git reference to update your app to. This could be a branch name, tag, SHA hash, or even `"master"`. 35 | 36 | Run it! 37 | 38 | For example, to update the `"test-kitchen"` app to release `"v1.4.0.beta.2"` (note that this project puts a `"v"` in front of release tags), 39 | 40 | For Unix distros: 41 | 42 | ```sh 43 | sudo -E ./bin/chefdk-update-app.sh test-kitchen -r v1.4.0.beta.2 44 | ``` 45 | 46 | For Windows distros from PowerShell: 47 | 48 | ```powershell 49 | & bin\chefdk-update-app.bat test-kitchen -r v1.4.0.beta.2 50 | ``` 51 | 52 | ## Development 53 | 54 | * Source hosted at [GitHub][repo] 55 | * Report issues/questions/feature requests on [GitHub Issues][issues] 56 | 57 | Pull requests are very welcome! Make sure your patches are well tested. 58 | Ideally create a topic branch for every separate change you make. For 59 | example: 60 | 61 | 1. Fork the repo 62 | 2. Create your feature branch (`git checkout -b my-new-feature`) 63 | 3. Commit your changes (`git commit -am 'Added some feature'`) 64 | 4. Push to the branch (`git push origin my-new-feature`) 65 | 5. Create new Pull Request 66 | 67 | ## Authors 68 | 69 | Created and maintained by [Fletcher Nichol][fnichol] () 70 | 71 | ## License 72 | 73 | Apache 2.0 (see [LICENSE.txt][license]) 74 | 75 | [license]: https://github.com/fnichol/chefdk-update-app/blob/master/LICENSE.txt 76 | [fnichol]: https://github.com/fnichol 77 | [repo]: https://github.com/fnichol/chefdk-update-app 78 | [issues]: https://github.com/fnichol/chefdk-update-app/issues 79 | -------------------------------------------------------------------------------- /bin/chefdk-update-app.bat: -------------------------------------------------------------------------------- 1 | @ECHO OFF 2 | SETLOCAL 3 | 4 | SET MYPATH=%~dp0 5 | 6 | %SYSTEMDRIVE%\opscode\chefdk\embedded\bin\ruby.exe "%MYPATH:~0,-1%\chefdk-update-app.rb" %* 7 | 8 | SET "RUBY_EXIT_STATUS=%ERRORLEVEL%" 9 | exit /b %RUBY_EXIT_STATUS% 10 | -------------------------------------------------------------------------------- /bin/chefdk-update-app.rb: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env ruby 2 | # -*- encoding: utf-8 -*- 3 | # 4 | # Author:: Fletcher Nichol () 5 | # 6 | # Copyright (C) 2015 Fletcher Nichol 7 | # 8 | # Licensed under the Apache License, Version 2.0 (the "License"); 9 | # you may not use this file except in compliance with the License. 10 | # You may obtain a copy of the License at 11 | # 12 | # http://www.apache.org/licenses/LICENSE-2.0 13 | # 14 | # Unless required by applicable law or agreed to in writing, software 15 | # distributed under the License is distributed on an "AS IS" BASIS, 16 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 17 | # See the License for the specific language governing permissions and 18 | # limitations under the License. 19 | 20 | require "pathname" 21 | require "optparse" 22 | 23 | App = Struct.new(:name, :url, :bundle_without) do 24 | def to_s 25 | name 26 | end 27 | end 28 | 29 | CHEFDK_APPS = [ 30 | App.new( 31 | "berkshelf", 32 | "https://github.com/berkshelf/berkshelf.git", 33 | "guard test development" 34 | ), 35 | App.new( 36 | "chef", 37 | "https://github.com/chef/chef.git", 38 | "server docgen test development" 39 | ), 40 | App.new( 41 | "chef-dk", 42 | "https://github.com/chef/chef-dk.git", 43 | "dev test development" 44 | ), 45 | App.new( 46 | "chef-vault", 47 | "https://github.com/Nordstrom/chef-vault.git", 48 | "test development" 49 | ), 50 | App.new( 51 | "foodcritic", 52 | "https://github.com/acrmp/foodcritic.git", 53 | "test development" 54 | ), 55 | App.new( 56 | "ohai", 57 | "https://github.com/chef/ohai.git", 58 | "test development" 59 | ), 60 | App.new( 61 | "test-kitchen", 62 | "https://github.com/test-kitchen/test-kitchen.git", 63 | "guard test development" 64 | ) 65 | ].freeze 66 | 67 | class Updater 68 | attr_reader :app, :ref 69 | 70 | def initialize(options) 71 | @app = options[:app] 72 | @ref = options[:ref] 73 | end 74 | 75 | def start 76 | if !windows? && Process.uid != 0 77 | abort "#{$0} needs to be run as root user or with sudo" 78 | end 79 | 80 | banner("Cleaning #{app} checkout") 81 | app_dir.rmtree if app_dir.directory? 82 | 83 | banner("Cloning #{app} from #{app.url}") 84 | run("git clone #{app.url} #{app_dir}") 85 | 86 | banner("Checking out #{app} to #{ref}") 87 | Dir.chdir(app_dir) do 88 | run("git checkout #{ref}") 89 | end 90 | 91 | banner("Installing dependencies") 92 | Dir.chdir(app_dir) do 93 | ruby("#{bin_dir.join("bundle")} install --without #{app.bundle_without}") 94 | end 95 | 96 | banner("Updating appbundler binstubs for #{app}") 97 | Dir.chdir(app_dir) do 98 | ruby("#{bin_dir.join("appbundler")} #{app_dir} #{chefdk.join("bin")}") 99 | end 100 | 101 | banner("Finished!") 102 | end 103 | 104 | private 105 | 106 | ENV_KEYS = %w[ 107 | BUNDLE_BIN_PATH BUNDLE_GEMFILE GEM_HOME GEM_PATH GEM_CACHE RUBYOPT 108 | ].freeze 109 | 110 | def app_dir 111 | chefdk.join("embedded/apps/#{app}") 112 | end 113 | 114 | def banner(msg) 115 | puts "-----> #{msg}" 116 | end 117 | 118 | def bin_dir 119 | chefdk.join("embedded/bin") 120 | end 121 | 122 | def chefdk 123 | if windows? 124 | Pathname.new(File.join(ENV["SYSTEMDRIVE"], "opscode", "chefdk")) 125 | else 126 | Pathname.new("/opt/chefdk") 127 | end 128 | end 129 | 130 | def ruby(script) 131 | ruby = bin_dir.join("ruby").to_s.tap { |p| p.concat(".exe") if windows? } 132 | 133 | run([ruby, script].join(" ")) 134 | end 135 | 136 | def run(cmd) 137 | ENV_KEYS.each { |key| ENV["_YOLO_#{key}"] = ENV[key]; ENV.delete(key) } 138 | system(cmd) or raise("Command [#{cmd}] failed!") 139 | ENV_KEYS.each { |key| ENV[key] = ENV.delete("_YOLO_#{key}") } 140 | end 141 | 142 | def windows? 143 | @windows ||= RUBY_PLATFORM =~ /mswin|mingw|windows/ 144 | end 145 | end 146 | 147 | class CLI 148 | def self.options 149 | new.options 150 | end 151 | 152 | attr_reader :options, :parser 153 | 154 | def initialize 155 | @options = Hash.new 156 | @parser = OptionParser.new { |opts| 157 | opts.banner = "Usage: #{$0} APP_NAME [options]" 158 | opts.on("-r REF", "--ref REF", "Git reference (branch, tag, sha, etc.)") do |r| 159 | options[:ref] = r 160 | end 161 | opts.on("-h", "--help", "Prints this help") do 162 | puts opts 163 | exit 164 | end 165 | opts.separator("") 166 | opts.separator("App names:") 167 | CHEFDK_APPS.each { |a| opts.separator(" * #{a.name}") } 168 | } 169 | @parser.parse! 170 | validate! 171 | end 172 | 173 | def validate! 174 | die("APP_NAME is required") if ARGV.empty? 175 | options[:app] = CHEFDK_APPS.find { |a| a.name == ARGV.first } 176 | die("Invalid APP_NAME: #{ARGV.first}") if options[:app].nil? 177 | die("--ref flag is required") if options[:ref].nil? 178 | end 179 | 180 | def die(msg) 181 | $stderr.puts msg 182 | $stderr.puts parser 183 | exit 1 184 | end 185 | end 186 | 187 | Updater.new(CLI.options).start 188 | -------------------------------------------------------------------------------- /bin/chefdk-update-app.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | 3 | exec "/opt/chefdk/embedded/bin/ruby" "`dirname $0`/chefdk-update-app.rb" "$@" 4 | -------------------------------------------------------------------------------- /test/integration/bootstrap.ps1: -------------------------------------------------------------------------------- 1 | Function Get-ChefMetadata($url) { 2 | Try { $response = ($c = Make-WebClient).DownloadString($url) } 3 | Finally { if ($c -ne $null) { $c.Dispose() } } 4 | 5 | $md = ConvertFrom-StringData $response.Replace("`t", "=") 6 | return @($md.url, $md.md5) 7 | } 8 | 9 | Function Get-MD5Sum($src) { 10 | Try { 11 | $c = New-Object -TypeName System.Security.Cryptography.MD5CryptoServiceProvider 12 | $bytes = $c.ComputeHash(($in = (Get-Item $src).OpenRead())) 13 | return ([System.BitConverter]::ToString($bytes)).Replace("-", "").ToLower() 14 | } Finally { if (($c -ne $null) -and ($c.GetType().GetMethod("Dispose") -ne $null)) { $c.Dispose() }; if ($in -ne $null) { $in.Dispose() } } 15 | } 16 | 17 | Function Download-Chef($md_url, $dst) { 18 | $url, $md5 = Get-ChefMetadata $md_url 19 | 20 | Try { 21 | Log "Downloading package from $url" 22 | ($c = Make-WebClient).DownloadFile($url, $dst) 23 | Log "Download complete." 24 | } Finally { if ($c -ne $null) { $c.Dispose() } } 25 | 26 | if (($dmd5 = Get-MD5Sum $dst) -eq $md5) { Log "Successfully verified $dst" } 27 | else { throw "MD5 for $dst $dmd5 does not match $md5" } 28 | } 29 | 30 | Function Install-Chef($msi) { 31 | Log "Installing ChefDK package $msi" 32 | $p = Start-Process -FilePath "msiexec.exe" -ArgumentList "/qn /i $msi" -Passthru -Wait 33 | 34 | if ($p.ExitCode -ne 0) { throw "msiexec was not successful. Received exit code $($p.ExitCode)" } 35 | 36 | Remove-Item $msi -Force 37 | Log "Installation complete" 38 | } 39 | 40 | Function Install-Git { 41 | iex ((Make-WebClient).DownloadString("https://chocolatey.org/install.ps1")) 42 | choco install git 43 | } 44 | 45 | Function Log($m) { Write-Host " $m`n" } 46 | 47 | Function Banner ($m) { Write-Host "-----> $m`n" } 48 | 49 | Function Make-WebClient { 50 | $proxy = New-Object -TypeName System.Net.WebProxy 51 | $proxy.Address = $env:http_proxy 52 | $client = New-Object -TypeName System.Net.WebClient 53 | $client.Proxy = $proxy 54 | return $client 55 | } 56 | 57 | Function Unresolve-Path($p) { 58 | if ($p -eq $null) { return $null } 59 | else { return $ExecutionContext.SessionState.Path.GetUnresolvedProviderPathFromPSPath($p) } 60 | } 61 | 62 | $chef_metadata_url = "http://www.chef.io/chef/metadata-chefdk?p=windows&m=x86_64&pv=2008r2&v=latest" 63 | $chefdk_root = Unresolve-Path "$env:systemdrive\opscode\chefdk" 64 | $msi = Unresolve-Path "$env:TEMP\chefdk-latest.msi" 65 | $git_path = (Get-Item "ENV:ProgramFiles(x86)").Value + "\Git\bin" 66 | 67 | if (-Not (Test-Path "$env:ALLUSERSPROFILE\chocolatey")) { Install-Git } 68 | 69 | if (-Not (($env:PATH).split(";") -contains "$git_path")) { $env:PATH += ";" + "$git_path" } 70 | 71 | if (-Not (Test-Path $chefdk_root)) { 72 | Banner "Installing ChefDK" 73 | Download-Chef "$chef_metadata_url" $msi 74 | Install-Chef $msi 75 | } else { 76 | Banner "ChefDK installation detected" 77 | } 78 | 79 | Banner "Running ChefDK app update" 80 | & "$env:TEMP\kitchen\data\chefdk-update-app.bat" test-kitchen -r v1.4.0.beta.1 81 | 82 | exit $LASTEXITCODE 83 | -------------------------------------------------------------------------------- /test/integration/bootstrap.sh: -------------------------------------------------------------------------------- 1 | #!/usr/bin/env sh 2 | set -e 3 | set -x 4 | 5 | if test ! -x /usr/bin/curl; then 6 | sudo -E apt-get update -y 7 | sudo -E apt-get install -y curl git-core 8 | fi 9 | 10 | if test ! -d /opt/chefdk; then 11 | curl -L http://www.chef.io/chef/install.sh | sudo -E bash -s -- -P chefdk 12 | fi 13 | 14 | sudo -E /tmp/kitchen/data/chefdk-update-app.sh test-kitchen -r v1.4.0.beta.1 15 | --------------------------------------------------------------------------------