~0x44/nova/bug838466

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
#!/usr/bin/env python
# vim: tabstop=4 shiftwidth=4 softtabstop=4

# Copyright 2010 United States Government as represented by the
# Administrator of the National Aeronautics and Space Administration.
# All Rights Reserved.
#
#    Licensed under the Apache License, Version 2.0 (the "License"); you may
#    not use this file except in compliance with the License. You may obtain
#    a copy of the License at
#
#         http://www.apache.org/licenses/LICENSE-2.0
#
#    Unless required by applicable law or agreed to in writing, software
#    distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
#    WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
#    License for the specific language governing permissions and limitations
#    under the License.

"""
  Twistd daemon for the nova compute nodes.
  Receives messages via AMQP, manages pool of worker threads
  for async tasks.
"""

import logging
import os
import sys

# NOTE(termie): kludge so that we can run this from the bin directory in the
#               checkout without having to screw with paths
NOVA_PATH = os.path.join(os.path.dirname(os.path.dirname(__file__)), 'nova')
if os.path.exists(NOVA_PATH):
    sys.path.insert(0, os.path.dirname(NOVA_PATH))


from carrot import connection
from carrot import messaging
from twisted.internet import task
from twisted.application import service

from nova import flags
from nova import rpc
from nova import twistd
from nova.compute import node


FLAGS = flags.FLAGS
# NOTE(termie): This file will necessarily be re-imported under different
#               context when the twistd.serve() call is made below so any
#               flags we define here will have to be conditionally defined,
#               flags defined by imported modules are safe.
if 'node_report_state_interval' not in FLAGS:
    flags.DEFINE_integer('node_report_state_interval', 10,
                         'seconds between nodes reporting state to cloud',
                         lower_bound=1)
logging.getLogger().setLevel(logging.DEBUG)

def main():
    logging.warn('Starting compute node')
    n = node.Node()
    d = n.adopt_instances()
    d.addCallback(lambda x: logging.info('Adopted %d instances', x))

    conn = rpc.Connection.instance()
    consumer_all = rpc.AdapterConsumer(
            connection=conn,
            topic='%s' % FLAGS.compute_topic,
            proxy=n)

    consumer_node = rpc.AdapterConsumer(
            connection=conn,
            topic='%s.%s' % (FLAGS.compute_topic, FLAGS.node_name),
            proxy=n)

    bin_name = os.path.basename(__file__)
    pulse = task.LoopingCall(n.report_state, FLAGS.node_name, bin_name)
    pulse.start(interval=FLAGS.node_report_state_interval, now=False)

    injected = consumer_all.attach_to_twisted()
    injected = consumer_node.attach_to_twisted()

    # This is the parent service that twistd will be looking for when it
    # parses this file, return it so that we can get it into globals below
    application = service.Application(bin_name)
    n.setServiceParent(application)
    return application


# NOTE(termie): When this script is executed from the commandline what it will
#               actually do is tell the twistd application runner that it
#               should run this file as a twistd application (see below).
if __name__ == '__main__':
    twistd.serve(__file__)

# NOTE(termie): When this script is loaded by the twistd application runner
#               this code path will be executed and twistd will expect a
#               variable named 'application' to be available, it will then
#               handle starting it and stopping it.
if __name__ == '__builtin__':
    application = main()