Fix various RTMP push issues:
- Fix RTMPS pushes, fix RTMP push "secret" Adobe handshake - Switch RTMP URL parsing to split at last slash rather than first slash - "Fix" compatibility with GO-RTMP-based RTMP implementations. - Fix compatibility with Odysee, which does a strict RTMP version check and doesn't like our version 1.2.3.4 I guess..?
This commit is contained in:
		
							parent
							
								
									7b18307981
								
							
						
					
					
						commit
						2f6cc46f88
					
				
					 1 changed files with 35 additions and 8 deletions
				
			
		|  | @ -109,7 +109,7 @@ namespace Mist{ | ||||||
|     RTMPStream::lastrecv.clear(); |     RTMPStream::lastrecv.clear(); | ||||||
| 
 | 
 | ||||||
|     std::string app = Encodings::URL::encode(pushUrl.path, "/:=@[]"); |     std::string app = Encodings::URL::encode(pushUrl.path, "/:=@[]"); | ||||||
|     size_t slash = app.find('/'); |     size_t slash = app.rfind('/'); | ||||||
|     if (slash != std::string::npos){app = app.substr(0, slash);} |     if (slash != std::string::npos){app = app.substr(0, slash);} | ||||||
| 
 | 
 | ||||||
|     if (pushUrl.protocol == "rtmp"){myConn.open(pushUrl.host, pushUrl.getPort(), false);} |     if (pushUrl.protocol == "rtmp"){myConn.open(pushUrl.host, pushUrl.getPort(), false);} | ||||||
|  | @ -128,25 +128,45 @@ namespace Mist{ | ||||||
|       return; |       return; | ||||||
|     } |     } | ||||||
|     *((uint32_t *)temp) = 0;                         // time zero
 |     *((uint32_t *)temp) = 0;                         // time zero
 | ||||||
|     *(((uint32_t *)(temp + 4))) = htonl(0x01020304); // version 1 2 3 4
 |     *(((uint32_t *)(temp + 4))) = 0; // version 0
 | ||||||
|     for (int i = 8; i < 3072; ++i){ |     for (int i = 8; i < 3072; ++i){ | ||||||
|       temp[i] = FILLER_DATA[i % sizeof(FILLER_DATA)]; |       temp[i] = FILLER_DATA[i % sizeof(FILLER_DATA)]; | ||||||
|     }//"random" data
 |     }//"random" data
 | ||||||
|     myConn.SendNow(temp, 3072); | 
 | ||||||
|  |     // Calculate the SHA265 digest over the data, insert it in the "secret" location
 | ||||||
|  |     size_t digest_pos = (temp[9] + temp[10] + temp[11] + temp[12]) % 728 + 12; | ||||||
|  |     // Copy data except for the 32 bytes where the digest is stored
 | ||||||
|  |     Util::ResizeablePointer digest_data; | ||||||
|  |     digest_data.append(temp+1, digest_pos); | ||||||
|  |     digest_data.append(temp+1+digest_pos+32, 1504-digest_pos); | ||||||
|  |     Secure::hmac_sha256bin(digest_data, digest_data.size(), "Genuine Adobe Flash Player 001", 30, temp + 1 + digest_pos); | ||||||
|  | 
 | ||||||
|  |     myConn.SendNow(temp, 1536); | ||||||
|  |     while (!myConn.Received().available(1537) && myConn.connected() && config->is_active){ | ||||||
|  |       myConn.spool(); | ||||||
|  |     } | ||||||
|  |     if (!myConn || !config->is_active){ | ||||||
|  |       WARN_MSG("Lost connection while waiting for S0/S1 packets!"); | ||||||
|  |       return; | ||||||
|  |     } | ||||||
|  |     // Send back copy of S1 (S0 is the first byte, skip it)
 | ||||||
