live
GenericMediaServer.hh
Go to the documentation of this file.
1/**********
2This library is free software; you can redistribute it and/or modify it under
3the terms of the GNU Lesser General Public License as published by the
4Free Software Foundation; either version 3 of the License, or (at your
5option) any later version. (See <http://www.gnu.org/copyleft/lesser.html>.)
6
7This library is distributed in the hope that it will be useful, but WITHOUT
8ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
9FOR A PARTICULAR PURPOSE. See the GNU Lesser General Public License for
10more details.
11
12You should have received a copy of the GNU Lesser General Public License
13along with this library; if not, write to the Free Software Foundation, Inc.,
1451 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
15**********/
16// "liveMedia"
17// Copyright (c) 1996-2025 Live Networks, Inc. All rights reserved.
18// A generic media server class, used to implement a RTSP server, and any other server that uses
19// "ServerMediaSession" objects to describe media to be served.
20// C++ header
21
22#ifndef _GENERIC_MEDIA_SERVER_HH
23#define _GENERIC_MEDIA_SERVER_HH
24
25#ifndef _MEDIA_HH
26#include "Media.hh"
27#endif
28#ifndef _SERVER_MEDIA_SESSION_HH
29#include "ServerMediaSession.hh"
30#endif
31
32#ifndef REQUEST_BUFFER_SIZE
33#define REQUEST_BUFFER_SIZE 20000 // for incoming requests
34#endif
35#ifndef RESPONSE_BUFFER_SIZE
36#define RESPONSE_BUFFER_SIZE 20000
37#endif
38
39// Typedef for a handler function that gets called when "lookupServerMediaSession()"
40// (defined below) completes:
41typedef void lookupServerMediaSessionCompletionFunc(void* clientData,
42 ServerMediaSession* sessionLookedUp);
43
45public:
46 virtual void addServerMediaSession(ServerMediaSession* serverMediaSession);
47
48 virtual void lookupServerMediaSession(char const* streamName,
50 void* completionClientData,
51 Boolean isFirstLookupInSession = True);
52 // Note: This is a virtual function, so can be reimplemented by subclasses.
53 void lookupServerMediaSession(char const* streamName,
54 void (GenericMediaServer::*memberFunc)(ServerMediaSession*));
55 // Special case of "lookupServerMediaSession()" where the 'completion function' is a
56 // member function of "GenericMediaServer" (and the 'completion client data' is "this".)
57
59 // Removes the "ServerMediaSession" object from our lookup table, so it will no longer be accessible by new clients.
60 // (However, any *existing* client sessions that use this "ServerMediaSession" object will continue streaming.
61 // The "ServerMediaSession" object will not get deleted until all of these client sessions have closed.)
62 // (To both delete the "ServerMediaSession" object *and* close all client sessions that use it,
63 // call "deleteServerMediaSession(serverMediaSession)" instead.)
64 virtual void removeServerMediaSession(char const* streamName);
65 // ditto
66
68 // Closes (from the server) all client sessions that are currently using this "ServerMediaSession" object.
69 // Note, however, that the "ServerMediaSession" object remains accessible by new clients.
70 virtual void closeAllClientSessionsForServerMediaSession(char const* streamName);
71 // ditto
72
74 // Equivalent to:
75 // "closeAllClientSessionsForServerMediaSession(serverMediaSession); removeServerMediaSession(serverMediaSession);"
76 virtual void deleteServerMediaSession(char const* streamName);
77 // Equivalent to:
78 // "closeAllClientSessionsForServerMediaSession(streamName); removeServerMediaSession(streamName);
79
80 unsigned numClientSessions() const { return fClientSessions->numEntries(); }
81
82protected:
83 GenericMediaServer(UsageEnvironment& env, int ourSocketIPv4, int ourSocketIPv6, Port ourPort,
84 unsigned reclamationSeconds);
85 // If "reclamationSeconds" > 0, then the "ClientSession" state for each client will get
86 // reclaimed if no activity from the client is detected in at least "reclamationSeconds".
87 // we're an abstract base class
89 void cleanup(); // MUST be called in the destructor of any subclass of us
90
91 static int setUpOurSocket(UsageEnvironment& env, Port& ourPort, int domain);
92
93 static void incomingConnectionHandlerIPv4(void*, int /*mask*/);
94 static void incomingConnectionHandlerIPv6(void*, int /*mask*/);
97 void incomingConnectionHandlerOnSocket(int serverSocket);
98
99 void setTLSFileNames(char const* certFileName, char const* privKeyFileName);
100
101public: // should be protected, but some old compilers complain otherwise
102 // The state of a TCP connection used by a client:
104 protected:
106 int clientSocket, struct sockaddr_storage const& clientAddr,
107 Boolean useTLS);
109
112
113 static void incomingRequestHandler(void*, int /*mask*/);
115 virtual void handleRequestBytes(int newBytesRead) = 0;
117
118 protected:
119 friend class GenericMediaServer;
120 friend class ClientSession;
121 friend class RTSPServer; // needed to make some broken Windows compilers work; remove this in the future when we end support for Windows
124 struct sockaddr_storage fClientAddr;
128
129 // Optional support for TLS:
131 ServerTLSState* fInputTLS; // by default, just points to "fTLS", but subclasses may change
133 };
134
135 // The state of an individual client session (using one or more sequential TCP connections) handled by a server:
137 protected:
138 ClientSession(GenericMediaServer& ourServer, u_int32_t sessionId);
139 virtual ~ClientSession();
140
143 static void noteClientLiveness(ClientSession* clientSession);
144 static void livenessTimeoutTask(ClientSession* clientSession);
145
146 protected:
147 friend class GenericMediaServer;
148 friend class ClientConnection;
150 u_int32_t fOurSessionId;
153 };
154
155protected:
156 virtual ClientConnection* createNewClientConnection(int clientSocket, struct sockaddr_storage const& clientAddr) = 0;
157 virtual ClientSession* createNewClientSession(u_int32_t sessionId) = 0;
158
160 // Generates a new (unused) random session id, and calls the "createNewClientSession()"
161 // virtual function with this session id as parameter.
162
163 // Lookup a "ClientSession" object by sessionId (integer, and string):
164 ClientSession* lookupClientSession(u_int32_t sessionId);
165 ClientSession* lookupClientSession(char const* sessionIdStr);
166
167 // An iterator over our "ServerMediaSession" objects:
169 public:
173 private:
175 };
176
177 // The basic, synchronous "ServerMediaSession" lookup operation; only for subclasses:
179
180protected:
181 friend class ClientConnection;
182 friend class ClientSession;
187 HashTable* fServerMediaSessions; // maps 'stream name' strings to "ServerMediaSession" objects
188 HashTable* fClientConnections; // the "ClientConnection" objects that we're using
189 HashTable* fClientSessions; // maps 'session id' strings to "ClientSession" objects
190
191private:
193
196};
197
198// A data structure used for optional user/password authentication:
199
201public:
204 // If "passwordsAreMD5" is True, then each password stored into, or removed from,
205 // the database is actually the value computed
206 // by md5(<username>:<realm>:<actual-password>)
208
209 virtual void addUserRecord(char const* username, char const* password);
210 virtual void removeUserRecord(char const* username);
211
212 virtual char const* lookupPassword(char const* username);
213 // returns NULL if the user name was not present
214
215 char const* realm() { return fRealm; }
217
218protected:
220 char* fRealm;
222};
223
224#endif
const Boolean False
Definition: Boolean.hh:28
const Boolean True
Definition: Boolean.hh:31
unsigned char Boolean
Definition: Boolean.hh:25
void lookupServerMediaSessionCompletionFunc(void *clientData, ServerMediaSession *sessionLookedUp)
#define RESPONSE_BUFFER_SIZE
#define REQUEST_BUFFER_SIZE
#define NULL
void * TaskToken
virtual void handleRequestBytes(int newBytesRead)=0
static void incomingRequestHandler(void *, int)
unsigned char fRequestBuffer[REQUEST_BUFFER_SIZE]
unsigned char fResponseBuffer[RESPONSE_BUFFER_SIZE]
ClientConnection(GenericMediaServer &ourServer, int clientSocket, struct sockaddr_storage const &clientAddr, Boolean useTLS)
ClientSession(GenericMediaServer &ourServer, u_int32_t sessionId)
static void livenessTimeoutTask(ClientSession *clientSession)
ServerMediaSession * fOurServerMediaSession
static void noteClientLiveness(ClientSession *clientSession)
ServerMediaSessionIterator(GenericMediaServer &server)
unsigned numClientSessions() const
void lookupServerMediaSession(char const *streamName, void(GenericMediaServer::*memberFunc)(ServerMediaSession *))
ClientSession * createNewClientSessionWithId()
void deleteServerMediaSession(ServerMediaSession *serverMediaSession)
char const * fTLSCertificateFileName
char const * fTLSPrivateKeyFileName
u_int32_t fPreviousClientSessionId
virtual void deleteServerMediaSession(char const *streamName)
virtual void closeAllClientSessionsForServerMediaSession(char const *streamName)
void incomingConnectionHandlerIPv6()
ServerMediaSession * getServerMediaSession(char const *streamName)
void removeServerMediaSession(ServerMediaSession *serverMediaSession)
ClientSession * lookupClientSession(u_int32_t sessionId)
virtual void addServerMediaSession(ServerMediaSession *serverMediaSession)
ClientSession * lookupClientSession(char const *sessionIdStr)
static void incomingConnectionHandlerIPv4(void *, int)
static void incomingConnectionHandlerIPv6(void *, int)
HashTable * fClientConnections
HashTable * fServerMediaSessions
virtual ClientConnection * createNewClientConnection(int clientSocket, struct sockaddr_storage const &clientAddr)=0
void setTLSFileNames(char const *certFileName, char const *privKeyFileName)
void incomingConnectionHandlerOnSocket(int serverSocket)
virtual void lookupServerMediaSession(char const *streamName, lookupServerMediaSessionCompletionFunc *completionFunc, void *completionClientData, Boolean isFirstLookupInSession=True)
GenericMediaServer(UsageEnvironment &env, int ourSocketIPv4, int ourSocketIPv6, Port ourPort, unsigned reclamationSeconds)
virtual ~GenericMediaServer()
virtual ClientSession * createNewClientSession(u_int32_t sessionId)=0
static int setUpOurSocket(UsageEnvironment &env, Port &ourPort, int domain)
void incomingConnectionHandlerIPv4()
virtual void removeServerMediaSession(char const *streamName)
void closeAllClientSessionsForServerMediaSession(ServerMediaSession *serverMediaSession)
virtual unsigned numEntries() const =0
Definition: Media.hh:50
UsageEnvironment & envir() const
Definition: Media.hh:59
virtual void removeUserRecord(char const *username)
virtual void addUserRecord(char const *username, char const *password)
virtual char const * lookupPassword(char const *username)
UserAuthenticationDatabase(char const *realm=NULL, Boolean passwordsAreMD5=False)
virtual ~UserAuthenticationDatabase()