-
Notifications
You must be signed in to change notification settings - Fork 0
/
ServerWorker.py
160 lines (125 loc) · 4.12 KB
/
ServerWorker.py
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
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
from random import randint
import sys, traceback, threading, socket
from VideoStream import VideoStream
from RtpPacket import RtpPacket
class ServerWorker:
SETUP = 'SETUP'
PLAY = 'PLAY'
PAUSE = 'PAUSE'
TEARDOWN = 'TEARDOWN'
INIT = 0
READY = 1
PLAYING = 2
state = INIT
OK_200 = 0
FILE_NOT_FOUND_404 = 1
CON_ERR_500 = 2
clientInfo = {}
def __init__(self, clientInfo):
self.clientInfo = clientInfo
def run(self):
threading.Thread(target=self.recvRtspRequest).start()
def recvRtspRequest(self):
"""Receive RTSP request from the client."""
connSocket = self.clientInfo['rtspSocket'][0]
while True:
data = connSocket.recv(256)
if data:
print ("Data received:\n" + data.decode())
self.processRtspRequest(data)
def processRtspRequest(self, data):
"""Process RTSP request sent from the client."""
# Get the request type
request = data.decode().split('\n')
line1 = request[0].split(' ')
requestType = line1[0]
# Get the media file name
filename = line1[1]
# Get the RTSP sequence number
seq = request[1].split(' ')
# Process SETUP request
if requestType == self.SETUP:
if self.state == self.INIT:
# Update state
print ("processing SETUP\n")
try:
self.clientInfo['videoStream'] = VideoStream(filename)
self.state = self.READY
except IOError:
self.replyRtsp(self.FILE_NOT_FOUND_404, seq[1])
# Generate a randomized RTSP session ID
self.clientInfo['session'] = randint(100000, 999999)
# Send RTSP reply
self.replyRtsp(self.OK_200, seq[1])
# Get the RTP/UDP port from the last line
self.clientInfo['rtpPort'] = request[2].split(' ')[3]
# Process PLAY request
elif requestType == self.PLAY:
if self.state == self.READY:
print ("processing PLAY\n")
self.state = self.PLAYING
# Create a new socket for RTP/UDP
self.clientInfo["rtpSocket"] = socket.socket(socket.AF_INET, socket.SOCK_DGRAM)
self.replyRtsp(self.OK_200, seq[1])
# Create a new thread and start sending RTP packets
self.clientInfo['event'] = threading.Event()
self.clientInfo['worker']= threading.Thread(target=self.sendRtp)
self.clientInfo['worker'].start()
# Process PAUSE request
elif requestType == self.PAUSE:
if self.state == self.PLAYING:
print ("processing PAUSE\n")
self.state = self.READY
self.clientInfo['event'].set()
self.replyRtsp(self.OK_200, seq[1])
# Process TEARDOWN request
elif requestType == self.TEARDOWN:
print ("processing TEARDOWN\n")
self.clientInfo['event'].set()
self.replyRtsp(self.OK_200, seq[1])
# Close the RTP socket
self.clientInfo['rtpSocket'].close()
def sendRtp(self):
"""Send RTP packets over UDP."""
while True:
self.clientInfo['event'].wait(0.05)
# Stop sending if request is PAUSE or TEARDOWN
if self.clientInfo['event'].isSet():
break
data = self.clientInfo['videoStream'].nextFrame()
if data:
frameNumber = self.clientInfo['videoStream'].frameNbr()
try:
address = self.clientInfo['rtspSocket'][1][0]
port = int(self.clientInfo['rtpPort'])
self.clientInfo['rtpSocket'].sendto(self.makeRtp(data, frameNumber),(address,port))
except:
print ("Connection Error")
#print '-'*60
#traceback.print_exc(file=sys.stdout)
#print '-'*60
def makeRtp(self, payload, frameNbr):
"""RTP-packetize the video data."""
version = 2
padding = 0
extension = 0
cc = 0
marker = 0
pt = 26 # MJPEG type
seqnum = frameNbr
ssrc = 0
rtpPacket = RtpPacket()
rtpPacket.encode(version, padding, extension, cc, seqnum, marker, pt, ssrc, payload)
return rtpPacket.getPacket()
def replyRtsp(self, code, seq):
"""Send RTSP reply to the client."""
if code == self.OK_200:
#print "200 OK"
reply = 'RTSP/1.0 200 OK\nCSeq: ' + seq + '\nSession: ' + str(self.clientInfo['session'])
connSocket = self.clientInfo['rtspSocket'][0]
connSocket.send(reply.encode())
# Error messages
elif code == self.FILE_NOT_FOUND_404:
print ("404 NOT FOUND")
elif code == self.CON_ERR_500:
print ("500 CONNECTION ERROR")