|  |     Util::ResizeablePointer s0s1; | ||||||
|  |     myConn.Received().remove(s0s1, 1537); | ||||||
|  |     myConn.SendNow(s0s1 + 1, 1536); | ||||||
|     free(temp); |     free(temp); | ||||||
|     setBlocking(true); |     setBlocking(true); | ||||||
|     while (!myConn.Received().available(3073) && myConn.connected() && config->is_active){ |     while (!myConn.Received().available(1536) && myConn.connected() && config->is_active){ | ||||||
|       myConn.spool(); |       myConn.spool(); | ||||||
|     } |     } | ||||||
|     if (!myConn || !config->is_active){return;} |     if (!myConn || !config->is_active){return;} | ||||||
|     myConn.Received().remove(3073); |     myConn.Received().remove(1536); | ||||||
|     RTMPStream::rec_cnt += 3073; |     RTMPStream::rec_cnt += 3073; | ||||||
|     RTMPStream::snd_cnt += 3073; |     RTMPStream::snd_cnt += 3073; | ||||||
|     setBlocking(false); |     setBlocking(false); | ||||||
|     VERYHIGH_MSG("Push out handshake completed"); |     VERYHIGH_MSG("Push out handshake completed"); | ||||||
|     std::string pushHost = "rtmp://" + pushUrl.host + "/"; |     std::string pushHost = pushUrl.protocol + "://" + pushUrl.host + "/"; | ||||||
|     if (pushUrl.getPort() != 1935){ |     if (pushUrl.getPort() != 1935){ | ||||||
|       pushHost = "rtmp://" + pushUrl.host + ":" + JSON::Value(pushUrl.getPort()).asString() + "/"; |       pushHost = pushUrl.protocol + "://" + pushUrl.host + ":" + JSON::Value(pushUrl.getPort()).asString() + "/"; | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|     AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); |     AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); | ||||||
|  | @ -789,7 +809,10 @@ namespace Mist{ | ||||||
|     for (std::map<size_t, Comms::Users>::iterator it = userSelect.begin(); it != userSelect.end(); it++){ |     for (std::map<size_t, Comms::Users>::iterator it = userSelect.begin(); it != userSelect.end(); it++){ | ||||||
|       selectedTracks.insert(it->first); |       selectedTracks.insert(it->first); | ||||||
|     } |     } | ||||||
|     tag.DTSCMetaInit(meta, selectedTracks); |     //GO-RTMP can't handle the complexity of our metadata, but also doesn't need it... so let's not.
 | ||||||
|  |     if (UA != "GO-RTMP/0,0,0,0"){ | ||||||
|  |       tag.DTSCMetaInit(meta, selectedTracks); | ||||||
|  |     } | ||||||
|     if (tag.len){ |     if (tag.len){ | ||||||
|       tag.tagTime(currentTime() - rtmpOffset); |       tag.tagTime(currentTime() - rtmpOffset); | ||||||
|       myConn.SendNow(RTMPStream::SendMedia(tag)); |       myConn.SendNow(RTMPStream::SendMedia(tag)); | ||||||
|  | @ -1535,6 +1558,10 @@ namespace Mist{ | ||||||
|         (amfData.getContentP(0)->StrValue() == "onStatus")){ |         (amfData.getContentP(0)->StrValue() == "onStatus")){ | ||||||
|       if (isRecording() && amfData.getContentP(0)->StrValue() == "_result" && |       if (isRecording() && amfData.getContentP(0)->StrValue() == "_result" && | ||||||
|           amfData.getContentP(1)->NumValue() == 1){ |           amfData.getContentP(1)->NumValue() == 1){ | ||||||
|  |         if (amfData.getContentP(2)->GetType() == AMF::AMF0_OBJECT && amfData.getContentP(2)->getContentP("fmsVer")){ | ||||||
|  |           UA = amfData.getContentP(2)->getContentP("fmsVer")->StrValue(); | ||||||
|  |           INFO_MSG("Server version: %s", UA.c_str()); | ||||||
|  |         } | ||||||
|         { |         { | ||||||
|           AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); |           AMF::Object amfReply("container", AMF::AMF0_DDV_CONTAINER); | ||||||
|           amfReply.addContent(AMF::Object("", "releaseStream"));     // command
 |           amfReply.addContent(AMF::Object("", "releaseStream"));     // command
 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue
	
	 Thulinma
						Thulinma