Merge from HEAD. Signed off by tmack.
[plcapi.git] / PLC / Methods / AddSliceAttribute.py
1 from PLC.Faults import *
2 from PLC.Method import Method
3 from PLC.Parameter import Parameter, Mixed
4 from PLC.SliceAttributeTypes import SliceAttributeType, SliceAttributeTypes
5 from PLC.Slices import Slice, Slices
6 from PLC.Nodes import Node, Nodes
7 from PLC.SliceAttributes import SliceAttribute, SliceAttributes
8 from PLC.InitScripts import InitScript, InitScripts
9 from PLC.Auth import Auth
10
11 class AddSliceAttribute(Method):
12     """
13     Sets the specified attribute of the slice (or sliver, if
14     node_id_or_hostname is specified) to the specified value.
15
16     Attributes may require the caller to have a particular role in
17     order to be set or changed. Users may only set attributes of
18     slices or slivers of which they are members. PIs may only set
19     attributes of slices or slivers at their sites, or of which they
20     are members. Admins may set attributes of any slice or sliver.
21
22     Returns the new slice_attribute_id (> 0) if successful, faults
23     otherwise.
24     """
25
26     roles = ['admin', 'pi', 'user']
27
28     accepts = [
29         Auth(),
30         Mixed(SliceAttribute.fields['slice_id'],
31               SliceAttribute.fields['name']),
32         Mixed(SliceAttribute.fields['attribute_type_id'],
33               SliceAttribute.fields['name']),
34         Mixed(SliceAttribute.fields['value'],
35               InitScript.fields['initscript_id']),
36         Mixed(Node.fields['node_id'],
37               Node.fields['hostname'])
38         ]
39
40     returns = Parameter(int, 'New slice_attribute_id (> 0) if successful')
41
42     def call(self, auth, slice_id_or_name, attribute_type_id_or_name, value, node_id_or_hostname = None):
43         slices = Slices(self.api, [slice_id_or_name])
44         if not slices:
45             raise PLCInvalidArgument, "No such slice"
46         slice = slices[0]
47
48         attribute_types = SliceAttributeTypes(self.api, [attribute_type_id_or_name])
49         if not attribute_types:
50             raise PLCInvalidArgument, "No such slice attribute type"
51         attribute_type = attribute_types[0]
52
53         if 'admin' not in self.caller['roles']:
54             if self.caller['person_id'] in slice['person_ids']:
55                 pass
56             elif 'pi' not in self.caller['roles']:
57                 raise PLCPermissionDenied, "Not a member of the specified slice"
58             elif slice['site_id'] not in self.caller['site_ids']:
59                 raise PLCPermissionDenied, "Specified slice not associated with any of your sites"
60
61             if attribute_type['min_role_id'] is not None and \
62                min(self.caller['role_ids']) > attribute_type['min_role_id']:
63                 raise PLCPermissionDenied, "Not allowed to set the specified slice attribute"
64
65         # if initscript is specified, validate value
66         if attribute_type['name'] in ['plc_initscript_id']:
67             initscripts = InitScripts(self.api, [value])
68             if not initscripts: 
69                 raise PLCInvalidArgument, "No such plc initscript"      
70
71         slice_attribute = SliceAttribute(self.api)
72         slice_attribute['slice_id'] = slice['slice_id']
73         slice_attribute['attribute_type_id'] = attribute_type['attribute_type_id']
74         slice_attribute['value'] = value
75
76         # Sliver attribute if node is specified
77         if node_id_or_hostname is not None:
78             nodes = Nodes(self.api, [node_id_or_hostname])
79             if not nodes:
80                 raise PLCInvalidArgument, "No such node"
81             node = nodes[0]
82             
83             if node['node_id'] not in slice['node_ids']:
84                 raise PLCInvalidArgument, "Node not in the specified slice"
85
86             slice_attribute['node_id'] = node['node_id']
87
88         slice_attribute.sync()
89         self.event_objects = {'SliceAttribute': [slice_attribute['slice_attribute_id']]}
90
91         return slice_attribute['slice_attribute_id']