mirror of
https://github.com/adelphes/android-dev-ext.git
synced 2025-12-22 17:39:19 +00:00
* initial working language server * first hacky version of source parsing and type checking * first iteration of method body parser * add support for prefix/postfix inc expressions * add basic support for parsing new expressions * different attempt to parse using collapsable text ranges * fix parsing of binary operstors following a bracket expression * updated validation to use new JavaTypes module instead of MTIs * add support for array-literal expressions * fix || and && not being tokenized as operators allow float literals starting with dot * add new method body parser to use direct linear parsing * add super as an object literal * fix interface constructors check constructor type modifiers * fix assignment operator types * Fix resolving of enclosed type identifiers * add default constructor for class types with no explicit constructors * add missing constructor validator * add constructor parameters to list of resolvable types * update SourceMethod to pass name in super constructor * add Any* classes to reduce cascading errors * update method call parameter checking use isTypeAssignable instead of getParameterCompatibleTypeSignatures * tidy up isTypeAssignable allow class equivilents for primitives * add more info when methods/ctrs cannot be matched * allow interfaces to be cast to class instances * use isTypeAssignable for checking branch test expressions * allow AnyValue to be a constant value * split shift operators from bitwise operators * add support for literal numbers to be assignable to multiple primtive types * clear diagnostics when document is closed * update check for cast expression * casting only applies to qualified term not a whole expression * allow all primitive-number-type casts * add support for synchronized statement * update primitive type compatibility * allow null to be cast to any non-primitve * use better regex for string literals * allow character literals to be assigned to number types * add support for array qualifiers after a variable name * make sure any long specifier is stripped from a bigint value * improve invalid array expression message add AnyType array element to prevent cascading errors * make default a modifer keyword for interface default method support * initial support for wildcard type arguments * fix parse issue with nested generic types * allow generic types to be assigned to inherited types with compatible type arguments * allow unicode characters, $ and _ in identifiers * map primitive types to their boxed versions for class member * support assert statement * allow unicode char literals * make type parser and body parser use same tokenizer * reuse parsed tokens instead of tokenizing each method body * re-add throws as a keyword * treat default and synchronized as modifiers * add SourceInitialiser support * refactor to prepare for merging with type parsing * add support for array qualifiers in type identifiers * pass scoped type instead of method to typeIdent * update ResolvableType to use same type resolving as method body parsing * add support for post-name array qualifiers in fields and parameters * post-name array qualifiers in method decls * add type variables to SourceMethod * initial attempt to support type variable arguments in methods * specialise methods with type variables * don't require default interface methods to be implemented * make variable arity parameters an array type * tidy array constructors and fix some warnings * update isCallCompatible to handle variable arity calls * improve assert statement support * parse labels and break/continue targets * refactor new term qualifiers * add support for generic inferred-type arguments * improve modifier checks for interface types * improve reporting of unresolved type errors * fix type checking of field and method declarations * add missng strictfp modifier * refactor in preparation for parsing local types * replace Locals with scopeable MethodDeclarations to allow labels and types to be stored * initial changes to support local type declarations * update to use new set of SourceX classes * refactor to allow expressions to have a type scope * replace regex parsing with linear parsing * generate source types before parsing * fix support for resolving type variables in method declarations * fix checking of array literal compatability * report errors from unit parsing * remove local modifier validation during parse add parameter modifier checking to validation * allow trailing comma for array literals * start separating validation from parsing * add support for parsing enum values * allow uppercase 0X in hex literals * include enclosing types in identifier search * add support for parsing parameterless lambdas * ignore unresolved types in extends/implements * implement specialisation of SourceType * allow super as a member qualifier * allow empty enums * don't report missing constructors if superclass has none * update typemap declarations to use CEIType instead of JavaType * fix resolving of class type variables * fix bad imports when resolving annotations * allow null scope in findIdentifier * add support for static member imports * import types from same package * remove this qualifier from isCastExpression * add hex exponent support * parse try-with-resources * fix resolving imported enclosed types * extract expression types into separate files * extract statement types into separate files * fix type warnings * extract literals into separate files * remove Value class, add NewExpression and separate out Any classes * rename source types module * remove some parse checks that should be in verify * support token extraction in expressions * implement resolveExpression * add type cast checking * check for valid type in class member expressions * allow assigns for assignable type arguments * improve reporting of unresolved identifiers * add new array validation * validate array literals * validate array indexes * improve validation of binary operators * rename ResolvedType to ResolvedValue * improve checking of number literals * support package name as a resolved value * implement method body and ststement validation * improve method call resolving * add support for this() and super() constructor calls * remove return type for source constructors * add checks for unary operators * ensure tokens are assigned for qualified expressions * check castability using type assignments * add implicit enum methods values() and valueOf() * add basic type checking of lambda expressions * fix return type check * fix assert statement checks * improve support for ternary operators in assignments and method invocations * perform more detailed search of implemented methods * initial test of context-dependant code completion * support package, type and static field import completion * support for member expressions * use exact type signatures for locating types for completion items * add support for field and method docs * add support for docs in source types * support member completion for array types improve comment formatting * ensure Object is always last in the list of inherited types * add owning method to statements create common keyword statement class * improve code completion list add method parameters order list items by scope * add source types to list hide this and super for non-methods * fix bad member resolution at end of block fix missing method and type docs * add support for editing multiple files * allow multiple source files to be used in parsing * load and parse files at startup * add support for displaying method signatures * add single trace function with timestamps * implement shceduleReparse to reduce parsing load while typing * remove parsed type list logging * wait for reparsing before returning method signatures * resolve new object contructors * improve extraction of parameter docs * update @types/vscode * cache decoded android library in globalStoragePath * load single android library cache from local folder * android-29 library cache * allow configurable app root setting * set configurable trace logging and update section names * description updates * handle null token passed to ParseProblem * refactoring * Rename language client extension to Android * ignore unnamed type declarations * handle java file change notifications * make sure we only try and parse java files * add option to allow language server to be shutdown * simplify handling of this and class member qualifiers * relocate java-mti package into project * get main node install to install langserver dependencies * remove debugging pause * rename body-parser3 to body-parser * clean up import resolving code * remove unused field from ResolvedImport * remove validation modules that used old parser types * remove old parser files * remove redundant types and functions used by old parser * move addproblem into TokenList * remove unused ResolvedType class * validate more statements * add support for parsing and validating anonymous types * hide some method modifiers which aren't useful to show * code comments and minor improvements * fix some type warnings * improve support for completion of enum values * add type name to parameter completion labels * ignore synthetic members in completion list * use a specialised map for handling case-insenstive file uris * add basic build script * reference java-mti package from GitHub * revert @types/vscode * update initial file loading to use URIs passed from the client changes to the appSourceRoot now require an extension restart * add support for loading filtered androidx libraries for code completion * update version of java-mti * add mixpanel package * add basic analytics * fix dependency versions * fix dependency versions * set empty cache file markers * add language server debug config * add file to build script * add unqualified type members when inside a method * apply statics filter to enum values * add basic debugger analytics * include current time in startup event * add terminate reason to debugger * update changelog and readme
284 lines
8.8 KiB
JavaScript
284 lines
8.8 KiB
JavaScript
const { CEIType } = require('java-mti');
|
|
const ParseProblem = require('./java/parsetypes/parse-problem');
|
|
const { parse } = require('./java/body-parser');
|
|
const { SourceUnit } = require('./java/source-types');
|
|
const { parseMethodBodies } = require('./java/validater');
|
|
const { time, timeEnd, trace } = require('./logging');
|
|
|
|
/**
|
|
* Marker to prevent early parsing of source files before we've completed our
|
|
* initial source file load (we cannot accurately parse individual files until we
|
|
* know what all the types are - hence the need to perform a first parse of all the source files).
|
|
*
|
|
* While we are waiting for the first parse to complete, individual files-to-parse are added
|
|
* to this set. Once the first scan and parse is done, these are reparsed and
|
|
* first_parse_waiting is set to `null`.
|
|
* @type {Set<string>}
|
|
*/
|
|
let first_parse_waiting = new Set();
|
|
|
|
/**
|
|
* Convert a line,character position to an absolute character offset
|
|
*
|
|
* @param {{line:number,character:number}} pos
|
|
* @param {string} content
|
|
*/
|
|
function indexAt(pos, content) {
|
|
let idx = 0;
|
|
for (let i = 0; i < pos.line; i++) {
|
|
idx = content.indexOf('\n', idx) + 1;
|
|
if (idx === 0) {
|
|
return content.length;
|
|
}
|
|
}
|
|
return Math.min(idx + pos.character, content.length);
|
|
}
|
|
|
|
/**
|
|
* Convert an absolute character offset to a line,character position
|
|
*
|
|
* @param {number} index
|
|
* @param {string} content
|
|
*/
|
|
function positionAt(index, content) {
|
|
let line = 0,
|
|
last_nl_idx = 0,
|
|
character = 0;
|
|
if (index <= 0) return { line, character };
|
|
for (let idx = 0; ;) {
|
|
idx = content.indexOf('\n', idx) + 1;
|
|
if (idx === 0 || idx > index) {
|
|
if (idx === 0) index = content.length;
|
|
character = index - last_nl_idx;
|
|
return { line, character };
|
|
}
|
|
last_nl_idx = idx;
|
|
line++;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* A specialised Map to allow for case-insensitive fileURIs on Windows.
|
|
*
|
|
* For cs-filesystems, this should work as a normal map.
|
|
* For ci-filesystems, if a file URI case changes, it should be picked up
|
|
* by the lowercase map
|
|
*/
|
|
class FileURIMap extends Map {
|
|
lowerMap = new Map();
|
|
|
|
/**
|
|
* @param {string} key
|
|
*/
|
|
get(key) {
|
|
return super.get(key) || this.lowerMap.get(key.toLowerCase());
|
|
}
|
|
|
|
/**
|
|
* @param {string} key
|
|
*/
|
|
has(key) {
|
|
return super.has(key) || this.lowerMap.has(key.toLowerCase());
|
|
}
|
|
|
|
/**
|
|
* @param {string} key
|
|
* @param {*} value
|
|
*/
|
|
set(key, value) {
|
|
super.set(key, value);
|
|
this.lowerMap.set(key.toLowerCase(), value);
|
|
return this;
|
|
}
|
|
|
|
/**
|
|
* @param {string} key
|
|
*/
|
|
delete(key) {
|
|
this.lowerMap.delete(key.toLowerCase());
|
|
return super.delete(key);
|
|
}
|
|
|
|
clear() {
|
|
super.clear();
|
|
this.lowerMap.clear();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Class for storing data about Java source files
|
|
*/
|
|
class JavaDocInfo {
|
|
/**
|
|
* @param {string} uri the file URI
|
|
* @param {string} content the full file content
|
|
* @param {number} version revision number for edited files (each edit increments the version)
|
|
*/
|
|
constructor(uri, content, version) {
|
|
this.uri = uri;
|
|
this.content = content;
|
|
this.version = version;
|
|
/**
|
|
* The result of the Java parse
|
|
* @type {ParsedInfo}
|
|
*/
|
|
this.parsed = null;
|
|
|
|
/**
|
|
* Promise linked to a timer which resolves a short time after the user stops typing
|
|
* - This is used to prevent constant reparsing while the user is typing in the document
|
|
* @type {Promise}
|
|
*/
|
|
this.reparseWaiter = Promise.resolve();
|
|
|
|
/** @type {{ resolve: () => void, timer: * }} */
|
|
this.waitInfo = null;
|
|
}
|
|
|
|
/**
|
|
* Schedule this document for reparsing.
|
|
*
|
|
* To prevent redundant parsing while typing, a small delay is required
|
|
* before the reparse happens.
|
|
* When a key is pressed, `scheduleReparse()` starts a timer. If more
|
|
* keys are typed before the timer expires, the timer is restarted.
|
|
* Once typing pauses, the timer expires and the content reparsed.
|
|
*
|
|
* A `reparseWaiter` promise is used to delay actions like completion items
|
|
* retrieval and method signature resolving until the reparse is complete.
|
|
*
|
|
* @param {Map<string,JavaDocInfo>} liveParsers
|
|
* @param {Map<string,CEIType>|Promise<Map<string,CEIType>>} androidLibrary
|
|
*/
|
|
scheduleReparse(liveParsers, androidLibrary) {
|
|
const createWaitTimer = () => {
|
|
return setTimeout(() => {
|
|
// reparse the content, resolve the reparseWaiter promise
|
|
// and reset the fields
|
|
reparse([this.uri], liveParsers, androidLibrary, { includeMethods: true });
|
|
this.waitInfo.resolve();
|
|
this.waitInfo = null;
|
|
}, 250);
|
|
}
|
|
if (this.waitInfo) {
|
|
// we already have a promise pending - just restart the timer
|
|
trace('restart timer');
|
|
clearTimeout(this.waitInfo.timer);
|
|
this.waitInfo.timer = createWaitTimer();
|
|
return;
|
|
}
|
|
// create a new pending promise and start the timer
|
|
trace('start timer');
|
|
this.waitInfo = {
|
|
resolve: null,
|
|
timer: createWaitTimer(),
|
|
}
|
|
this.reparseWaiter = new Promise(resolve => this.waitInfo.resolve = resolve);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Result from parsing a Java file
|
|
*/
|
|
class ParsedInfo {
|
|
/**
|
|
* @param {string} uri the file URI
|
|
* @param {string} content the full file content
|
|
* @param {number} version the version this parse applies to
|
|
* @param {Map<string,CEIType>} typemap the set of known types
|
|
* @param {SourceUnit} unit the parsed unit
|
|
* @param {ParseProblem[]} problems
|
|
*/
|
|
constructor(uri, content, version, typemap, unit, problems) {
|
|
this.uri = uri;
|
|
this.content = content;
|
|
this.version = version;
|
|
this.typemap = typemap;
|
|
this.unit = unit;
|
|
this.problems = problems;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @param {string[]} uris
|
|
* @param {Map<string, JavaDocInfo>} liveParsers
|
|
* @param {Map<string,CEIType>|Promise<Map<string,CEIType>>} androidLibrary
|
|
* @param {{includeMethods: boolean, first_parse?: boolean}} [opts]
|
|
*/
|
|
function reparse(uris, liveParsers, androidLibrary, opts) {
|
|
trace(`reparse`);
|
|
if (!Array.isArray(uris)) {
|
|
return;
|
|
}
|
|
if (first_parse_waiting) {
|
|
if (!opts || !opts.first_parse) {
|
|
// we are waiting for the first parse to complete - add this file to the list
|
|
uris.forEach(uri => first_parse_waiting.add(uri));
|
|
trace('waiting for first parse')
|
|
return;
|
|
}
|
|
}
|
|
|
|
if (androidLibrary instanceof Promise) {
|
|
// reparse after the library has finished loading
|
|
androidLibrary.then(lib => reparse(uris, liveParsers, lib, opts));
|
|
return;
|
|
}
|
|
|
|
const cached_units = [], parsers = [];
|
|
for (let docinfo of liveParsers.values()) {
|
|
if (uris.includes(docinfo.uri)) {
|
|
// make a copy of the content + version in case the source file is edited while we're parsing
|
|
parsers.push({uri: docinfo.uri, content: docinfo.content, version: docinfo.version});
|
|
} else if (docinfo.parsed) {
|
|
cached_units.push(docinfo.parsed.unit);
|
|
}
|
|
}
|
|
|
|
// Each parse uses a unique typemap, initialised from the android library
|
|
const typemap = new Map(androidLibrary);
|
|
|
|
// perform the parse
|
|
const units = parse(parsers, cached_units, typemap);
|
|
|
|
// create new ParsedInfo instances for each of the parsed units
|
|
units.forEach(unit => {
|
|
const parser = parsers.find(p => p.uri === unit.uri);
|
|
if (!parser) return;
|
|
const doc = liveParsers.get(unit.uri);
|
|
if (!doc) return;
|
|
doc.parsed = new ParsedInfo(doc.uri, parser.content, parser.version, typemap, unit, []);
|
|
});
|
|
|
|
let method_body_uris = [];
|
|
if (first_parse_waiting) {
|
|
// this is the first parse - parse the bodies of any waiting URIs and
|
|
// set first_parse_waiting to null
|
|
method_body_uris = [...first_parse_waiting];
|
|
first_parse_waiting = null;
|
|
}
|
|
|
|
if (opts && opts.includeMethods) {
|
|
method_body_uris = uris;
|
|
}
|
|
|
|
if (method_body_uris.length) {
|
|
time('parse-methods');
|
|
method_body_uris.forEach(uri => {
|
|
const doc = liveParsers.get(uri);
|
|
if (!doc || !doc.parsed) {
|
|
return;
|
|
}
|
|
parseMethodBodies(doc.parsed.unit, typemap);
|
|
})
|
|
timeEnd('parse-methods');
|
|
}
|
|
}
|
|
|
|
exports.indexAt = indexAt;
|
|
exports.positionAt = positionAt;
|
|
exports.FileURIMap = FileURIMap;
|
|
exports.JavaDocInfo = JavaDocInfo;
|
|
exports.ParsedInfo = ParsedInfo;
|
|
exports.reparse = reparse;
|