| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  | /** | 
					
						
							|  |  |  |  * Copyright (c) Microsoft Corporation. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * 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. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-08-22 07:07:13 -07:00
										 |  |  | import { makeWaitForNextTask } from '../utils/utils'; | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  | export interface WritableStream { | 
					
						
							|  |  |  |   write(data: Buffer): void; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export interface ReadableStream { | 
					
						
							|  |  |  |   on(event: 'data', callback: (b: Buffer) => void): void; | 
					
						
							|  |  |  |   on(event: 'close', callback: () => void): void; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export interface ClosableStream { | 
					
						
							|  |  |  |   close(): void; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  | export class Transport { | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |   private _pipeWrite: WritableStream; | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |   private _data = Buffer.from([]); | 
					
						
							| 
									
										
										
										
											2020-08-22 07:07:13 -07:00
										 |  |  |   private _waitForNextTask = makeWaitForNextTask(); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |   private _closed = false; | 
					
						
							|  |  |  |   private _bytesLeft = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-13 08:31:20 -07:00
										 |  |  |   onmessage?: (message: string) => void; | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |   onclose?: () => void; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |   private _endian: 'be' | 'le'; | 
					
						
							|  |  |  |   private _closeableStream: ClosableStream | undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   constructor(pipeWrite: WritableStream, pipeRead: ReadableStream, closeable?: ClosableStream, endian: 'be' | 'le' = 'le') { | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |     this._pipeWrite = pipeWrite; | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |     this._endian = endian; | 
					
						
							|  |  |  |     this._closeableStream = closeable; | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |     pipeRead.on('data', buffer => this._dispatch(buffer)); | 
					
						
							| 
									
										
										
										
											2020-12-16 14:21:59 -08:00
										 |  |  |     pipeRead.on('close', () => { | 
					
						
							|  |  |  |       this._closed = true; | 
					
						
							|  |  |  |       if (this.onclose) | 
					
						
							|  |  |  |         this.onclose(); | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |     this.onmessage = undefined; | 
					
						
							|  |  |  |     this.onclose = undefined; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 18:04:08 -07:00
										 |  |  |   send(message: string) { | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |     if (this._closed) | 
					
						
							|  |  |  |       throw new Error('Pipe has been closed'); | 
					
						
							| 
									
										
										
										
											2020-07-03 18:04:08 -07:00
										 |  |  |     const data = Buffer.from(message, 'utf-8'); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |     const dataLength = Buffer.alloc(4); | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |     if (this._endian === 'be') | 
					
						
							|  |  |  |       dataLength.writeUInt32BE(data.length, 0); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |       dataLength.writeUInt32LE(data.length, 0); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |     this._pipeWrite.write(dataLength); | 
					
						
							|  |  |  |     this._pipeWrite.write(data); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   close() { | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |     // Let it throw.
 | 
					
						
							|  |  |  |     this._closeableStream!.close(); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   _dispatch(buffer: Buffer) { | 
					
						
							|  |  |  |     this._data = Buffer.concat([this._data, buffer]); | 
					
						
							|  |  |  |     while (true) { | 
					
						
							|  |  |  |       if (!this._bytesLeft && this._data.length < 4) { | 
					
						
							|  |  |  |         // Need more data.
 | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!this._bytesLeft) { | 
					
						
							| 
									
										
										
										
											2020-12-09 15:06:57 -08:00
										 |  |  |         this._bytesLeft = this._endian === 'be' ? this._data.readUInt32BE(0) : this._data.readUInt32LE(0); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |         this._data = this._data.slice(4); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!this._bytesLeft || this._data.length < this._bytesLeft) { | 
					
						
							|  |  |  |         // Need more data.
 | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       const message = this._data.slice(0, this._bytesLeft); | 
					
						
							|  |  |  |       this._data = this._data.slice(this._bytesLeft); | 
					
						
							|  |  |  |       this._bytesLeft = 0; | 
					
						
							|  |  |  |       this._waitForNextTask(() => { | 
					
						
							|  |  |  |         if (this.onmessage) | 
					
						
							| 
									
										
										
										
											2020-07-03 18:04:08 -07:00
										 |  |  |           this.onmessage(message.toString('utf-8')); | 
					
						
							| 
									
										
										
										
											2020-06-27 11:32:27 -07:00
										 |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } |