chef-zero 1.4.0.alpha-x86-mingw32

Sign up to get free protection for your applications and to get access to all the features.
Files changed (61) hide show
  1. data/LICENSE +201 -0
  2. data/README.md +145 -0
  3. data/Rakefile +11 -0
  4. data/bin/chef-zero +43 -0
  5. data/lib/chef_zero.rb +7 -0
  6. data/lib/chef_zero/cookbook_data.rb +223 -0
  7. data/lib/chef_zero/data_normalizer.rb +142 -0
  8. data/lib/chef_zero/data_store/data_already_exists_error.rb +29 -0
  9. data/lib/chef_zero/data_store/data_error.rb +31 -0
  10. data/lib/chef_zero/data_store/data_not_found_error.rb +29 -0
  11. data/lib/chef_zero/data_store/memory_store.rb +167 -0
  12. data/lib/chef_zero/endpoints/actor_endpoint.rb +68 -0
  13. data/lib/chef_zero/endpoints/actors_endpoint.rb +32 -0
  14. data/lib/chef_zero/endpoints/authenticate_user_endpoint.rb +25 -0
  15. data/lib/chef_zero/endpoints/cookbook_endpoint.rb +39 -0
  16. data/lib/chef_zero/endpoints/cookbook_version_endpoint.rb +110 -0
  17. data/lib/chef_zero/endpoints/cookbooks_base.rb +65 -0
  18. data/lib/chef_zero/endpoints/cookbooks_endpoint.rb +19 -0
  19. data/lib/chef_zero/endpoints/data_bag_endpoint.rb +45 -0
  20. data/lib/chef_zero/endpoints/data_bag_item_endpoint.rb +25 -0
  21. data/lib/chef_zero/endpoints/data_bags_endpoint.rb +22 -0
  22. data/lib/chef_zero/endpoints/environment_cookbook_endpoint.rb +24 -0
  23. data/lib/chef_zero/endpoints/environment_cookbook_versions_endpoint.rb +109 -0
  24. data/lib/chef_zero/endpoints/environment_cookbooks_endpoint.rb +22 -0
  25. data/lib/chef_zero/endpoints/environment_endpoint.rb +33 -0
  26. data/lib/chef_zero/endpoints/environment_nodes_endpoint.rb +23 -0
  27. data/lib/chef_zero/endpoints/environment_recipes_endpoint.rb +22 -0
  28. data/lib/chef_zero/endpoints/environment_role_endpoint.rb +36 -0
  29. data/lib/chef_zero/endpoints/file_store_file_endpoint.rb +22 -0
  30. data/lib/chef_zero/endpoints/node_endpoint.rb +17 -0
  31. data/lib/chef_zero/endpoints/not_found_endpoint.rb +11 -0
  32. data/lib/chef_zero/endpoints/principal_endpoint.rb +30 -0
  33. data/lib/chef_zero/endpoints/rest_list_endpoint.rb +40 -0
  34. data/lib/chef_zero/endpoints/rest_object_endpoint.rb +61 -0
  35. data/lib/chef_zero/endpoints/role_endpoint.rb +16 -0
  36. data/lib/chef_zero/endpoints/role_environments_endpoint.rb +14 -0
  37. data/lib/chef_zero/endpoints/sandbox_endpoint.rb +27 -0
  38. data/lib/chef_zero/endpoints/sandboxes_endpoint.rb +51 -0
  39. data/lib/chef_zero/endpoints/search_endpoint.rb +188 -0
  40. data/lib/chef_zero/endpoints/searches_endpoint.rb +18 -0
  41. data/lib/chef_zero/log.rb +7 -0
  42. data/lib/chef_zero/rest_base.rb +133 -0
  43. data/lib/chef_zero/rest_error_response.rb +11 -0
  44. data/lib/chef_zero/rest_request.rb +56 -0
  45. data/lib/chef_zero/rest_router.rb +44 -0
  46. data/lib/chef_zero/rspec.rb +107 -0
  47. data/lib/chef_zero/server.rb +309 -0
  48. data/lib/chef_zero/solr/query/binary_operator.rb +53 -0
  49. data/lib/chef_zero/solr/query/phrase.rb +23 -0
  50. data/lib/chef_zero/solr/query/range_query.rb +34 -0
  51. data/lib/chef_zero/solr/query/regexpable_query.rb +29 -0
  52. data/lib/chef_zero/solr/query/subquery.rb +35 -0
  53. data/lib/chef_zero/solr/query/term.rb +45 -0
  54. data/lib/chef_zero/solr/query/unary_operator.rb +43 -0
  55. data/lib/chef_zero/solr/solr_doc.rb +62 -0
  56. data/lib/chef_zero/solr/solr_parser.rb +193 -0
  57. data/lib/chef_zero/version.rb +3 -0
  58. data/spec/run.rb +25 -0
  59. data/spec/support/pedant.rb +117 -0
  60. data/spec/support/stickywicket.pem +27 -0
  61. metadata +204 -0
