implemented the majority of server-side python. still need to add settings server...
[OpenSRF.git] / src / python / osrf / stack.py
1 # -----------------------------------------------------------------------
2 # Copyright (C) 2007  Georgia Public Library Service
3 # Bill Erickson <billserickson@gmail.com>
4
5 # This program is free software; you can redistribute it and/or
6 # modify it under the terms of the GNU General Public License
7 # as published by the Free Software Foundation; either version 2
8 # of the License, or (at your option) any later version.
9
10 # This program is distributed in the hope that it will be useful,
11 # but WITHOUT ANY WARRANTY; without even the implied warranty of
12 # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13 # GNU General Public License for more details.
14 # -----------------------------------------------------------------------
15
16 import time
17 import osrf.json, osrf.log, osrf.ex, osrf.ses, osrf.const, osrf.app
18
19 def push(net_msg):
20
21     ses = osrf.ses.Session.find_or_create(net_msg.thread)
22     ses.set_remote_id(net_msg.sender)
23     if not ses.service:
24         ses.service = osrf.app.Application.name
25
26     omessages = osrf.json.to_object(net_msg.body)
27
28     osrf.log.log_internal("stack.push(): received %d messages" % len(omessages))
29
30     # Pass each bundled opensrf message to the message handler
31     start = time.time()
32     for msg in omessages:
33         handle_message(ses, msg)
34     duration = time.time() - start
35
36     if isinstance(ses, osrf.ses.ServerSession):
37         osrf.log.log_info("Message processing duration %f" % duration)
38
39 def handle_message(session, message):
40
41     osrf.log.log_internal("handle_message(): processing message of "
42         "type %s" % message.type())
43
44     if isinstance(session, osrf.ses.ClientSession):
45         handle_client(session, message)
46     else:
47         handle_server(session, message)
48
49
50 def handle_client(session, message):
51
52     if message.type() == osrf.const.OSRF_MESSAGE_TYPE_RESULT:
53         session.push_response_queue(message)
54         return
55
56     if message.type() == osrf.const.OSRF_MESSAGE_TYPE_STATUS:
57
58         status_code = int(message.payload().statusCode())
59         status_text = message.payload().status()
60         osrf.log.log_internal("handle_message(): processing STATUS, "
61             "status_code =  %d" % status_code)
62
63         if status_code == osrf.const.OSRF_STATUS_COMPLETE:
64             # The server has informed us that this request is complete
65             req = session.find_request(message.threadTrace())
66             if req: 
67                 osrf.log.log_internal("marking request as complete: %d" % req.rid)
68                 req.set_complete()
69             return
70
71         if status_code == osrf.const.OSRF_STATUS_OK:
72             # We have connected successfully
73             osrf.log.log_debug("Successfully connected to " + session.service)
74             session.state = OSRF_APP_SESSION_CONNECTED
75             return
76
77         if status_code == osrf.const.OSRF_STATUS_CONTINUE:
78             # server is telling us to reset our wait timeout and keep waiting for a response
79             session.reset_request_timeout(message.threadTrace())
80             return
81
82         if status_code == osrf.const.OSRF_STATUS_TIMEOUT:
83             osrf.log.log_debug("The server did not receive a request from us in time...")
84             session.state = OSRF_APP_SESSION_DISCONNECTED
85             return
86
87         if status_code == osrf.const.OSRF_STATUS_NOTFOUND:
88             osrf.log.log_error("Requested method was not found on the server: %s" % status_text)
89             session.state = OSRF_APP_SESSION_DISCONNECTED
90             raise osrf.ex.OSRFServiceException(status_text)
91
92         if status_code == osrf.const.OSRF_STATUS_INTERNALSERVERERROR:
93             raise osrf.ex.OSRFServiceException("Server error %d : %s" % (status_code, status_text))
94
95         raise osrf.ex.OSRFProtocolException("Unknown message status: %d" % status_code)
96
97
98 def handle_server(session, message):
99
100     if message.type() == osrf.const.OSRF_MESSAGE_TYPE_REQUEST:
101         osrf.log.log_debug("server received REQUEST from %s" % session.remote_id)
102         osrf.app.Application.handle_request(session, message)
103         return
104
105     if message.type() == osrf.const.OSRF_MESSAGE_TYPE_CONNECT:
106         osrf.log.log_debug("server received CONNECT from %s" % session.remote_id)
107         session.state == osrf.const.OSRF_APP_SESSION_CONNECTED 
108         session.send_connect_ok(message.threadTrace())
109         return
110
111     if message.type() == osrf.const.OSRF_MESSAGE_TYPE_DISCONNECT:
112         osrf.log.log_debug("server received DISCONNECT from %s" % session.remote_id)
113         session.state = osrf.const.OSRF_APP_SESSION_DISCONNECTED
114         return
115
116     if message.type() == osrf.const.OSRF_MESSAGE_TYPE_STATUS:
117         # Should never get here
118         osrf.log.log_warn("server received STATUS from %s" % session.remote_id)
119         return
120
121