Statistics
| Branch: | Tag: | Revision:

root / snf-cyclades-app / synnefo / logic / utils.py @ a1baa42b

History | View | Annotate | Download (6 kB)

1
# Copyright 2011 GRNET S.A. All rights reserved.
2
#
3
# Redistribution and use in source and binary forms, with or without
4
# modification, are permitted provided that the following conditions
5
# are met:
6
#
7
#   1. Redistributions of source code must retain the above copyright
8
#      notice, this list of conditions and the following disclaimer.
9
#
10
#  2. Redistributions in binary form must reproduce the above copyright
11
#     notice, this list of conditions and the following disclaimer in the
12
#     documentation and/or other materials provided with the distribution.
13
#
14
# THIS SOFTWARE IS PROVIDED BY THE REGENTS AND CONTRIBUTORS ``AS IS'' AND
15
# ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
16
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
17
# ARE DISCLAIMED.  IN NO EVENT SHALL THE REGENTS OR CONTRIBUTORS BE LIABLE
18
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
19
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS
20
# OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
21
# HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT
22
# LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY
23
# OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
24
# SUCH DAMAGE.
25
#
26
# The views and conclusions contained in the software and documentation are
27
# those of the authors and should not be interpreted as representing official
28
# policies, either expressed or implied, of GRNET S.A.
29

    
30
# Utility functions
31

    
32
from synnefo.db.models import VirtualMachine, Network
33
from django.conf import settings
34
from copy import deepcopy
35

    
36

    
37
def id_from_instance_name(name):
38
    """Returns VirtualMachine's Django id, given a ganeti machine name.
39

40
    Strips the ganeti prefix atm. Needs a better name!
41

42
    """
43
    sname = str(name)
44
    if not sname.startswith(settings.BACKEND_PREFIX_ID):
45
        raise VirtualMachine.InvalidBackendIdError(sname)
46
    ns = sname.replace(settings.BACKEND_PREFIX_ID, "", 1)
47
    if not ns.isdigit():
48
        raise VirtualMachine.InvalidBackendIdError(sname)
49

    
50
    return int(ns)
51

    
52

    
53
def id_to_instance_name(id):
54
    return "%s%s" % (settings.BACKEND_PREFIX_ID, str(id))
55

    
56

    
57
def id_from_network_name(name):
58
    """Returns Network's Django id, given a ganeti network name.
59

60
    Strips the ganeti prefix atm. Needs a better name!
61

62
    """
63
    if not str(name).startswith(settings.BACKEND_PREFIX_ID):
64
        raise Network.InvalidBackendIdError(str(name))
65
    ns = str(name).replace(settings.BACKEND_PREFIX_ID + 'net-', "", 1)
66
    if not ns.isdigit():
67
        raise Network.InvalidBackendIdError(str(name))
68

    
69
    return int(ns)
70

    
71

    
72
def id_to_network_name(id):
73
    return "%snet-%s" % (settings.BACKEND_PREFIX_ID, str(id))
74

    
75

    
76
def id_from_nic_name(name):
77
    """Returns NIC's Django id, given a Ganeti's NIC name.
78

79
    """
80
    if not str(name).startswith(settings.BACKEND_PREFIX_ID):
81
        raise ValueError("Invalid NIC name: %s" % name)
82
    ns = str(name).replace(settings.BACKEND_PREFIX_ID + 'nic-', "", 1)
83
    if not ns.isdigit():
84
        raise ValueError("Invalid NIC name: %s" % name)
85

    
86
    return int(ns)
87

    
88

    
89
def get_rsapi_state(vm):
90
    """Returns the API state for a virtual machine
91

92
    The API state for an instance of VirtualMachine is derived as follows:
93

94
    * If the deleted flag has been set, it is "DELETED".
95
    * Otherwise, it is a mapping of the last state reported by Ganeti
96
      (vm.operstate) through the RSAPI_STATE_FROM_OPER_STATE dictionary.
97

98
      The last state reported by Ganeti is set whenever Ganeti reports
99
      successful completion of an operation. If Ganeti says an
100
      OP_INSTANCE_STARTUP operation succeeded, vm.operstate is set to
101
      "STARTED".
102

103
    * To support any transitional states defined by the API (only REBOOT for
104
    the time being) this mapping is amended with information reported by Ganeti
105
    regarding any outstanding operation. If an OP_INSTANCE_STARTUP had
106
    succeeded previously and an OP_INSTANCE_REBOOT has been reported as in
107
    progress, the API state is "REBOOT".
108

109
    """
110
    try:
111
        r = VirtualMachine.RSAPI_STATE_FROM_OPER_STATE[vm.operstate]
112
    except KeyError:
113
        return "UNKNOWN"
114
    # A machine is DELETED if the deleted flag has been set
115
    if vm.deleted:
116
        return "DELETED"
117
    # A machine is in REBOOT if an OP_INSTANCE_REBOOT request is in progress
118
    in_reboot = (r == "ACTIVE") and\
119
                (vm.backendopcode == "OP_INSTANCE_REBOOT") and\
120
                (vm.backendjobstatus in ("queued", "waiting", "running"))
121
    if in_reboot:
122
        return "REBOOT"
123
    in_resize = (r == "STOPPED") and\
124
                (vm.backendopcode == "OP_INSTANCE_MODIFY") and\
125
                (vm.task == "RESIZE") and \
126
                (vm.backendjobstatus in ("queued", "waiting", "running"))
127
    if in_resize:
128
        return "RESIZE"
129
    return r
130

    
131

    
132
TASK_STATE_FROM_ACTION = {
133
    "BUILD": "BULDING",
134
    "START": "STARTING",
135
    "STOP": "STOPPING",
136
    "REBOOT": "REBOOTING",
137
    "DESTROY": "DESTROYING",
138
    "RESIZE": "RESIZING",
139
    "CONNECT": "CONNECTING",
140
    "DISCONNECT": "DISCONNECTING"}
141

    
142

    
143
def get_task_state(vm):
144
    if vm.task is None:
145
        return ""
146
    try:
147
        return TASK_STATE_FROM_ACTION[vm.task]
148
    except KeyError:
149
        return "UNKNOWN"
150

    
151

    
152
OPCODE_TO_ACTION = {
153
    "OP_INSTANCE_CREATE": "BUILD",
154
    "OP_INSTANCE_START": "START",
155
    "OP_INSTANCE_STOP": "STOP",
156
    "OP_INSTANCE_REBOOT": "REBOOT",
157
    "OP_INSTANCE_REMOVE": "DESTROY"}
158

    
159

    
160
def get_action_from_opcode(opcode, job_fields):
161
    if opcode == "OP_INSTANCE_SET_PARAMS":
162
        nics = job_fields.get("nics")
163
        beparams = job_fields.get("beparams")
164
        if nics:
165
            #TODO: check the nic format
166
            return "CONNECT" or "DISCONNECT"
167
        elif beparams:
168
            return "RESIZE"
169
        else:
170
            return None
171
    else:
172
        return OPCODE_TO_ACTION.get(opcode, None)
173

    
174

    
175
def hide_pass(kw):
176
    if 'osparams' in kw and 'img_passwd' in kw['osparams']:
177
        kw1 = deepcopy(kw)
178
        kw1['osparams']['img_passwd'] = 'x' * 8
179
        return kw1
180
    else:
181
        return kw