data/LICENSE ADDED
@@ -0,0 +1,201 @@
1
+ Apache License
2
+ Version 2.0, January 2004
3
+ http://www.apache.org/licenses/
4
+
5
+ TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION
6
+
7
+ 1. Definitions.
8
+
9
+ "License" shall mean the terms and conditions for use, reproduction,
10
+ and distribution as defined by Sections 1 through 9 of this document.
11
+
12
+ "Licensor" shall mean the copyright owner or entity authorized by
13
+ the copyright owner that is granting the License.
14
+
15
+ "Legal Entity" shall mean the union of the acting entity and all
16
+ other entities that control, are controlled by, or are under common
17
+ control with that entity. For the purposes of this definition,
18
+ "control" means (i) the power, direct or indirect, to cause the
19
+ direction or management of such entity, whether by contract or
20
+ otherwise, or (ii) ownership of fifty percent (50%) or more of the
21
+ outstanding shares, or (iii) beneficial ownership of such entity.
22
+
23
+ "You" (or "Your") shall mean an individual or Legal Entity
24
+ exercising permissions granted by this License.
25
+
26
+ "Source" form shall mean the preferred form for making modifications,
27
+ including but not limited to software source code, documentation
28
+ source, and configuration files.
29
+
30
+ "Object" form shall mean any form resulting from mechanical
31
+ transformation or translation of a Source form, including but
32
+ not limited to compiled object code, generated documentation,
33
+ and conversions to other media types.
34
+
35
+ "Work" shall mean the work of authorship, whether in Source or
36
+ Object form, made available under the License, as indicated by a
37
+ copyright notice that is included in or attached to the work
38
+ (an example is provided in the Appendix below).
39
+
40
+ "Derivative Works" shall mean any work, whether in Source or Object
41
+ form, that is based on (or derived from) the Work and for which the
42
+ editorial revisions, annotations, elaborations, or other modifications
43
+ represent, as a whole, an original work of authorship. For the purposes
44
+ of this License, Derivative Works shall not include works that remain
45
+ separable from, or merely link (or bind by name) to the interfaces of,
46
+ the Work and Derivative Works thereof.
47
+
48
+ "Contribution" shall mean any work of authorship, including
49
+ the original version of the Work and any modifications or additions
50
+ to that Work or Derivative Works thereof, that is intentionally
51
+ submitted to Licensor for inclusion in the Work by the copyright owner
52
+ or by an individual or Legal Entity authorized to submit on behalf of
53
+ the copyright owner. For the purposes of this definition, "submitted"
54
+ means any form of electronic, verbal, or written communication sent
55
+ to the Licensor or its representatives, including but not limited to
56
+ communication on electronic mailing lists, source code control systems,
57
+ and issue tracking systems that are managed by, or on behalf of, the
58
+ Licensor for the purpose of discussing and improving the Work, but
59
+ excluding communication that is conspicuously marked or otherwise
60
+ designated in writing by the copyright owner as "Not a Contribution."
61
+
62
+ "Contributor" shall mean Licensor and any individual or Legal Entity
63
+ on behalf of whom a Contribution has been received by Licensor and
64
+ subsequently incorporated within the Work.
65
+
66
+ 2. Grant of Copyright License. Subject to the terms and conditions of
67
+ this License, each Contributor hereby grants to You a perpetual,
68
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
69
+ copyright license to reproduce, prepare Derivative Works of,
70
+ publicly display, publicly perform, sublicense, and distribute the
71
+ Work and such Derivative Works in Source or Object form.
72
+
73
+ 3. Grant of Patent License. Subject to the terms and conditions of
74
+ this License, each Contributor hereby grants to You a perpetual,
75
+ worldwide, non-exclusive, no-charge, royalty-free, irrevocable
76
+ (except as stated in this section) patent license to make, have made,
77
+ use, offer to sell, sell, import, and otherwise transfer the Work,
78
+ where such license applies only to those patent claims licensable
79
+ by such Contributor that are necessarily infringed by their
80
+ Contribution(s) alone or by combination of their Contribution(s)
81
+ with the Work to which such Contribution(s) was submitted. If You
82
+ institute patent litigation against any entity (including a
83
+ cross-claim or counterclaim in a lawsuit) alleging that the Work
84
+ or a Contribution incorporated within the Work constitutes direct
85
+ or contributory patent infringement, then any patent licenses
86
+ granted to You under this License for that Work shall terminate
87
+ as of the date such litigation is filed.
88
+
89
+ 4. Redistribution. You may reproduce and distribute copies of the
90
+ Work or Derivative Works thereof in any medium, with or without
91
+ modifications, and in Source or Object form, provided that You
92
+ meet the following conditions:
93
+
94
+ (a) You must give any other recipients of the Work or
95
+ Derivative Works a copy of this License; and
96
+
97
+ (b) You must cause any modified files to carry prominent notices
98
+ stating that You changed the files; and
99
+
100
+ (c) You must retain, in the Source form of any Derivative Works
101
+ that You distribute, all copyright, patent, trademark, and
102
+ attribution notices from the Source form of the Work,
103
+ excluding those notices that do not pertain to any part of
104
+ the Derivative Works; and
105
+
106
+ (d) If the Work includes a "NOTICE" text file as part of its
107
+ distribution, then any Derivative Works that You distribute must
108
+ include a readable copy of the attribution notices contained
109
+ within such NOTICE file, excluding those notices that do not
110
+ pertain to any part of the Derivative Works, in at least one
111
+ of the following places: within a NOTICE text file distributed
112
+ as part of the Derivative Works; within the Source form or
113
+ documentation, if provided along with the Derivative Works; or,
114
+ within a display generated by the Derivative Works, if and
115
+ wherever such third-party notices normally appear. The contents
116
+ of the NOTICE file are for informational purposes only and
117
+ do not modify the License. You may add Your own attribution
118
+ notices within Derivative Works that You distribute, alongside
119
+ or as an addendum to the NOTICE text from the Work, provided
120
+ that such additional attribution notices cannot be construed
121
+ as modifying the License.
122
+
123
+ You may add Your own copyright statement to Your modifications and
124
+ may provide additional or different license terms and conditions
125
+ for use, reproduction, or distribution of Your modifications, or
126
+ for any such Derivative Works as a whole, provided Your use,
127
+ reproduction, and distribution of the Work otherwise complies with
128
+ the conditions stated in this License.
129
+
130
+ 5. Submission of Contributions. Unless You explicitly state otherwise,
131
+ any Contribution intentionally submitted for inclusion in the Work
132
+ by You to the Licensor shall be under the terms and conditions of
133
+ this License, without any additional terms or conditions.
134
+ Notwithstanding the above, nothing herein shall supersede or modify
135
+ the terms of any separate license agreement you may have executed
136
+ with Licensor regarding such Contributions.
137
+
138
+ 6. Trademarks. This License does not grant permission to use the trade
139
+ names, trademarks, service marks, or product names of the Licensor,
140
+ except as required for reasonable and customary use in describing the
141
+ origin of the Work and reproducing the content of the NOTICE file.
142
+
143
+ 7. Disclaimer of Warranty. Unless required by applicable law or
144
+ agreed to in writing, Licensor provides the Work (and each
145
+ Contributor provides its Contributions) on an "AS IS" BASIS,
146
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or
147
+ implied, including, without limitation, any warranties or conditions
148
+ of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A
149
+ PARTICULAR PURPOSE. You are solely responsible for determining the
150
+ appropriateness of using or redistributing the Work and assume any
151
+ risks associated with Your exercise of permissions under this License.
152
+
153
+ 8. Limitation of Liability. In no event and under no legal theory,
154
+ whether in tort (including negligence), contract, or otherwise,
155
+ unless required by applicable law (such as deliberate and grossly
156
+ negligent acts) or agreed to in writing, shall any Contributor be
157
+ liable to You for damages, including any direct, indirect, special,
158
+ incidental, or consequential damages of any character arising as a
159
+ result of this License or out of the use or inability to use the
160
+ Work (including but not limited to damages for loss of goodwill,
161
+ work stoppage, computer failure or malfunction, or any and all
162
+ other commercial damages or losses), even if such Contributor
163
+ has been advised of the possibility of such damages.
164
+
165
+ 9. Accepting Warranty or Additional Liability. While redistributing
166
+ the Work or Derivative Works thereof, You may choose to offer,
167
+ and charge a fee for, acceptance of support, warranty, indemnity,
168
+ or other liability obligations and/or rights consistent with this
169
+ License. However, in accepting such obligations, You may act only
170
+ on Your own behalf and on Your sole responsibility, not on behalf
171
+ of any other Contributor, and only if You agree to indemnify,
172
+ defend, and hold each Contributor harmless for any liability
173
+ incurred by, or claims asserted against, such Contributor by reason
174
+ of your accepting any such warranty or additional liability.
175
+
176
+ END OF TERMS AND CONDITIONS
177
+
178
+ APPENDIX: How to apply the Apache License to your work.
179
+
180
+ To apply the Apache License to your work, attach the following
181
+ boilerplate notice, with the fields enclosed by brackets "[]"
182
+ replaced with your own identifying information. (Don't include
183
+ the brackets!) The text should be enclosed in the appropriate
184
+ comment syntax for the file format. We also recommend that a
185
+ file or class name and description of purpose be included on the
186
+ same "printed page" as the copyright notice for easier
187
+ identification within third-party archives.
188
+
189
+ Copyright [yyyy] [name of copyright owner]
190
+
191
+ Licensed under the Apache License, Version 2.0 (the "License");
192
+ you may not use this file except in compliance with the License.
193
+ You may obtain a copy of the License at
194
+
195
+ http://www.apache.org/licenses/LICENSE-2.0
196
+
197
+ Unless required by applicable law or agreed to in writing, software
198
+ distributed under the License is distributed on an "AS IS" BASIS,
199
+ WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
200
+ See the License for the specific language governing permissions and
201
+ limitations under the License.
data/README.md ADDED
@@ -0,0 +1,145 @@
1
+ Chef Zero
2
+ =========
3
+
4
+ Description
5
+ -----------
6
+ Chef Zero is a simple, easy-install, in-memory Chef server that can be useful
7
+ for Chef Client testing and chef-solo-like tasks that require a full Chef
8
+ Cerver. It IS intended to be simple, Chef 11 compliant, easy to run and fast
9
+ to start. It is NOT intended to be secure, scalable, performant or persistent.
10
+ It does NO input validation, authentication or authorization (it will not
11
+ throw a 400, 401 or 403). It does not save data, and will start up empty each
12
+ time you start it.
13
+
14
+ Because Chef Zero runs in memory, it's super fast and lightweight. This makes
15
+ it perfect for testing against a "real" Chef Server without mocking the
16
+ entire Internet.
17
+
18
+
19
+ Installation
20
+ ------------
21
+ This server can be installed as a Ruby Gem.
22
+
23
+ $ gem install chef-zero
24
+
25
+ If you're using bundler, add `chef-zero` as a development dependency:
26
+
27
+ ```ruby
28
+ group :development do
29
+ gem 'chef-zero'
30
+ end
31
+ ```
32
+
33
+ Or in a `.gemspec`
34
+
35
+ ```ruby
36
+ s.add_development_dependency 'chef-zero'
37
+ ```
38
+
39
+ You can also clone the source repository and install it using `rake install`.
40
+
41
+ Usage
42
+ -----
43
+ One of chef-zero's primary uses is as a small test server for people writing and
44
+ testing clients. Here's a simple example of starting it up:
45
+
46
+ ```ruby
47
+ require 'chef_zero/server'
48
+ server = ChefZero::Server.new(port: 4000)
49
+ server.start
50
+ ```
51
+
52
+ This will create a server instance in the foreground. To stop the server:
53
+
54
+ ```ruby
55
+ server.stop
56
+ ```
57
+
58
+ This is great for debugging and logging requests, but you'll probably want
59
+ to run this in the background so you have full control of your thread.
60
+
61
+ To run Chef Zero in the background, simply issue the `start_background` command:
62
+
63
+ ```ruby
64
+ require 'chef_zero/server'
65
+ server = ChefZero::Server.new(port: 4000)
66
+ server.start_background
67
+ ```
68
+
69
+ You can stop the server the same way:
70
+
71
+ ```ruby
72
+ server.stop
73
+ ```
74
+
75
+ ### Valid Options
76
+ You may currently pass the following options to the initializer:
77
+
78
+ - `host` - the host to run on (Default: '127.0.0.1')
79
+ - `port` - the port to run on (Default: 8889)
80
+ - `debug` - run in debug mode to see all requests and responses (Default: false)
81
+
82
+
83
+ CLI (Comand Line)
84
+ -----------------
85
+ If you don't want to use Chef Zero as a library, you can simply start an instance
86
+ with the included `chef-zero` executable:
87
+
88
+ $ chef-zero
89
+
90
+ Note, this will run in the foreground.
91
+
92
+ You now have a fully functional (empty) Chef Server running.
93
+
94
+ To try it out, go into the `chef-zero/playground` directory and run `knife`. It
95
+ will behave the same as a normal Chef Server, and all normal knife commands will
96
+ work (show, list, delete, from file, upload, download, diff ...). For example,
97
+ with +knife-essentials+ (or Chef 11) you can upload everything in the repo:
98
+
99
+ chef-zero/playground> knife upload .
100
+ Created remote/cookbooks/blah
101
+ Created remote/cookbooks/blork
102
+ Created remote/data_bags/foo/
103
+ Created remote/data_bags/foo/bar.json
104
+ Created remote/data_bags/foo/baz.json
105
+ Created remote/data_bags/foo/blarghle.json
106
+ Created remote/data_bags/foom/
107
+ Created remote/data_bags/foom/x.json
108
+ Created remote/data_bags/widdle/
109
+ Created remote/data_bags/widdle/blank.json
110
+ Created remote/data_bags/widdle/wow.json
111
+ Created remote/environments/desert.json
112
+ Created remote/environments/rainforest.json
113
+ Created remote/environments/semi_arid_plains.json
114
+
115
+ chef-zero/playground> knife environment list
116
+ _default
117
+ desert
118
+ rainforest
119
+ semi_arid_plains
120
+
121
+ To use it in your own repository, create a `knife.rb` like so:
122
+
123
+ chef_server_url 'http://127.0.0.1:8889'
124
+ node_name 'stickywicket'
125
+ client_key 'path_to_any_pem_file.pem'
126
+
127
+ And use knife like you normally would.
128
+
129
+ Since Chef Zero does no authentication, any `.pem` file will do. The client just
130
+ needs something to sign requests with (which will be ignored on the server). Even
131
+ though it's ignored, the `.pem` must still be a valid format.
132
+
133
+ Now, stop the Chef Zero server and all the data is gone!
134
+
135
+ Run `chef-zero --help` to see a list of the supported flags and options:
136
+
137
+ ```text
138
+ Usage: chef-zero [ARGS]
139
+ -H, --host HOST Host to bind to (default: 127.0.0.1)
140
+ -p, --port PORT Port to listen on
141
+ --[no-]generate-keys Whether to generate actual keys or fake it (faster). Default: false.
142
+ -l, --log-level LEVEL Set the output log level
143
+ -h, --help Show this message
144
+ --version Show version
145
+ ```
data/Rakefile ADDED
@@ -0,0 +1,11 @@
1
+ require 'bundler'
2
+ require 'bundler/gem_helper'
3
+
4
+ # Get build, install and release tasks for each gem flavor
5
+ %w(chef-zero chef-zero-x86-mingw32).each do |gem_name|
6
+ Bundler::GemHelper.install_tasks :name => gem_name
7
+ end
8
+
9
+ task :spec do
10
+ sh 'ruby spec/run.rb'
11
+ end
data/bin/chef-zero ADDED
@@ -0,0 +1,43 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ require 'rubygems'
4
+ $:.unshift(File.expand_path(File.join(File.dirname(__FILE__), "..", "lib")))
5
+
6
+ require 'chef_zero/version'
7
+ require 'chef_zero/server'
8
+ require 'optparse'
9
+
10
+ options = {}
11
+
12
+ OptionParser.new do |opts|
13
+ opts.banner = "Usage: chef-zero [ARGS]"
14
+
15
+ opts.on("-H", "--host HOST", "Host to bind to (default: 127.0.0.1)") do |value|
16
+ options[:host] = value
17
+ end
18
+
19
+ opts.on("-p", "--port PORT", Integer, "Port to listen on") do |value|
20
+ options[:port] = value
21
+ end
22
+
23
+ opts.on("--[no-]generate-keys", "Whether to generate actual keys or fake it (faster). Default: false.") do |value|
24
+ options[:generate_real_keys] = value
25
+ end
26
+
27
+ opts.on("-l", "--log-level LEVEL", "Set the output log level") do |value|
28
+ options[:log_level] = value
29
+ end
30
+
31
+ opts.on_tail("-h", "--help", "Show this message") do
32
+ puts opts
33
+ exit
34
+ end
35
+
36
+ opts.on_tail("--version", "Show version") do
37
+ puts ChefZero::VERSION
38
+ exit
39
+ end
40
+ end.parse!
41
+
42
+ server = ChefZero::Server.new(options)
43
+ server.start(:publish => true)
data/lib/chef_zero.rb ADDED
@@ -0,0 +1,7 @@
1
+ module ChefZero
2
+ require 'chef_zero/log'
3
+
4
+ CERTIFICATE = "-----BEGIN CERTIFICATE-----\nMIIDMzCCApygAwIBAgIBATANBgkqhkiG9w0BAQUFADCBnjELMAkGA1UEBhMCVVMx\nEzARBgNVBAgMCldhc2hpbmd0b24xEDAOBgNVBAcMB1NlYXR0bGUxFjAUBgNVBAoM\nDU9wc2NvZGUsIEluYy4xHDAaBgNVBAsME0NlcnRpZmljYXRlIFNlcnZpY2UxMjAw\nBgNVBAMMKW9wc2NvZGUuY29tL2VtYWlsQWRkcmVzcz1hdXRoQG9wc2NvZGUuY29t\nMB4XDTEyMTEyMTAwMzQyMVoXDTIyMTExOTAwMzQyMVowgZsxEDAOBgNVBAcTB1Nl\nYXR0bGUxEzARBgNVBAgTCldhc2hpbmd0b24xCzAJBgNVBAYTAlVTMRwwGgYDVQQL\nExNDZXJ0aWZpY2F0ZSBTZXJ2aWNlMRYwFAYDVQQKEw1PcHNjb2RlLCBJbmMuMS8w\nLQYDVQQDFCZVUkk6aHR0cDovL29wc2NvZGUuY29tL0dVSURTL3VzZXJfZ3VpZDCC\nASIwDQYJKoZIhvcNAQEBBQADggEPADCCAQoCggEBANLDmPbR71bS2esZlZh/HfC6\n0azXFjl2677wq2ovk9xrUb0Ui4ZLC66TqQ9C/RBzOjXU4TRf3hgPTqvlCgHusl0d\nIcLCrsSl6kPEhJpYWWfRoroIAwf82A9yLQekhqXZEXu5EKkwoUMqyF6m0ZCasaE1\ny8niQxdLAsk3ady/CGQlFqHTPKFfU5UASR2LRtYC1MCIvJHDFRKAp9kPJbQo9P37\nZ8IU7cDudkZFgNLmDixlWsh7C0ghX8fgAlj1P6FgsFufygam973k79GhIP54dELB\nc0S6E8ekkRSOXU9jX/IoiXuFglBvFihAdhvED58bMXzj2AwXUyeAlxItnvs+NVUC\nAwEAATANBgkqhkiG9w0BAQUFAAOBgQBkFZRbMoywK3hb0/X7MXmPYa7nlfnd5UXq\nr2n32ettzZNmEPaI2d1j+//nL5qqhOlrWPS88eKEPnBOX/jZpUWOuAAddnrvFzgw\nrp/C2H7oMT+29F+5ezeViLKbzoFYb4yECHBoi66IFXNae13yj7taMboBeUmE664G\nTB/MZpRr8g==\n-----END CERTIFICATE-----\n"
5
+ PUBLIC_KEY = "-----BEGIN PUBLIC KEY-----\nMIIBIjANBgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEA0sOY9tHvVtLZ6xmVmH8d\n8LrRrNcWOXbrvvCrai+T3GtRvRSLhksLrpOpD0L9EHM6NdThNF/eGA9Oq+UKAe6y\nXR0hwsKuxKXqQ8SEmlhZZ9GiuggDB/zYD3ItB6SGpdkRe7kQqTChQyrIXqbRkJqx\noTXLyeJDF0sCyTdp3L8IZCUWodM8oV9TlQBJHYtG1gLUwIi8kcMVEoCn2Q8ltCj0\n/ftnwhTtwO52RkWA0uYOLGVayHsLSCFfx+ACWPU/oWCwW5/KBqb3veTv0aEg/nh0\nQsFzRLoTx6SRFI5dT2Nf8iiJe4WCUG8WKEB2G8QPnxsxfOPYDBdTJ4CXEi2e+z41\nVQIDAQAB\n-----END PUBLIC KEY-----\n"
6
+ PRIVATE_KEY = "-----BEGIN RSA PRIVATE KEY-----\nMIIEpAIBAAKCAQEA0sOY9tHvVtLZ6xmVmH8d8LrRrNcWOXbrvvCrai+T3GtRvRSL\nhksLrpOpD0L9EHM6NdThNF/eGA9Oq+UKAe6yXR0hwsKuxKXqQ8SEmlhZZ9GiuggD\nB/zYD3ItB6SGpdkRe7kQqTChQyrIXqbRkJqxoTXLyeJDF0sCyTdp3L8IZCUWodM8\noV9TlQBJHYtG1gLUwIi8kcMVEoCn2Q8ltCj0/ftnwhTtwO52RkWA0uYOLGVayHsL\nSCFfx+ACWPU/oWCwW5/KBqb3veTv0aEg/nh0QsFzRLoTx6SRFI5dT2Nf8iiJe4WC\nUG8WKEB2G8QPnxsxfOPYDBdTJ4CXEi2e+z41VQIDAQABAoIBAALhqbW2KQ+G0nPk\nZacwFbi01SkHx8YBWjfCEpXhEKRy0ytCnKW5YO+CFU2gHNWcva7+uhV9OgwaKXkw\nKHLeUJH1VADVqI4Htqw2g5mYm6BPvWnNsjzpuAp+BR+VoEGkNhj67r9hatMAQr0I\nitTvSH5rvd2EumYXIHKfz1K1SegUk1u1EL1RcMzRmZe4gDb6eNBs9Sg4im4ybTG6\npPIytA8vBQVWhjuAR2Tm+wZHiy0Az6Vu7c2mS07FSX6FO4E8SxWf8idaK9ijMGSq\nFvIS04mrY6XCPUPUC4qm1qNnhDPpOr7CpI2OO98SqGanStS5NFlSFXeXPpM280/u\nfZUA0AECgYEA+x7QUnffDrt7LK2cX6wbvn4mRnFxet7bJjrfWIHf+Rm0URikaNma\nh0/wNKpKBwIH+eHK/LslgzcplrqPytGGHLOG97Gyo5tGAzyLHUWBmsNkRksY2sPL\nuHq6pYWJNkqhnWGnIbmqCr0EWih82x/y4qxbJYpYqXMrit0wVf7yAgkCgYEA1twI\ngFaXqesetTPoEHSQSgC8S4D5/NkdriUXCYb06REcvo9IpFMuiOkVUYNN5d3MDNTP\nIdBicfmvfNELvBtXDomEUD8ls1UuoTIXRNGZ0VsZXu7OErXCK0JKNNyqRmOwcvYL\nJRqLfnlei5Ndo1lu286yL74c5rdTLs/nI2p4e+0CgYB079ZmcLeILrmfBoFI8+Y/\ngJLmPrFvXBOE6+lRV7kqUFPtZ6I3yQzyccETZTDvrnx0WjaiFavUPH27WMjY01S2\nTMtO0Iq1MPsbSrglO1as8MvjB9ldFcvp7gy4Q0Sv6XT0yqJ/S+vo8Df0m+H4UBpU\nf5o6EwBSd/UQxwtZIE0lsQKBgQCswfjX8Eg8KL/lJNpIOOE3j4XXE9ptksmJl2sB\njxDnQYoiMqVO808saHVquC/vTrpd6tKtNpehWwjeTFuqITWLi8jmmQ+gNTKsC9Gn\n1Pxf2Gb67PqnEpwQGln+TRtgQ5HBrdHiQIi+5am+gnw89pDrjjO5rZwhanAo6KPJ\n1zcPNQKBgQDxFu8v4frDmRNCVaZS4f1B6wTrcMrnibIDlnzrK9GG6Hz1U7dDv8s8\nNf4UmeMzDXjlPWZVOvS5+9HKJPdPj7/onv8B2m18+lcgTTDJBkza7R1mjL1Cje/Z\nKcVGsryKN6cjE7yCDasnA7R2rVBV/7NWeJV77bmzT5O//rW4yIfUIg==\n-----END RSA PRIVATE KEY-----\n"
7
+ end
@@ -0,0 +1,223 @@
1
+ require 'digest/md5'
2
+ require 'hashie/mash'
3
+
4
+ module ChefZero
5
+ module CookbookData
6
+ def self.to_hash(cookbook, name, version=nil)
7
+ frozen = false
8
+ if cookbook.has_key?(:frozen)
9
+ frozen = cookbook[:frozen]
10
+ cookbook = cookbook.dup
11
+ cookbook.delete(:frozen)
12
+ end
13
+
14
+ result = files_from(cookbook)
15
+ recipe_names = result[:recipes].map do |recipe|
16
+ recipe_name = recipe[:name][0..-2]
17
+ recipe_name == 'default' ? name : "#{name}::#{recipe_name}"
18
+ end
19
+ result[:metadata] = metadata_from(cookbook, name, version, recipe_names)
20
+ result[:name] = "#{name}-#{result[:metadata][:version]}"
21
+ result[:json_class] = 'Chef::CookbookVersion'
22
+ result[:cookbook_name] = name
23
+ result[:version] = result[:metadata][:version]
24
+ result[:chef_type] = 'cookbook_version'
25
+ result[:frozen?] = true if frozen
26
+ result
27
+ end
28
+
29
+ def self.metadata_from(directory, name, version, recipe_names)
30
+ metadata = PretendCookbookMetadata.new(PretendCookbook.new(name, recipe_names))
31
+ # If both .rb and .json exist, read .rb
32
+ # TODO if recipes has 3 recipes in it, and the Ruby/JSON has only one, should
33
+ # the resulting recipe list have 1, or 3-4 recipes in it?
34
+ if has_child(directory, 'metadata.rb')
35
+ metadata.instance_eval(read_file(directory, 'metadata.rb'))
36
+ elsif has_child(directory, 'metadata.json')
37
+ metadata.from_json(read_file(directory, 'metadata.json'))
38
+ end
39
+ result = {}
40
+ metadata.to_hash.each_pair do |key,value|
41
+ result[key.to_sym] = value
42
+ end
43
+ result[:version] = version if version
44
+ result
45
+ end
46
+
47
+ private
48
+
49
+ # Just enough cookbook to make a Metadata object
50
+ class PretendCookbook
51
+ def initialize(name, fully_qualified_recipe_names)
52
+ @name = name
53
+ @fully_qualified_recipe_names = fully_qualified_recipe_names
54
+ end
55
+ attr_reader :name, :fully_qualified_recipe_names
56
+ end
57
+
58
+ # Handles loading configuration values from a Chef config file
59
+ #
60
+ # @author Justin Campbell <justin.campbell@riotgames.com>
61
+ class PretendCookbookMetadata < Hash
62
+ # @param [String] path
63
+ def initialize(cookbook)
64
+ self.name(cookbook.name)
65
+ self.recipes(cookbook.fully_qualified_recipe_names)
66
+ %w(dependencies supports recommendations suggestions conflicting providing replacing recipes).each do |cookbook_arg|
67
+ self[cookbook_arg.to_sym] = Hashie::Mash.new
68
+ end
69
+ end
70
+
71
+ def from_json(filepath)
72
+ self.merge!(JSON.parse(File.read(filepath)))
73
+ end
74
+
75
+ private
76
+
77
+ def depends(cookbook, *version_constraints)
78
+ cookbook_arg(:dependencies, cookbook, version_constraints)
79
+ end
80
+
81
+ def supports(cookbook, *version_constraints)
82
+ cookbook_arg(:supports, cookbook, version_constraints)
83
+ end
84
+
85
+ def recommends(cookbook, *version_constraints)
86
+ cookbook_arg(:recommendations, cookbook, version_constraints)
87
+ end
88
+
89
+ def suggests(cookbook, *version_constraints)
90
+ cookbook_arg(:suggestions, cookbook, version_constraints)
91
+ end
92
+
93
+ def conflicts(cookbook, *version_constraints)
94
+ cookbook_arg(:conflicting, cookbook, version_constraints)
95
+ end
96
+
97
+ def provides(cookbook, *version_constraints)
98
+ cookbook_arg(:providing, cookbook, version_constraints)
99
+ end
100
+
101
+ def replaces(cookbook, *version_constraints)
102
+ cookbook_arg(:replacing, cookbook, version_constraints)
103
+ end
104
+
105
+ def recipe(recipe, description)
106
+ self[:recipes][recipe] = description
107
+ end
108
+
109
+ def attribute(name, options)
110
+ self[:attributes][name] = options
111
+ end
112
+
113
+ def grouping(name, options)
114
+ self[:grouping][name] = options
115
+ end
116
+
117
+ def cookbook_arg(key, cookbook, version_constraints)
118
+ self[key][cookbook] = version_constraints.first || ">= 0.0.0"
119
+ end
120
+
121
+ def method_missing(key, value = nil)
122
+ if value.nil?
123
+ self[key.to_sym]
124
+ else
125
+ store key.to_sym, value
126
+ end
127
+ end
128
+ end
129
+
130
+ def self.files_from(directory)
131
+ # TODO some support .rb only
132
+ result = {
133
+ :attributes => load_child_files(directory, 'attributes', false),
134
+ :definitions => load_child_files(directory, 'definitions', false),
135
+ :recipes => load_child_files(directory, 'recipes', false),
136
+ :libraries => load_child_files(directory, 'libraries', false),
137
+ :templates => load_child_files(directory, 'templates', true),
138
+ :files => load_child_files(directory, 'files', true),
139
+ :resources => load_child_files(directory, 'resources', true),
140
+ :providers => load_child_files(directory, 'providers', true),
141
+ :root_files => load_files(directory, false)
142
+ }
143
+ set_specificity(result[:templates])
144
+ set_specificity(result[:files])
145
+ result
146
+ end
147
+
148
+ def self.has_child(directory, name)
149
+ if directory.is_a?(Hash)
150
+ directory.has_key?(name)
151
+ else
152
+ directory.child(name).exists?
153
+ end
154
+ end
155
+
156
+ def self.read_file(directory, name)
157
+ if directory.is_a?(Hash)
158
+ directory[name]
159
+ else
160
+ directory.child(name).read
161
+ end
162
+ end
163
+
164
+ def self.get_directory(directory, name)
165
+ if directory.is_a?(Hash)
166
+ directory[name].is_a?(Hash) ? directory[name] : nil
167
+ else
168
+ result = directory.child(name)
169
+ result.dir? ? result : nil
170
+ end
171
+ end
172
+
173
+ def self.list(directory)
174
+ if directory.is_a?(Hash)
175
+ directory.keys
176
+ else
177
+ directory.children.map { |c| c.name }
178
+ end
179
+ end
180
+
181
+ def self.load_child_files(parent, key, recursive)
182
+ result = load_files(get_directory(parent, key), recursive)
183
+ result.each do |file|
184
+ file[:path] = "#{key}/#{file[:path]}"
185
+ end
186
+ result
187
+ end
188
+
189
+ def self.load_files(directory, recursive)
190
+ result = []
191
+ if directory
192
+ list(directory).each do |child_name|
193
+ dir = get_directory(directory, child_name)
194
+ if dir
195
+ if recursive
196
+ result += load_child_files(directory, child_name, recursive)
197
+ end
198
+ else
199
+ result += load_file(read_file(directory, child_name), child_name)
200
+ end
201
+ end
202
+ end
203
+ result
204
+ end
205
+
206
+ def self.load_file(value, name)
207
+ [{
208
+ :name => name,
209
+ :path => name,
210
+ :checksum => Digest::MD5.hexdigest(value),
211
+ :specificity => 'default'
212
+ }]
213
+ end
214
+
215
+ def self.set_specificity(files)
216
+ files.each do |file|
217
+ parts = file[:path].split('/')
218
+ raise "Only directories are allowed directly under templates or files: #{file[:path]}" if parts.size == 2
219
+ file[:specificity] = parts[1]
220
+ end
221
+ end
222
+ end
223
+ end