| 
									
										
										
										
											2016-09-30 09:30:08 +00:00
										 |  |  | <?php | 
					
						
							| 
									
										
										
										
											2016-09-30 09:54:49 +00:00
										 |  |  | declare(strict_types = 1); | 
					
						
							| 
									
										
										
										
											2016-09-30 09:30:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | namespace LanguageServer; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-11-14 09:25:44 +00:00
										 |  |  | use Throwable; | 
					
						
							| 
									
										
										
										
											2016-10-10 13:06:02 +00:00
										 |  |  | use InvalidArgumentException; | 
					
						
							| 
									
										
										
										
											2016-11-18 14:22:24 +00:00
										 |  |  | use PhpParser\Node; | 
					
						
							| 
									
										
										
										
											2016-11-14 09:25:44 +00:00
										 |  |  | use Sabre\Event\{Loop, Promise}; | 
					
						
							| 
									
										
										
										
											2016-09-30 09:30:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Transforms an absolute file path into a URI as used by the language server protocol. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param string $filepath | 
					
						
							|  |  |  |  * @return string | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-10-24 17:35:37 +00:00
										 |  |  | function pathToUri(string $filepath): string | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2016-09-30 09:30:08 +00:00
										 |  |  |     $filepath = trim(str_replace('\\', '/', $filepath), '/'); | 
					
						
							| 
									
										
										
										
											2016-10-10 13:06:02 +00:00
										 |  |  |     $parts = explode('/', $filepath); | 
					
						
							|  |  |  |     // Don't %-encode the colon after a Windows drive letter
 | 
					
						
							|  |  |  |     $first = array_shift($parts); | 
					
						
							|  |  |  |     if (substr($first, -1) !== ':') { | 
					
						
							|  |  |  |         $first = urlencode($first); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-11-07 09:24:49 +00:00
										 |  |  |     $parts = array_map('rawurlencode', $parts); | 
					
						
							| 
									
										
										
										
											2016-10-10 13:06:02 +00:00
										 |  |  |     array_unshift($parts, $first); | 
					
						
							|  |  |  |     $filepath = implode('/', $parts); | 
					
						
							| 
									
										
										
										
											2016-09-30 09:30:08 +00:00
										 |  |  |     return 'file:///' . $filepath; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-10-10 13:06:02 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Transforms URI into file path | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param string $uri | 
					
						
							|  |  |  |  * @return string | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function uriToPath(string $uri) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     $fragments = parse_url($uri); | 
					
						
							|  |  |  |     if ($fragments === null || !isset($fragments['scheme']) || $fragments['scheme'] !== 'file') { | 
					
						
							|  |  |  |         throw new InvalidArgumentException("Not a valid file URI: $uri"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     $filepath = urldecode($fragments['path']); | 
					
						
							| 
									
										
										
										
											2016-10-11 22:53:21 +00:00
										 |  |  |     if (strpos($filepath, ':') !== false) { | 
					
						
							|  |  |  |         if ($filepath[0] === '/') { | 
					
						
							|  |  |  |             $filepath = substr($filepath, 1); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         $filepath = str_replace('/', '\\', $filepath); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return $filepath; | 
					
						
							| 
									
										
										
										
											2016-10-10 13:06:02 +00:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2016-11-14 09:25:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Throws an exception on the next tick. | 
					
						
							|  |  |  |  * Useful for letting a promise crash the process on rejection. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param Throwable $err | 
					
						
							|  |  |  |  * @return void | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function crash(Throwable $err) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Loop\nextTick(function () use ($err) { | 
					
						
							|  |  |  |         throw $err; | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns a promise that is resolved after x seconds. | 
					
						
							|  |  |  |  * Useful for giving back control to the event loop inside a coroutine. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param int $seconds | 
					
						
							|  |  |  |  * @return Promise <void> | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function timeout($seconds = 0): Promise | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     $promise = new Promise; | 
					
						
							|  |  |  |     Loop\setTimeout([$promise, 'fulfill'], $seconds); | 
					
						
							|  |  |  |     return $promise; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-11-18 14:22:24 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | /** | 
					
						
							|  |  |  |  * Returns the closest node of a specific type | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @param Node $node | 
					
						
							|  |  |  |  * @param string $type The node class name | 
					
						
							|  |  |  |  * @return Node|null $type | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | function getClosestNode(Node $node, string $type) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     $n = $node; | 
					
						
							|  |  |  |     while ($n = $n->getAttribute('parentNode')) { | 
					
						
							|  |  |  |         if ($n instanceof $type) { | 
					
						
							|  |  |  |             return $n; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } |