diff options
author | Steven Danna <steve@opscode.com> | 2014-11-19 13:49:27 +0000 |
---|---|---|
committer | Lamont Granquist <lamont@scriptkiddie.org> | 2015-01-25 14:06:51 -0800 |
commit | 875b90d97a5c65efbda7853b4db25c18e808bd61 (patch) | |
tree | 51d8ea95290afde750d6b40e24cf254b3c5a2aa1 /lib/chef/org.rb | |
parent | f923755a505c31745f6df14e201149128ba4ebec (diff) | |
download | chef-875b90d97a5c65efbda7853b4db25c18e808bd61.tar.gz |
Add Chef::Org model class
Chef Server 12 added multi-tenancy to the open source Chef Server. An
organization is the Chef Servers name for a tenant. The Chef::Org
class provides a model of an organization to facilitate writting knife
commands that can interact with (create, destroy, list) Chef
organizations.
This implementation is copied directly from knife-opc, the knife
plugin currently in use by chef-server-ctl:
https://github.com/opscode/knife-opc
Diffstat (limited to 'lib/chef/org.rb')
-rw-r--r-- | lib/chef/org.rb | 143 |
1 files changed, 143 insertions, 0 deletions
diff --git a/lib/chef/org.rb b/lib/chef/org.rb new file mode 100644 index 0000000000..bbc4263835 --- /dev/null +++ b/lib/chef/org.rb @@ -0,0 +1,143 @@ +require 'chef/json_compat' +require 'chef/mixin/params_validate' +require 'chef/rest' + +class Chef + class Org + + include Chef::Mixin::ParamsValidate + + def initialize(name='') + @name = name + @full_name = '' + # The Chef API returns the private key of the validator + # client on create + @private_key = nil + @guid = nil + end + + def chef_rest + @chef_rest ||= Chef::REST.new(Chef::Config[:chef_server_root]) + end + + def name(arg=nil) + set_or_return(:name, arg, + :regex => /^[a-z0-9\-_]+$/) + end + + def full_name(arg=nil) + set_or_return(:full_name, + arg, :kind_of => String) + end + + def private_key(arg=nil) + set_or_return(:private_key, + arg, :kind_of => String) + end + + def guid(arg=nil) + set_or_return(:guid, + arg, :kind_of => String) + end + + def to_hash + result = { + "name" => @name, + "full_name" => @full_name + } + result["private_key"] = @private_key if @private_key + result["guid"] = @guid if @guid + result + end + + def to_json(*a) + Chef::JSONCompat.to_json(to_hash, *a) + end + + def create + payload = {:name => self.name, :full_name => self.full_name} + new_org = chef_rest.post_rest("organizations", payload) + Chef::Org.from_hash(self.to_hash.merge(new_org)) + end + + def update + payload = {:name => self.name, :full_name => self.full_name} + new_org = chef_rest.put_rest("organizations/#{name}", payload) + Chef::Org.from_hash(self.to_hash.merge(new_org)) + end + + def destroy + chef_rest.delete_rest("organizations/#{@name}") + end + + def save + begin + create + rescue Net::HTTPServerException => e + if e.response.code == "409" + update + else + raise e + end + end + end + + def associate_user(username) + request_body = {:user => username} + response = chef_rest.post_rest "organizations/#{@name}/association_requests", request_body + association_id = response["uri"].split("/").last + chef_rest.put_rest "users/#{username}/association_requests/#{association_id}", { :response => 'accept' } + end + + def dissociate_user(username) + chef_rest.delete_rest "organizations/#{name}/users/#{username}" + end + + def add_user_to_group(groupname, username) + group = chef_rest.get_rest "organizations/#{name}/groups/#{groupname}" + body_hash = { + :groupname => "#{groupname}", + :actors => { + "users" => group["actors"].concat([username]), + "groups" => group["groups"] + } + } + chef_rest.put_rest "organizations/#{name}/groups/#{groupname}", body_hash + end + + # Class methods + def self.from_hash(org_hash) + org = Chef::Org.new + org.name org_hash['name'] + org.full_name org_hash['full_name'] + org.private_key org_hash['private_key'] if org_hash.key?('private_key') + org.guid org_hash['guid'] if org_hash.key?('guid') + org + end + + def self.from_json(json) + Chef::Org.from_hash(Chef::JSONCompat.from_json(json)) + end + + class <<self + alias_method :json_create, :from_json + end + + def self.load(org_name) + response = Chef::REST.new(Chef::Config[:chef_server_root]).get_rest("organizations/#{org_name}") + Chef::Org.from_hash(response) + end + + def self.list(inflate=false) + orgs = Chef::REST.new(Chef::Config[:chef_server_root]).get_rest('organizations') + if inflate + orgs.inject({}) do |org_map, (name, _url)| + org_map[name] = Chef::Org.load(name) + org_map + end + else + orgs + end + end + end +end |