2021-01-12 02:13:41 +09:00
// Copyright 2018-2021 the Deno authors. All rights reserved. MIT license.
2020-12-07 21:46:39 +11:00
2020-12-24 21:53:03 +11:00
use super ::analysis ::ResolvedDependency ;
2020-12-21 14:44:26 +01:00
use super ::language_server ::StateSnapshot ;
2020-12-07 21:46:39 +11:00
use super ::text ;
use super ::utils ;
use crate ::media_type ::MediaType ;
2020-12-21 14:44:26 +01:00
use crate ::tokio_util ::create_basic_runtime ;
2020-12-16 06:34:39 +11:00
use crate ::tsc ;
2020-12-07 21:46:39 +11:00
use crate ::tsc ::ResolveArgs ;
use crate ::tsc_config ::TsConfig ;
2020-12-21 14:44:26 +01:00
use deno_core ::error ::anyhow ;
2020-12-07 21:46:39 +11:00
use deno_core ::error ::custom_error ;
use deno_core ::error ::AnyError ;
2020-12-21 14:44:26 +01:00
use deno_core ::futures ::Future ;
2020-12-07 21:46:39 +11:00
use deno_core ::json_op_sync ;
use deno_core ::serde ::Deserialize ;
2020-12-08 11:36:13 +01:00
use deno_core ::serde ::Serialize ;
2020-12-07 21:46:39 +11:00
use deno_core ::serde_json ;
use deno_core ::serde_json ::json ;
use deno_core ::serde_json ::Value ;
2020-12-30 09:58:20 +09:00
use deno_core ::url ::Url ;
2020-12-07 21:46:39 +11:00
use deno_core ::JsRuntime ;
use deno_core ::ModuleSpecifier ;
use deno_core ::OpFn ;
use deno_core ::RuntimeOptions ;
2020-12-21 14:44:26 +01:00
use lspower ::lsp_types ;
2020-12-07 21:46:39 +11:00
use regex ::Captures ;
use regex ::Regex ;
use std ::borrow ::Cow ;
use std ::collections ::HashMap ;
2020-12-21 14:44:26 +01:00
use std ::thread ;
use tokio ::sync ::mpsc ;
use tokio ::sync ::oneshot ;
type Request = (
RequestMethod ,
StateSnapshot ,
oneshot ::Sender < Result < Value , AnyError > > ,
) ;
#[ derive(Clone, Debug) ]
pub struct TsServer ( mpsc ::UnboundedSender < Request > ) ;
impl TsServer {
pub fn new ( ) -> Self {
let ( tx , mut rx ) = mpsc ::unbounded_channel ::< Request > ( ) ;
let _join_handle = thread ::spawn ( move | | {
// TODO(@kitsonk) we need to allow displaying diagnostics here, but the
// current compiler snapshot sends them to stdio which would totally break
// the language server...
let mut ts_runtime = start ( false ) . expect ( " could not start tsc " ) ;
2021-01-12 08:50:02 +01:00
let runtime = create_basic_runtime ( ) ;
2020-12-21 14:44:26 +01:00
runtime . block_on ( async {
while let Some ( ( req , state_snapshot , tx ) ) = rx . recv ( ) . await {
let value = request ( & mut ts_runtime , state_snapshot , req ) ;
if tx . send ( value ) . is_err ( ) {
warn! ( " Unable to send result to client. " ) ;
}
}
} )
} ) ;
Self ( tx )
}
pub async fn request (
& self ,
snapshot : StateSnapshot ,
req : RequestMethod ,
) -> Result < Value , AnyError > {
let ( tx , rx ) = oneshot ::channel ::< Result < Value , AnyError > > ( ) ;
if self . 0. send ( ( req , snapshot , tx ) ) . is_err ( ) {
return Err ( anyhow! ( " failed to send request to tsc thread " ) ) ;
}
rx . await ?
}
}
2020-12-07 21:46:39 +11:00
2020-12-16 06:34:39 +11:00
/// Optionally returns an internal asset, first checking for any static assets
/// in Rust, then checking any previously retrieved static assets from the
/// isolate, and then finally, the tsc isolate itself.
2020-12-21 14:44:26 +01:00
pub async fn get_asset (
2020-12-16 06:34:39 +11:00
specifier : & ModuleSpecifier ,
2020-12-21 14:44:26 +01:00
ts_server : & TsServer ,
state_snapshot : & StateSnapshot ,
2020-12-16 06:34:39 +11:00
) -> Result < Option < String > , AnyError > {
let specifier_str = specifier . to_string ( ) . replace ( " asset:/// " , " " ) ;
if let Some ( asset_text ) = tsc ::get_asset ( & specifier_str ) {
Ok ( Some ( asset_text . to_string ( ) ) )
} else {
2020-12-21 14:44:26 +01:00
{
2020-12-31 14:33:44 +11:00
let assets = state_snapshot . assets . lock ( ) . unwrap ( ) ;
2020-12-21 14:44:26 +01:00
if let Some ( asset ) = assets . get ( specifier ) {
return Ok ( asset . clone ( ) ) ;
}
2020-12-07 21:46:39 +11:00
}
2020-12-21 14:44:26 +01:00
let asset : Option < String > = serde_json ::from_value (
ts_server
. request (
state_snapshot . clone ( ) ,
RequestMethod ::GetAsset ( specifier . clone ( ) ) ,
)
. await ? ,
) ? ;
2020-12-31 14:33:44 +11:00
let mut assets = state_snapshot . assets . lock ( ) . unwrap ( ) ;
2020-12-21 14:44:26 +01:00
assets . insert ( specifier . clone ( ) , asset . clone ( ) ) ;
Ok ( asset )
2020-12-07 21:46:39 +11:00
}
}
fn display_parts_to_string (
maybe_parts : Option < Vec < SymbolDisplayPart > > ,
) -> Option < String > {
maybe_parts . map ( | parts | {
parts
. into_iter ( )
. map ( | p | p . text )
. collect ::< Vec < String > > ( )
. join ( " " )
} )
}
fn get_tag_body_text ( tag : & JSDocTagInfo ) -> Option < String > {
tag . text . as_ref ( ) . map ( | text | match tag . name . as_str ( ) {
" example " = > {
let caption_regex =
Regex ::new ( r "<caption>(.*?)</caption>\s*\r?\n((?:\s|\S)*)" ) . unwrap ( ) ;
if caption_regex . is_match ( & text ) {
caption_regex
. replace ( text , | c : & Captures | {
format! ( " {} \n \n {} " , & c [ 1 ] , make_codeblock ( & c [ 2 ] ) )
} )
. to_string ( )
} else {
make_codeblock ( text )
}
}
" author " = > {
let email_match_regex = Regex ::new ( r "(.+)\s<([-.\w]+@[-.\w]+)>" ) . unwrap ( ) ;
email_match_regex
. replace ( text , | c : & Captures | format! ( " {} {} " , & c [ 1 ] , & c [ 2 ] ) )
. to_string ( )
}
" default " = > make_codeblock ( text ) ,
_ = > replace_links ( text ) ,
} )
}
fn get_tag_documentation ( tag : & JSDocTagInfo ) -> String {
match tag . name . as_str ( ) {
" augments " | " extends " | " param " | " template " = > {
if let Some ( text ) = & tag . text {
let part_regex = Regex ::new ( r "^(\S+)\s*-?\s*" ) . unwrap ( ) ;
let body : Vec < & str > = part_regex . split ( & text ) . collect ( ) ;
if body . len ( ) = = 3 {
let param = body [ 1 ] ;
let doc = body [ 2 ] ;
let label = format! ( " *@ {} * ` {} ` " , tag . name , param ) ;
if doc . is_empty ( ) {
return label ;
}
if doc . contains ( '\n' ) {
return format! ( " {} \n {} " , label , replace_links ( doc ) ) ;
} else {
return format! ( " {} - {} " , label , replace_links ( doc ) ) ;
}
}
}
}
_ = > ( ) ,
}
let label = format! ( " *@ {} * " , tag . name ) ;
let maybe_text = get_tag_body_text ( tag ) ;
if let Some ( text ) = maybe_text {
if text . contains ( '\n' ) {
format! ( " {} \n {} " , label , text )
} else {
format! ( " {} - {} " , label , text )
}
} else {
label
}
}
fn make_codeblock ( text : & str ) -> String {
let codeblock_regex = Regex ::new ( r "^\s*[~`]{3}" ) . unwrap ( ) ;
if codeblock_regex . is_match ( text ) {
text . to_string ( )
} else {
format! ( " ``` \n {} \n ``` " , text )
}
}
/// Replace JSDoc like links (`{@link http://example.com}`) with markdown links
fn replace_links ( text : & str ) -> String {
let jsdoc_links_regex = Regex ::new ( r "(?i)\{@(link|linkplain|linkcode) (https?://[^ |}]+?)(?:[| ]([^{}\n]+?))?\}" ) . unwrap ( ) ;
jsdoc_links_regex
. replace_all ( text , | c : & Captures | match & c [ 1 ] {
" linkcode " = > format! (
" [`{}`]({}) " ,
if c . get ( 3 ) . is_none ( ) {
& c [ 2 ]
} else {
c [ 3 ] . trim ( )
} ,
& c [ 2 ]
) ,
_ = > format! (
" [{}]({}) " ,
if c . get ( 3 ) . is_none ( ) {
& c [ 2 ]
} else {
c [ 3 ] . trim ( )
} ,
& c [ 2 ]
) ,
} )
. to_string ( )
}
2020-12-08 11:36:13 +01:00
#[ derive(Debug, Clone, Deserialize) ]
2020-12-07 21:46:39 +11:00
pub enum ScriptElementKind {
#[ serde(rename = " " ) ]
Unknown ,
#[ serde(rename = " warning " ) ]
Warning ,
#[ serde(rename = " keyword " ) ]
Keyword ,
#[ serde(rename = " script " ) ]
ScriptElement ,
#[ serde(rename = " module " ) ]
ModuleElement ,
#[ serde(rename = " class " ) ]
ClassElement ,
#[ serde(rename = " local class " ) ]
LocalClassElement ,
#[ serde(rename = " interface " ) ]
InterfaceElement ,
#[ serde(rename = " type " ) ]
TypeElement ,
#[ serde(rename = " enum " ) ]
EnumElement ,
#[ serde(rename = " enum member " ) ]
EnumMemberElement ,
#[ serde(rename = " var " ) ]
VariableElement ,
#[ serde(rename = " local var " ) ]
LocalVariableElement ,
#[ serde(rename = " function " ) ]
FunctionElement ,
#[ serde(rename = " local function " ) ]
LocalFunctionElement ,
#[ serde(rename = " method " ) ]
MemberFunctionElement ,
#[ serde(rename = " getter " ) ]
MemberGetAccessorElement ,
#[ serde(rename = " setter " ) ]
MemberSetAccessorElement ,
#[ serde(rename = " property " ) ]
MemberVariableElement ,
#[ serde(rename = " constructor " ) ]
ConstructorImplementationElement ,
#[ serde(rename = " call " ) ]
CallSignatureElement ,
#[ serde(rename = " index " ) ]
IndexSignatureElement ,
#[ serde(rename = " construct " ) ]
ConstructSignatureElement ,
#[ serde(rename = " parameter " ) ]
ParameterElement ,
#[ serde(rename = " type parameter " ) ]
TypeParameterElement ,
#[ serde(rename = " primitive type " ) ]
PrimitiveType ,
#[ serde(rename = " label " ) ]
Label ,
#[ serde(rename = " alias " ) ]
Alias ,
#[ serde(rename = " const " ) ]
ConstElement ,
#[ serde(rename = " let " ) ]
LetElement ,
#[ serde(rename = " directory " ) ]
Directory ,
#[ serde(rename = " external module name " ) ]
ExternalModuleName ,
#[ serde(rename = " JSX attribute " ) ]
JsxAttribute ,
#[ serde(rename = " string " ) ]
String ,
}
2020-12-08 11:36:13 +01:00
impl From < ScriptElementKind > for lsp_types ::CompletionItemKind {
fn from ( kind : ScriptElementKind ) -> Self {
2020-12-21 14:44:26 +01:00
use lspower ::lsp_types ::CompletionItemKind ;
2020-12-08 11:36:13 +01:00
match kind {
ScriptElementKind ::PrimitiveType | ScriptElementKind ::Keyword = > {
CompletionItemKind ::Keyword
}
ScriptElementKind ::ConstElement = > CompletionItemKind ::Constant ,
ScriptElementKind ::LetElement
| ScriptElementKind ::VariableElement
| ScriptElementKind ::LocalVariableElement
| ScriptElementKind ::Alias = > CompletionItemKind ::Variable ,
ScriptElementKind ::MemberVariableElement
| ScriptElementKind ::MemberGetAccessorElement
| ScriptElementKind ::MemberSetAccessorElement = > {
CompletionItemKind ::Field
}
ScriptElementKind ::FunctionElement = > CompletionItemKind ::Function ,
ScriptElementKind ::MemberFunctionElement
| ScriptElementKind ::ConstructSignatureElement
| ScriptElementKind ::CallSignatureElement
| ScriptElementKind ::IndexSignatureElement = > CompletionItemKind ::Method ,
ScriptElementKind ::EnumElement = > CompletionItemKind ::Enum ,
ScriptElementKind ::ModuleElement
| ScriptElementKind ::ExternalModuleName = > CompletionItemKind ::Module ,
ScriptElementKind ::ClassElement | ScriptElementKind ::TypeElement = > {
CompletionItemKind ::Class
}
ScriptElementKind ::InterfaceElement = > CompletionItemKind ::Interface ,
ScriptElementKind ::Warning | ScriptElementKind ::ScriptElement = > {
CompletionItemKind ::File
}
ScriptElementKind ::Directory = > CompletionItemKind ::Folder ,
ScriptElementKind ::String = > CompletionItemKind ::Constant ,
_ = > CompletionItemKind ::Property ,
}
}
}
#[ derive(Debug, Clone, Deserialize) ]
2020-12-07 21:46:39 +11:00
#[ serde(rename_all = " camelCase " ) ]
pub struct TextSpan {
start : u32 ,
length : u32 ,
}
impl TextSpan {
pub fn to_range ( & self , line_index : & [ u32 ] ) -> lsp_types ::Range {
lsp_types ::Range {
start : text ::to_position ( line_index , self . start ) ,
end : text ::to_position ( line_index , self . start + self . length ) ,
}
}
}
#[ derive(Debug, Deserialize, Clone) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct SymbolDisplayPart {
text : String ,
kind : String ,
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct JSDocTagInfo {
name : String ,
text : Option < String > ,
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct QuickInfo {
kind : ScriptElementKind ,
kind_modifiers : String ,
text_span : TextSpan ,
display_parts : Option < Vec < SymbolDisplayPart > > ,
documentation : Option < Vec < SymbolDisplayPart > > ,
tags : Option < Vec < JSDocTagInfo > > ,
}
impl QuickInfo {
pub fn to_hover ( & self , line_index : & [ u32 ] ) -> lsp_types ::Hover {
let mut contents = Vec ::< lsp_types ::MarkedString > ::new ( ) ;
if let Some ( display_string ) =
display_parts_to_string ( self . display_parts . clone ( ) )
{
contents . push ( lsp_types ::MarkedString ::from_language_code (
" typescript " . to_string ( ) ,
display_string ,
) ) ;
}
if let Some ( documentation ) =
display_parts_to_string ( self . documentation . clone ( ) )
{
contents . push ( lsp_types ::MarkedString ::from_markdown ( documentation ) ) ;
}
if let Some ( tags ) = & self . tags {
let tags_preview = tags
. iter ( )
. map ( get_tag_documentation )
. collect ::< Vec < String > > ( )
. join ( " \n \n " ) ;
if ! tags_preview . is_empty ( ) {
contents . push ( lsp_types ::MarkedString ::from_markdown ( format! (
" \n \n {} " ,
tags_preview
) ) ) ;
}
}
lsp_types ::Hover {
contents : lsp_types ::HoverContents ::Array ( contents ) ,
range : Some ( self . text_span . to_range ( line_index ) ) ,
}
}
}
2020-12-30 09:58:20 +09:00
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct RenameLocation {
// inherit from DocumentSpan
text_span : TextSpan ,
file_name : String ,
original_text_span : Option < TextSpan > ,
original_file_name : Option < String > ,
context_span : Option < TextSpan > ,
original_context_span : Option < TextSpan > ,
// RenameLocation props
prefix_text : Option < String > ,
suffix_text : Option < String > ,
}
pub struct RenameLocations {
pub locations : Vec < RenameLocation > ,
}
impl RenameLocations {
pub async fn into_workspace_edit < F , Fut > (
self ,
snapshot : StateSnapshot ,
index_provider : F ,
new_name : & str ,
) -> Result < lsp_types ::WorkspaceEdit , AnyError >
where
F : Fn ( ModuleSpecifier ) -> Fut ,
Fut : Future < Output = Result < Vec < u32 > , AnyError > > ,
{
let mut text_document_edit_map : HashMap < Url , lsp_types ::TextDocumentEdit > =
HashMap ::new ( ) ;
for location in self . locations . iter ( ) {
let uri = utils ::normalize_file_name ( & location . file_name ) ? ;
let specifier = ModuleSpecifier ::resolve_url ( & location . file_name ) ? ;
// ensure TextDocumentEdit for `location.file_name`.
if text_document_edit_map . get ( & uri ) . is_none ( ) {
text_document_edit_map . insert (
uri . clone ( ) ,
lsp_types ::TextDocumentEdit {
text_document : lsp_types ::OptionalVersionedTextDocumentIdentifier {
uri : uri . clone ( ) ,
version : snapshot
. doc_data
. get ( & specifier )
. map_or_else ( | | None , | data | data . version ) ,
} ,
edits : Vec ::<
lsp_types ::OneOf <
lsp_types ::TextEdit ,
lsp_types ::AnnotatedTextEdit ,
> ,
> ::new ( ) ,
} ,
) ;
}
// push TextEdit for ensured `TextDocumentEdit.edits`.
let document_edit = text_document_edit_map . get_mut ( & uri ) . unwrap ( ) ;
document_edit
. edits
. push ( lsp_types ::OneOf ::Left ( lsp_types ::TextEdit {
range : location
. text_span
. to_range ( & index_provider ( specifier . clone ( ) ) . await ? ) ,
new_text : new_name . to_string ( ) ,
} ) ) ;
}
Ok ( lsp_types ::WorkspaceEdit {
2021-01-12 08:50:02 +01:00
change_annotations : None ,
2020-12-30 09:58:20 +09:00
changes : None ,
document_changes : Some ( lsp_types ::DocumentChanges ::Edits (
text_document_edit_map . values ( ) . cloned ( ) . collect ( ) ,
) ) ,
} )
}
}
2020-12-07 21:46:39 +11:00
#[ derive(Debug, Deserialize) ]
pub enum HighlightSpanKind {
#[ serde(rename = " none " ) ]
None ,
#[ serde(rename = " definition " ) ]
Definition ,
#[ serde(rename = " reference " ) ]
Reference ,
#[ serde(rename = " writtenReference " ) ]
WrittenReference ,
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct HighlightSpan {
file_name : Option < String > ,
is_in_string : Option < bool > ,
text_span : TextSpan ,
context_span : Option < TextSpan > ,
kind : HighlightSpanKind ,
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct DefinitionInfo {
kind : ScriptElementKind ,
name : String ,
container_kind : Option < ScriptElementKind > ,
container_name : Option < String > ,
text_span : TextSpan ,
pub file_name : String ,
original_text_span : Option < TextSpan > ,
original_file_name : Option < String > ,
context_span : Option < TextSpan > ,
original_context_span : Option < TextSpan > ,
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct DefinitionInfoAndBoundSpan {
pub definitions : Option < Vec < DefinitionInfo > > ,
text_span : TextSpan ,
}
impl DefinitionInfoAndBoundSpan {
2020-12-21 14:44:26 +01:00
pub async fn to_definition < F , Fut > (
2020-12-07 21:46:39 +11:00
& self ,
line_index : & [ u32 ] ,
2020-12-21 14:44:26 +01:00
index_provider : F ,
2020-12-07 21:46:39 +11:00
) -> Option < lsp_types ::GotoDefinitionResponse >
where
2020-12-21 14:44:26 +01:00
F : Fn ( ModuleSpecifier ) -> Fut ,
Fut : Future < Output = Result < Vec < u32 > , AnyError > > ,
2020-12-07 21:46:39 +11:00
{
if let Some ( definitions ) = & self . definitions {
2020-12-21 14:44:26 +01:00
let mut location_links = Vec ::< lsp_types ::LocationLink > ::new ( ) ;
for di in definitions {
let target_specifier =
ModuleSpecifier ::resolve_url ( & di . file_name ) . unwrap ( ) ;
if let Ok ( target_line_index ) = index_provider ( target_specifier ) . await {
2020-12-07 21:46:39 +11:00
let target_uri = utils ::normalize_file_name ( & di . file_name ) . unwrap ( ) ;
let ( target_range , target_selection_range ) =
if let Some ( context_span ) = & di . context_span {
(
context_span . to_range ( & target_line_index ) ,
di . text_span . to_range ( & target_line_index ) ,
)
} else {
(
di . text_span . to_range ( & target_line_index ) ,
di . text_span . to_range ( & target_line_index ) ,
)
} ;
2020-12-21 14:44:26 +01:00
location_links . push ( lsp_types ::LocationLink {
2020-12-07 21:46:39 +11:00
origin_selection_range : Some ( self . text_span . to_range ( line_index ) ) ,
target_uri ,
target_range ,
target_selection_range ,
2020-12-21 14:44:26 +01:00
} ) ;
}
}
2020-12-07 21:46:39 +11:00
Some ( lsp_types ::GotoDefinitionResponse ::Link ( location_links ) )
} else {
None
}
}
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct DocumentHighlights {
file_name : String ,
highlight_spans : Vec < HighlightSpan > ,
}
impl DocumentHighlights {
pub fn to_highlight (
& self ,
line_index : & [ u32 ] ,
) -> Vec < lsp_types ::DocumentHighlight > {
self
. highlight_spans
. iter ( )
. map ( | hs | lsp_types ::DocumentHighlight {
range : hs . text_span . to_range ( line_index ) ,
kind : match hs . kind {
HighlightSpanKind ::WrittenReference = > {
Some ( lsp_types ::DocumentHighlightKind ::Write )
}
_ = > Some ( lsp_types ::DocumentHighlightKind ::Read ) ,
} ,
} )
. collect ( )
}
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct ReferenceEntry {
is_write_access : bool ,
pub is_definition : bool ,
is_in_string : Option < bool > ,
text_span : TextSpan ,
pub file_name : String ,
original_text_span : Option < TextSpan > ,
original_file_name : Option < String > ,
context_span : Option < TextSpan > ,
original_context_span : Option < TextSpan > ,
}
impl ReferenceEntry {
pub fn to_location ( & self , line_index : & [ u32 ] ) -> lsp_types ::Location {
let uri = utils ::normalize_file_name ( & self . file_name ) . unwrap ( ) ;
lsp_types ::Location {
uri ,
range : self . text_span . to_range ( line_index ) ,
}
}
}
2020-12-08 11:36:13 +01:00
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct CompletionInfo {
entries : Vec < CompletionEntry > ,
is_member_completion : bool ,
}
impl CompletionInfo {
pub fn into_completion_response (
self ,
line_index : & [ u32 ] ,
) -> lsp_types ::CompletionResponse {
let items = self
. entries
. into_iter ( )
. map ( | entry | entry . into_completion_item ( line_index ) )
. collect ( ) ;
lsp_types ::CompletionResponse ::Array ( items )
}
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct CompletionEntry {
kind : ScriptElementKind ,
kind_modifiers : Option < String > ,
name : String ,
sort_text : String ,
insert_text : Option < String > ,
replacement_span : Option < TextSpan > ,
has_action : Option < bool > ,
source : Option < String > ,
is_recommended : Option < bool > ,
}
impl CompletionEntry {
pub fn into_completion_item (
self ,
line_index : & [ u32 ] ,
) -> lsp_types ::CompletionItem {
let mut item = lsp_types ::CompletionItem {
label : self . name ,
kind : Some ( self . kind . into ( ) ) ,
sort_text : Some ( self . sort_text . clone ( ) ) ,
// TODO(lucacasonato): missing commit_characters
.. Default ::default ( )
} ;
if let Some ( true ) = self . is_recommended {
// Make sure isRecommended property always comes first
// https://github.com/Microsoft/vscode/issues/40325
item . preselect = Some ( true ) ;
} else if self . source . is_some ( ) {
// De-prioritze auto-imports
// https://github.com/Microsoft/vscode/issues/40311
item . sort_text = Some ( " \u{ffff} " . to_string ( ) + & self . sort_text )
}
match item . kind {
Some ( lsp_types ::CompletionItemKind ::Function )
| Some ( lsp_types ::CompletionItemKind ::Method ) = > {
item . insert_text_format = Some ( lsp_types ::InsertTextFormat ::Snippet ) ;
}
_ = > { }
}
let mut insert_text = self . insert_text ;
let replacement_range : Option < lsp_types ::Range > =
self . replacement_span . map ( | span | span . to_range ( line_index ) ) ;
// TODO(lucacasonato): port other special cases from https://github.com/theia-ide/typescript-language-server/blob/fdf28313833cd6216d00eb4e04dc7f00f4c04f09/server/src/completion.ts#L49-L55
if let Some ( kind_modifiers ) = self . kind_modifiers {
if kind_modifiers . contains ( " \\ optional \\ " ) {
if insert_text . is_none ( ) {
insert_text = Some ( item . label . clone ( ) ) ;
}
if item . filter_text . is_none ( ) {
item . filter_text = Some ( item . label . clone ( ) ) ;
}
item . label + = " ? " ;
}
}
if let Some ( insert_text ) = insert_text {
if let Some ( replacement_range ) = replacement_range {
item . text_edit = Some ( lsp_types ::CompletionTextEdit ::Edit (
lsp_types ::TextEdit ::new ( replacement_range , insert_text ) ,
) ) ;
} else {
item . insert_text = Some ( insert_text ) ;
}
}
item
}
}
2020-12-07 21:46:39 +11:00
#[ derive(Debug, Clone, Deserialize) ]
struct Response {
id : usize ,
data : Value ,
}
struct State < ' a > {
2020-12-16 06:34:39 +11:00
asset : Option < String > ,
2020-12-07 21:46:39 +11:00
last_id : usize ,
response : Option < Response > ,
2020-12-21 14:44:26 +01:00
state_snapshot : StateSnapshot ,
2020-12-07 21:46:39 +11:00
snapshots : HashMap < ( Cow < ' a , str > , Cow < ' a , str > ) , String > ,
}
impl < ' a > State < ' a > {
2020-12-21 14:44:26 +01:00
fn new ( state_snapshot : StateSnapshot ) -> Self {
2020-12-07 21:46:39 +11:00
Self {
2020-12-16 06:34:39 +11:00
asset : None ,
2020-12-07 21:46:39 +11:00
last_id : 1 ,
response : None ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-07 21:46:39 +11:00
snapshots : Default ::default ( ) ,
}
}
}
/// If a snapshot is missing from the state cache, add it.
fn cache_snapshot (
state : & mut State ,
specifier : String ,
version : String ,
) -> Result < ( ) , AnyError > {
if ! state
. snapshots
. contains_key ( & ( specifier . clone ( ) . into ( ) , version . clone ( ) . into ( ) ) )
{
let s = ModuleSpecifier ::resolve_url ( & specifier ) ? ;
2020-12-21 14:44:26 +01:00
let content = {
2020-12-31 14:33:44 +11:00
let file_cache = state . state_snapshot . file_cache . lock ( ) . unwrap ( ) ;
2020-12-21 14:44:26 +01:00
let file_id = file_cache . lookup ( & s ) . unwrap ( ) ;
file_cache . get_contents ( file_id ) ?
} ;
2020-12-07 21:46:39 +11:00
state
. snapshots
. insert ( ( specifier . into ( ) , version . into ( ) ) , content ) ;
}
Ok ( ( ) )
}
fn op < F > ( op_fn : F ) -> Box < OpFn >
where
F : Fn ( & mut State , Value ) -> Result < Value , AnyError > + 'static ,
{
json_op_sync ( move | s , args , _bufs | {
let state = s . borrow_mut ::< State > ( ) ;
op_fn ( state , args )
} )
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
struct SourceSnapshotArgs {
specifier : String ,
version : String ,
}
/// The language service is dropping a reference to a source file snapshot, and
/// we can drop our version of that document.
fn dispose ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : SourceSnapshotArgs = serde_json ::from_value ( args ) ? ;
state
. snapshots
. remove ( & ( v . specifier . into ( ) , v . version . into ( ) ) ) ;
Ok ( json! ( true ) )
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
struct GetChangeRangeArgs {
specifier : String ,
old_length : u32 ,
old_version : String ,
version : String ,
}
/// The language service wants to compare an old snapshot with a new snapshot to
/// determine what source hash changed.
fn get_change_range ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : GetChangeRangeArgs = serde_json ::from_value ( args . clone ( ) ) ? ;
cache_snapshot ( state , v . specifier . clone ( ) , v . version . clone ( ) ) ? ;
if let Some ( current ) = state
. snapshots
. get ( & ( v . specifier . clone ( ) . into ( ) , v . version . into ( ) ) )
{
if let Some ( prev ) = state
. snapshots
. get ( & ( v . specifier . clone ( ) . into ( ) , v . old_version . clone ( ) . into ( ) ) )
{
Ok ( text ::get_range_change ( prev , current ) )
} else {
// when a local file is opened up in the editor, the compiler might
// already have a snapshot of it in memory, and will request it, but we
// now are working off in memory versions of the document, and so need
// to tell tsc to reset the whole document
Ok ( json! ( {
" span " : {
" start " : 0 ,
" length " : v . old_length ,
} ,
" newLength " : current . chars ( ) . count ( ) ,
} ) )
}
} else {
Err ( custom_error (
" MissingSnapshot " ,
format! (
" The current snapshot version is missing. \n Args: \" {} \" " ,
args
) ,
) )
}
}
fn get_length ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : SourceSnapshotArgs = serde_json ::from_value ( args ) ? ;
let specifier = ModuleSpecifier ::resolve_url ( & v . specifier ) ? ;
2020-12-21 14:44:26 +01:00
if state . state_snapshot . doc_data . contains_key ( & specifier ) {
2020-12-07 21:46:39 +11:00
cache_snapshot ( state , v . specifier . clone ( ) , v . version . clone ( ) ) ? ;
let content = state
. snapshots
. get ( & ( v . specifier . into ( ) , v . version . into ( ) ) )
. unwrap ( ) ;
Ok ( json! ( content . chars ( ) . count ( ) ) )
} else {
2020-12-31 14:33:44 +11:00
let mut sources = state . state_snapshot . sources . lock ( ) . unwrap ( ) ;
2020-12-07 21:46:39 +11:00
Ok ( json! ( sources . get_length ( & specifier ) . unwrap ( ) ) )
}
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
struct GetTextArgs {
specifier : String ,
version : String ,
start : usize ,
end : usize ,
}
fn get_text ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : GetTextArgs = serde_json ::from_value ( args ) ? ;
let specifier = ModuleSpecifier ::resolve_url ( & v . specifier ) ? ;
2020-12-21 14:44:26 +01:00
let content = if state . state_snapshot . doc_data . contains_key ( & specifier ) {
2020-12-07 21:46:39 +11:00
cache_snapshot ( state , v . specifier . clone ( ) , v . version . clone ( ) ) ? ;
state
. snapshots
. get ( & ( v . specifier . into ( ) , v . version . into ( ) ) )
. unwrap ( )
. clone ( )
} else {
2020-12-31 14:33:44 +11:00
let mut sources = state . state_snapshot . sources . lock ( ) . unwrap ( ) ;
2020-12-07 21:46:39 +11:00
sources . get_text ( & specifier ) . unwrap ( )
} ;
Ok ( json! ( text ::slice ( & content , v . start .. v . end ) ) )
}
fn resolve ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : ResolveArgs = serde_json ::from_value ( args ) ? ;
let mut resolved = Vec ::< Option < ( String , String ) > > ::new ( ) ;
let referrer = ModuleSpecifier ::resolve_url ( & v . base ) ? ;
2020-12-31 14:33:44 +11:00
let mut sources = if let Ok ( sources ) = state . state_snapshot . sources . lock ( ) {
2020-12-07 21:46:39 +11:00
sources
} else {
return Err ( custom_error ( " Deadlock " , " deadlock locking sources " ) ) ;
} ;
2020-12-21 14:44:26 +01:00
if let Some ( doc_data ) = state . state_snapshot . doc_data . get ( & referrer ) {
2020-12-07 21:46:39 +11:00
if let Some ( dependencies ) = & doc_data . dependencies {
for specifier in & v . specifiers {
if specifier . starts_with ( " asset:/// " ) {
resolved . push ( Some ( (
specifier . clone ( ) ,
MediaType ::from ( specifier ) . as_ts_extension ( ) ,
) ) )
} else if let Some ( dependency ) = dependencies . get ( specifier ) {
let resolved_import =
if let Some ( resolved_import ) = & dependency . maybe_type {
resolved_import . clone ( )
} else if let Some ( resolved_import ) = & dependency . maybe_code {
resolved_import . clone ( )
} else {
2020-12-24 21:53:03 +11:00
ResolvedDependency ::Err ( " missing dependency " . to_string ( ) )
2020-12-07 21:46:39 +11:00
} ;
2020-12-24 21:53:03 +11:00
if let ResolvedDependency ::Resolved ( resolved_specifier ) =
resolved_import
2020-12-07 21:46:39 +11:00
{
2020-12-10 06:50:47 +11:00
if state
2020-12-21 14:44:26 +01:00
. state_snapshot
2020-12-10 06:50:47 +11:00
. doc_data
. contains_key ( & resolved_specifier )
| | sources . contains ( & resolved_specifier )
2020-12-07 21:46:39 +11:00
{
2020-12-10 06:50:47 +11:00
let media_type = if let Some ( media_type ) =
sources . get_media_type ( & resolved_specifier )
{
media_type
} else {
MediaType ::from ( & resolved_specifier )
} ;
resolved . push ( Some ( (
resolved_specifier . to_string ( ) ,
media_type . as_ts_extension ( ) ,
) ) ) ;
2020-12-07 21:46:39 +11:00
} else {
2020-12-10 06:50:47 +11:00
resolved . push ( None ) ;
}
2020-12-07 21:46:39 +11:00
} else {
resolved . push ( None ) ;
}
}
}
}
} else if sources . contains ( & referrer ) {
for specifier in & v . specifiers {
if let Some ( ( resolved_specifier , media_type ) ) =
sources . resolve_import ( specifier , & referrer )
{
resolved . push ( Some ( (
resolved_specifier . to_string ( ) ,
media_type . as_ts_extension ( ) ,
) ) ) ;
} else {
resolved . push ( None ) ;
}
}
} else {
return Err ( custom_error (
" NotFound " ,
" the referring specifier is unexpectedly missing " ,
) ) ;
}
Ok ( json! ( resolved ) )
}
fn respond ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
state . response = Some ( serde_json ::from_value ( args ) ? ) ;
Ok ( json! ( true ) )
}
fn script_names ( state : & mut State , _args : Value ) -> Result < Value , AnyError > {
let script_names : Vec < & ModuleSpecifier > =
2020-12-21 14:44:26 +01:00
state . state_snapshot . doc_data . keys ( ) . collect ( ) ;
2020-12-07 21:46:39 +11:00
Ok ( json! ( script_names ) )
}
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
struct ScriptVersionArgs {
specifier : String ,
}
fn script_version ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : ScriptVersionArgs = serde_json ::from_value ( args ) ? ;
let specifier = ModuleSpecifier ::resolve_url ( & v . specifier ) ? ;
2020-12-21 14:44:26 +01:00
let maybe_doc_data = state . state_snapshot . doc_data . get ( & specifier ) ;
2020-12-07 21:46:39 +11:00
if let Some ( doc_data ) = maybe_doc_data {
if let Some ( version ) = doc_data . version {
return Ok ( json! ( version . to_string ( ) ) ) ;
}
} else {
2020-12-31 14:33:44 +11:00
let mut sources = state . state_snapshot . sources . lock ( ) . unwrap ( ) ;
2020-12-07 21:46:39 +11:00
if let Some ( version ) = sources . get_script_version ( & specifier ) {
return Ok ( json! ( version ) ) ;
}
}
Ok ( json! ( None ::< String > ) )
}
2020-12-16 06:34:39 +11:00
#[ derive(Debug, Deserialize) ]
#[ serde(rename_all = " camelCase " ) ]
struct SetAssetArgs {
text : Option < String > ,
}
fn set_asset ( state : & mut State , args : Value ) -> Result < Value , AnyError > {
let v : SetAssetArgs = serde_json ::from_value ( args ) ? ;
state . asset = v . text ;
Ok ( json! ( true ) )
}
2020-12-07 21:46:39 +11:00
/// Create and setup a JsRuntime based on a snapshot. It is expected that the
/// supplied snapshot is an isolate that contains the TypeScript language
/// server.
pub fn start ( debug : bool ) -> Result < JsRuntime , AnyError > {
let mut runtime = JsRuntime ::new ( RuntimeOptions {
2021-01-06 02:38:23 +01:00
startup_snapshot : Some ( tsc ::compiler_snapshot ( ) ) ,
2020-12-07 21:46:39 +11:00
.. Default ::default ( )
} ) ;
{
let op_state = runtime . op_state ( ) ;
let mut op_state = op_state . borrow_mut ( ) ;
2020-12-21 14:44:26 +01:00
op_state . put ( State ::new ( StateSnapshot ::default ( ) ) ) ;
2020-12-07 21:46:39 +11:00
}
runtime . register_op ( " op_dispose " , op ( dispose ) ) ;
runtime . register_op ( " op_get_change_range " , op ( get_change_range ) ) ;
runtime . register_op ( " op_get_length " , op ( get_length ) ) ;
runtime . register_op ( " op_get_text " , op ( get_text ) ) ;
runtime . register_op ( " op_resolve " , op ( resolve ) ) ;
runtime . register_op ( " op_respond " , op ( respond ) ) ;
runtime . register_op ( " op_script_names " , op ( script_names ) ) ;
runtime . register_op ( " op_script_version " , op ( script_version ) ) ;
2020-12-16 06:34:39 +11:00
runtime . register_op ( " op_set_asset " , op ( set_asset ) ) ;
2020-12-07 21:46:39 +11:00
let init_config = json! ( { " debug " : debug } ) ;
let init_src = format! ( " globalThis.serverInit( {} ); " , init_config ) ;
runtime . execute ( " [native code] " , & init_src ) ? ;
Ok ( runtime )
}
2020-12-08 11:36:13 +01:00
#[ derive(Debug, Serialize) ]
#[ serde(rename_all = " kebab-case " ) ]
#[ allow(dead_code) ]
pub enum QuotePreference {
Auto ,
Double ,
Single ,
}
#[ derive(Debug, Serialize) ]
#[ serde(rename_all = " kebab-case " ) ]
#[ allow(dead_code) ]
pub enum ImportModuleSpecifierPreference {
Auto ,
Relative ,
NonRelative ,
}
#[ derive(Debug, Serialize) ]
#[ serde(rename_all = " kebab-case " ) ]
#[ allow(dead_code) ]
pub enum ImportModuleSpecifierEnding {
Auto ,
Minimal ,
Index ,
Js ,
}
#[ derive(Debug, Serialize) ]
#[ serde(rename_all = " kebab-case " ) ]
#[ allow(dead_code) ]
pub enum IncludePackageJsonAutoImports {
Auto ,
On ,
Off ,
}
#[ derive(Debug, Default, Serialize) ]
#[ serde(rename_all = " camelCase " ) ]
pub struct UserPreferences {
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub disable_suggestions : Option < bool > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub quote_preference : Option < QuotePreference > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub include_completions_for_module_exports : Option < bool > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub include_automatic_optional_chain_completions : Option < bool > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub include_completions_with_insert_text : Option < bool > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub import_module_specifier_preference :
Option < ImportModuleSpecifierPreference > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub import_module_specifier_ending : Option < ImportModuleSpecifierEnding > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub allow_text_changes_in_new_files : Option < bool > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub provide_prefix_and_suffix_text_for_rename : Option < bool > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub include_package_json_auto_imports : Option < IncludePackageJsonAutoImports > ,
#[ serde(skip_serializing_if = " Option::is_none " ) ]
pub provide_refactor_not_applicable_reason : Option < bool > ,
}
2020-12-07 21:46:39 +11:00
/// Methods that are supported by the Language Service in the compiler isolate.
pub enum RequestMethod {
/// Configure the compilation settings for the server.
Configure ( TsConfig ) ,
2020-12-16 06:34:39 +11:00
/// Retrieve the text of an assets that exists in memory in the isolate.
GetAsset ( ModuleSpecifier ) ,
2020-12-30 12:46:58 +11:00
/// Return diagnostics for given file.
GetDiagnostics ( ModuleSpecifier ) ,
2020-12-07 21:46:39 +11:00
/// Return quick info at position (hover information).
GetQuickInfo ( ( ModuleSpecifier , u32 ) ) ,
/// Return document highlights at position.
GetDocumentHighlights ( ( ModuleSpecifier , u32 , Vec < ModuleSpecifier > ) ) ,
/// Get document references for a specific position.
GetReferences ( ( ModuleSpecifier , u32 ) ) ,
/// Get declaration information for a specific position.
GetDefinition ( ( ModuleSpecifier , u32 ) ) ,
2020-12-08 11:36:13 +01:00
/// Get completion information at a given position (IntelliSense).
GetCompletions ( ( ModuleSpecifier , u32 , UserPreferences ) ) ,
2020-12-30 09:58:20 +09:00
/// Get rename locations at a given position.
FindRenameLocations ( ( ModuleSpecifier , u32 , bool , bool , bool ) ) ,
2020-12-07 21:46:39 +11:00
}
impl RequestMethod {
pub fn to_value ( & self , id : usize ) -> Value {
match self {
RequestMethod ::Configure ( config ) = > json! ( {
" id " : id ,
" method " : " configure " ,
" compilerOptions " : config ,
} ) ,
2020-12-16 06:34:39 +11:00
RequestMethod ::GetAsset ( specifier ) = > json! ( {
" id " : id ,
" method " : " getAsset " ,
" specifier " : specifier ,
} ) ,
2020-12-30 12:46:58 +11:00
RequestMethod ::GetDiagnostics ( specifier ) = > json! ( {
2020-12-07 21:46:39 +11:00
" id " : id ,
2020-12-30 12:46:58 +11:00
" method " : " getDiagnostics " ,
2020-12-07 21:46:39 +11:00
" specifier " : specifier ,
} ) ,
RequestMethod ::GetQuickInfo ( ( specifier , position ) ) = > json! ( {
" id " : id ,
" method " : " getQuickInfo " ,
" specifier " : specifier ,
" position " : position ,
} ) ,
RequestMethod ::GetDocumentHighlights ( (
specifier ,
position ,
files_to_search ,
) ) = > json! ( {
" id " : id ,
" method " : " getDocumentHighlights " ,
" specifier " : specifier ,
" position " : position ,
" filesToSearch " : files_to_search ,
} ) ,
RequestMethod ::GetReferences ( ( specifier , position ) ) = > json! ( {
" id " : id ,
" method " : " getReferences " ,
" specifier " : specifier ,
" position " : position ,
} ) ,
RequestMethod ::GetDefinition ( ( specifier , position ) ) = > json! ( {
" id " : id ,
" method " : " getDefinition " ,
" specifier " : specifier ,
" position " : position ,
} ) ,
2020-12-08 11:36:13 +01:00
RequestMethod ::GetCompletions ( ( specifier , position , preferences ) ) = > {
json! ( {
" id " : id ,
" method " : " getCompletions " ,
" specifier " : specifier ,
" position " : position ,
" preferences " : preferences ,
} )
}
2020-12-30 09:58:20 +09:00
RequestMethod ::FindRenameLocations ( (
specifier ,
position ,
find_in_strings ,
find_in_comments ,
provide_prefix_and_suffix_text_for_rename ,
) ) = > {
json! ( {
" id " : id ,
" method " : " findRenameLocations " ,
" specifier " : specifier ,
" position " : position ,
" findInStrings " : find_in_strings ,
" findInComments " : find_in_comments ,
" providePrefixAndSuffixTextForRename " : provide_prefix_and_suffix_text_for_rename
} )
}
2020-12-07 21:46:39 +11:00
}
}
}
/// Send a request into a runtime and return the JSON value of the response.
pub fn request (
runtime : & mut JsRuntime ,
2020-12-21 14:44:26 +01:00
state_snapshot : StateSnapshot ,
2020-12-07 21:46:39 +11:00
method : RequestMethod ,
) -> Result < Value , AnyError > {
let id = {
let op_state = runtime . op_state ( ) ;
let mut op_state = op_state . borrow_mut ( ) ;
let state = op_state . borrow_mut ::< State > ( ) ;
2020-12-21 14:44:26 +01:00
state . state_snapshot = state_snapshot ;
2020-12-07 21:46:39 +11:00
state . last_id + = 1 ;
state . last_id
} ;
let request_params = method . to_value ( id ) ;
let request_src = format! ( " globalThis.serverRequest( {} ); " , request_params ) ;
runtime . execute ( " [native_code] " , & request_src ) ? ;
let op_state = runtime . op_state ( ) ;
let mut op_state = op_state . borrow_mut ( ) ;
let state = op_state . borrow_mut ::< State > ( ) ;
if let Some ( response ) = state . response . clone ( ) {
state . response = None ;
Ok ( response . data )
} else {
Err ( custom_error (
" RequestError " ,
" The response was not received for the request. " ,
) )
}
}
#[ cfg(test) ]
mod tests {
use super ::super ::memory_cache ::MemoryCache ;
use super ::* ;
2020-12-21 14:44:26 +01:00
use crate ::lsp ::language_server ::DocumentData ;
2020-12-07 21:46:39 +11:00
use std ::collections ::HashMap ;
use std ::sync ::Arc ;
2020-12-31 14:33:44 +11:00
use std ::sync ::Mutex ;
2020-12-07 21:46:39 +11:00
2020-12-21 14:44:26 +01:00
fn mock_state_snapshot ( sources : Vec < ( & str , & str , i32 ) > ) -> StateSnapshot {
2020-12-07 21:46:39 +11:00
let mut doc_data = HashMap ::new ( ) ;
let mut file_cache = MemoryCache ::default ( ) ;
for ( specifier , content , version ) in sources {
let specifier = ModuleSpecifier ::resolve_url ( specifier )
. expect ( " failed to create specifier " ) ;
doc_data . insert (
specifier . clone ( ) ,
DocumentData ::new ( specifier . clone ( ) , version , content , None ) ,
) ;
file_cache . set_contents ( specifier , Some ( content . as_bytes ( ) . to_vec ( ) ) ) ;
}
2020-12-31 14:33:44 +11:00
let file_cache = Arc ::new ( Mutex ::new ( file_cache ) ) ;
2020-12-21 14:44:26 +01:00
StateSnapshot {
2020-12-16 06:34:39 +11:00
assets : Default ::default ( ) ,
2020-12-07 21:46:39 +11:00
doc_data ,
file_cache ,
sources : Default ::default ( ) ,
}
}
fn setup (
debug : bool ,
config : Value ,
sources : Vec < ( & str , & str , i32 ) > ,
2020-12-21 14:44:26 +01:00
) -> ( JsRuntime , StateSnapshot ) {
let state_snapshot = mock_state_snapshot ( sources . clone ( ) ) ;
2020-12-07 21:46:39 +11:00
let mut runtime = start ( debug ) . expect ( " could not start server " ) ;
let ts_config = TsConfig ::new ( config ) ;
assert_eq! (
request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot . clone ( ) ,
2020-12-07 21:46:39 +11:00
RequestMethod ::Configure ( ts_config )
)
. expect ( " failed request " ) ,
json! ( true )
) ;
2020-12-21 14:44:26 +01:00
( runtime , state_snapshot )
2020-12-07 21:46:39 +11:00
}
#[ test ]
fn test_replace_links ( ) {
let actual = replace_links ( r "test {@link http://deno.land/x/mod.ts} test" ) ;
assert_eq! (
actual ,
r "test [http://deno.land/x/mod.ts](http://deno.land/x/mod.ts) test"
) ;
let actual =
replace_links ( r "test {@link http://deno.land/x/mod.ts a link} test" ) ;
assert_eq! ( actual , r "test [a link](http://deno.land/x/mod.ts) test" ) ;
let actual =
replace_links ( r "test {@linkcode http://deno.land/x/mod.ts a link} test" ) ;
assert_eq! ( actual , r "test [`a link`](http://deno.land/x/mod.ts) test" ) ;
}
#[ test ]
fn test_project_configure ( ) {
setup (
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" noEmit " : true ,
} ) ,
vec! [ ] ,
) ;
}
#[ test ]
fn test_project_reconfigure ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-07 21:46:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" noEmit " : true ,
} ) ,
vec! [ ] ,
) ;
let ts_config = TsConfig ::new ( json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" noEmit " : true ,
" lib " : [ " deno.ns " , " deno.worker " ]
} ) ) ;
let result = request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-07 21:46:39 +11:00
RequestMethod ::Configure ( ts_config ) ,
) ;
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
assert_eq! ( response , json! ( true ) ) ;
}
#[ test ]
2020-12-30 12:46:58 +11:00
fn test_get_diagnostics ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-07 21:46:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" noEmit " : true ,
} ) ,
vec! [ ( " file:///a.ts " , r # "console.log("hello deno");"# , 1 ) ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " file:///a.ts " )
. expect ( " could not resolve url " ) ;
let result = request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-30 12:46:58 +11:00
RequestMethod ::GetDiagnostics ( specifier ) ,
2020-12-07 21:46:39 +11:00
) ;
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
assert_eq! (
response ,
json! ( [
{
" start " : {
" line " : 0 ,
" character " : 0 ,
} ,
" end " : {
" line " : 0 ,
" character " : 7
} ,
" fileName " : " file:///a.ts " ,
" messageText " : " Cannot find name 'console'. Do you need to change your target library? Try changing the `lib` compiler option to include 'dom'. " ,
" sourceLine " : " console.log( \" hello deno \" ); " ,
" category " : 1 ,
" code " : 2584
}
] )
) ;
}
#[ test ]
fn test_module_resolution ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-07 21:46:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" lib " : [ " deno.ns " , " deno.window " ] ,
" noEmit " : true ,
} ) ,
vec! [ (
" file:///a.ts " ,
r #"
import { B } from " https://deno.land/x/b/mod.ts " ;
const b = new B ( ) ;
console . log ( b ) ;
" #,
1 ,
) ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " file:///a.ts " )
. expect ( " could not resolve url " ) ;
let result = request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-30 12:46:58 +11:00
RequestMethod ::GetDiagnostics ( specifier ) ,
2020-12-07 21:46:39 +11:00
) ;
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
assert_eq! ( response , json! ( [ ] ) ) ;
}
#[ test ]
fn test_bad_module_specifiers ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-07 21:46:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" lib " : [ " deno.ns " , " deno.window " ] ,
" noEmit " : true ,
} ) ,
vec! [ (
" file:///a.ts " ,
r #"
import { A } from " . " ;
" #,
1 ,
) ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " file:///a.ts " )
. expect ( " could not resolve url " ) ;
let result = request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-30 12:46:58 +11:00
RequestMethod ::GetDiagnostics ( specifier ) ,
2020-12-07 21:46:39 +11:00
) ;
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
2020-12-30 12:46:58 +11:00
assert_eq! (
response ,
json! ( [ {
" start " : {
" line " : 1 ,
" character " : 8
} ,
" end " : {
" line " : 1 ,
" character " : 30
} ,
" fileName " : " file:///a.ts " ,
" messageText " : " \' A \' is declared but its value is never read. " ,
" sourceLine " : " import { A } from \" . \" ; " ,
" category " : 2 ,
" code " : 6133 ,
" reportsUnnecessary " : true ,
} ] )
) ;
2020-12-07 21:46:39 +11:00
}
#[ test ]
fn test_remote_modules ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-07 21:46:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" lib " : [ " deno.ns " , " deno.window " ] ,
" noEmit " : true ,
} ) ,
vec! [ (
" file:///a.ts " ,
r #"
import { B } from " https://deno.land/x/b/mod.ts " ;
const b = new B ( ) ;
console . log ( b ) ;
" #,
1 ,
) ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " file:///a.ts " )
. expect ( " could not resolve url " ) ;
let result = request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-30 12:46:58 +11:00
RequestMethod ::GetDiagnostics ( specifier ) ,
2020-12-07 21:46:39 +11:00
) ;
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
assert_eq! ( response , json! ( [ ] ) ) ;
}
#[ test ]
fn test_partial_modules ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-07 21:46:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" lib " : [ " deno.ns " , " deno.window " ] ,
" noEmit " : true ,
} ) ,
vec! [ (
" file:///a.ts " ,
r #"
import {
Application ,
Context ,
Router ,
Status ,
} from " https://deno.land/x/oak@v6.3.2/mod.ts " ;
import * as test from
" #,
1 ,
) ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " file:///a.ts " )
. expect ( " could not resolve url " ) ;
let result = request (
& mut runtime ,
2020-12-21 14:44:26 +01:00
state_snapshot ,
2020-12-30 12:46:58 +11:00
RequestMethod ::GetDiagnostics ( specifier ) ,
2020-12-07 21:46:39 +11:00
) ;
2020-12-16 06:34:39 +11:00
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
assert_eq! (
response ,
json! ( [ {
2020-12-30 12:46:58 +11:00
" start " : {
" line " : 1 ,
" character " : 8
} ,
" end " : {
" line " : 6 ,
" character " : 55 ,
} ,
" fileName " : " file:///a.ts " ,
" messageText " : " All imports in import declaration are unused. " ,
" sourceLine " : " import { " ,
" category " : 2 ,
" code " : 6192 ,
" reportsUnnecessary " : true
} , {
2020-12-16 06:34:39 +11:00
" start " : {
" line " : 8 ,
" character " : 29
} ,
" end " : {
" line " : 8 ,
" character " : 29
} ,
" fileName " : " file:///a.ts " ,
" messageText " : " Expression expected. " ,
" sourceLine " : " import * as test from " ,
" category " : 1 ,
" code " : 1109
} ] )
) ;
}
2020-12-30 12:46:58 +11:00
#[ test ]
fn test_no_debug_failure ( ) {
let ( mut runtime , state_snapshot ) = setup (
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" lib " : [ " deno.ns " , " deno.window " ] ,
" noEmit " : true ,
} ) ,
vec! [ ( " file:///a.ts " , r # "const url = new URL("b.js", import."# , 1 ) ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " file:///a.ts " )
. expect ( " could not resolve url " ) ;
let result = request (
& mut runtime ,
state_snapshot ,
RequestMethod ::GetDiagnostics ( specifier ) ,
) ;
assert! ( result . is_ok ( ) ) ;
let response = result . unwrap ( ) ;
assert_eq! ( response , json! ( [ ] ) ) ;
}
2020-12-16 06:34:39 +11:00
#[ test ]
fn test_request_asset ( ) {
2020-12-21 14:44:26 +01:00
let ( mut runtime , state_snapshot ) = setup (
2020-12-16 06:34:39 +11:00
false ,
json! ( {
" target " : " esnext " ,
" module " : " esnext " ,
" lib " : [ " deno.ns " , " deno.window " ] ,
" noEmit " : true ,
} ) ,
vec! [ ] ,
) ;
let specifier = ModuleSpecifier ::resolve_url ( " asset:///lib.esnext.d.ts " )
. expect ( " could not resolve url " ) ;
2020-12-21 14:44:26 +01:00
let result = request (
& mut runtime ,
state_snapshot ,
RequestMethod ::GetAsset ( specifier ) ,
) ;
2020-12-16 06:34:39 +11:00
assert! ( result . is_ok ( ) ) ;
2020-12-21 14:44:26 +01:00
let response : Option < String > =
serde_json ::from_value ( result . unwrap ( ) ) . unwrap ( ) ;
2020-12-16 06:34:39 +11:00
assert! ( response . is_some ( ) ) ;
2020-12-07 21:46:39 +11:00
}
}