blob: 1212a591d6338996d4a1fb12f45f264043775772 [file] [log] [blame]
// Copyright (c) 2014, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
import 'dart:collection';
import 'dart:convert' hide JsonDecoder;
import 'package:analysis_server/protocol/protocol.dart';
import 'package:analysis_server/protocol/protocol_generated.dart';
import 'package:analyzer_plugin/protocol/protocol_common.dart';
import 'package:analyzer_plugin/src/protocol/protocol_internal.dart'
show JsonDecoder;
export 'package:analyzer_plugin/src/protocol/protocol_internal.dart'
show JsonDecoder;
final Map<String, RefactoringKind> REQUEST_ID_REFACTORING_KINDS =
new HashMap<String, RefactoringKind>();
/**
* Adds the given [sourceEdits] to the list in [sourceFileEdit].
*/
void addAllEditsForSource(
SourceFileEdit sourceFileEdit, Iterable<SourceEdit> edits) {
edits.forEach(sourceFileEdit.add);
}
/**
* Adds the given [sourceEdit] to the list in [sourceFileEdit].
*/
void addEditForSource(SourceFileEdit sourceFileEdit, SourceEdit sourceEdit) {
List<SourceEdit> edits = sourceFileEdit.edits;
int index = 0;
while (index < edits.length && edits[index].offset > sourceEdit.offset) {
index++;
}
edits.insert(index, sourceEdit);
}
/**
* Adds [edit] to the [FileEdit] for the given [file].
*/
void addEditToSourceChange(
SourceChange change, String file, int fileStamp, SourceEdit edit) {
SourceFileEdit fileEdit = change.getFileEdit(file);
if (fileEdit == null) {
fileEdit = new SourceFileEdit(file, fileStamp);
change.addFileEdit(fileEdit);
}
fileEdit.add(edit);
}
/**
* Get the result of applying the edit to the given [code]. Access via
* SourceEdit.apply().
*/
String applyEdit(String code, SourceEdit edit) {
if (edit.length < 0) {
throw new RangeError('length is negative');
}
return code.replaceRange(edit.offset, edit.end, edit.replacement);
}
/**
* Get the result of applying a set of [edits] to the given [code]. Edits
* are applied in the order they appear in [edits]. Access via
* SourceEdit.applySequence().
*/
String applySequenceOfEdits(String code, Iterable<SourceEdit> edits) {
edits.forEach((SourceEdit edit) {
code = edit.apply(code);
});
return code;
}
/**
* Returns the [FileEdit] for the given [file], maybe `null`.
*/
SourceFileEdit getChangeFileEdit(SourceChange change, String file) {
for (SourceFileEdit fileEdit in change.edits) {
if (fileEdit.file == file) {
return fileEdit;
}
}
return null;
}
/**
* Compare the lists [listA] and [listB], using [itemEqual] to compare
* list elements.
*/
bool listEqual(List listA, List listB, bool itemEqual(a, b)) {
if (listA == null) {
return listB == null;
}
if (listB == null) {
return false;
}
if (listA.length != listB.length) {
return false;
}
for (int i = 0; i < listA.length; i++) {
if (!itemEqual(listA[i], listB[i])) {
return false;
}
}
return true;
}
/**
* Compare the maps [mapA] and [mapB], using [valueEqual] to compare map
* values.
*/
bool mapEqual(Map mapA, Map mapB, bool valueEqual(a, b)) {
if (mapA == null) {
return mapB == null;
}
if (mapB == null) {
return false;
}
if (mapA.length != mapB.length) {
return false;
}
for (var key in mapA.keys) {
if (!mapB.containsKey(key)) {
return false;
}
if (!valueEqual(mapA[key], mapB[key])) {
return false;
}
}
return true;
}
/**
* Translate the input [map], applying [keyCallback] to all its keys, and
* [valueCallback] to all its values.
*/
Map<KR, VR> mapMap<KP, VP, KR, VR>(Map<KP, VP> map,
{KR keyCallback(KP key), VR valueCallback(VP value)}) {
Map<KR, VR> result = new HashMap<KR, VR>();
map.forEach((key, value) {
KR resultKey;
VR resultValue;
if (keyCallback != null) {
resultKey = keyCallback(key);
} else {
resultKey = key as KR;
}
if (valueCallback != null) {
resultValue = valueCallback(value);
} else {
resultValue = value as VR;
}
result[resultKey] = resultValue;
});
return result;
}
RefactoringProblemSeverity maxRefactoringProblemSeverity(
RefactoringProblemSeverity a, RefactoringProblemSeverity b) {
if (b == null) {
return a;
}
if (a == null) {
return b;
} else if (a == RefactoringProblemSeverity.INFO) {
return b;
} else if (a == RefactoringProblemSeverity.WARNING) {
if (b == RefactoringProblemSeverity.ERROR ||
b == RefactoringProblemSeverity.FATAL) {
return b;
}
} else if (a == RefactoringProblemSeverity.ERROR) {
if (b == RefactoringProblemSeverity.FATAL) {
return b;
}
}
return a;
}
/**
* Create a [RefactoringFeedback] corresponding the given [kind].
*/
RefactoringFeedback refactoringFeedbackFromJson(
JsonDecoder jsonDecoder, String jsonPath, Object json, Map feedbackJson) {
RefactoringKind kind = jsonDecoder.refactoringKind;
if (kind == RefactoringKind.EXTRACT_LOCAL_VARIABLE) {
return new ExtractLocalVariableFeedback.fromJson(
jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.EXTRACT_METHOD) {
return new ExtractMethodFeedback.fromJson(jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.INLINE_LOCAL_VARIABLE) {
return new InlineLocalVariableFeedback.fromJson(
jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.INLINE_METHOD) {
return new InlineMethodFeedback.fromJson(jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.RENAME) {
return new RenameFeedback.fromJson(jsonDecoder, jsonPath, json);
}
return null;
}
/**
* Create a [RefactoringOptions] corresponding the given [kind].
*/
RefactoringOptions refactoringOptionsFromJson(JsonDecoder jsonDecoder,
String jsonPath, Object json, RefactoringKind kind) {
if (kind == RefactoringKind.EXTRACT_LOCAL_VARIABLE) {
return new ExtractLocalVariableOptions.fromJson(
jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.EXTRACT_METHOD) {
return new ExtractMethodOptions.fromJson(jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.INLINE_METHOD) {
return new InlineMethodOptions.fromJson(jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.MOVE_FILE) {
return new MoveFileOptions.fromJson(jsonDecoder, jsonPath, json);
}
if (kind == RefactoringKind.RENAME) {
return new RenameOptions.fromJson(jsonDecoder, jsonPath, json);
}
return null;
}
/**
* Type of callbacks used to decode parts of JSON objects. [jsonPath] is a
* string describing the part of the JSON object being decoded, and [value] is
* the part to decode.
*/
typedef E JsonDecoderCallback<E>(String jsonPath, Object value);
/**
* Instances of the class [HasToJson] implement [toJson] method that returns
* a JSON presentation.
*/
abstract class HasToJson {
/**
* Returns a JSON presentation of the object.
*/
Map<String, Object> toJson();
}
/**
* JsonDecoder for decoding requests. Errors are reporting by throwing a
* [RequestFailure].
*/
class RequestDecoder extends JsonDecoder {
/**
* The request being deserialized.
*/
final Request _request;
RequestDecoder(this._request);
RefactoringKind get refactoringKind {
// Refactoring feedback objects should never appear in requests.
return null;
}
@override
dynamic mismatch(String jsonPath, String expected, [Object actual]) {
StringBuffer buffer = new StringBuffer();
buffer.write('Expected to be ');
buffer.write(expected);
if (actual != null) {
buffer.write('; found "');
buffer.write(JSON.encode(actual));
buffer.write('"');
}
return new RequestFailure(
new Response.invalidParameter(_request, jsonPath, buffer.toString()));
}
@override
dynamic missingKey(String jsonPath, String key) {
return new RequestFailure(new Response.invalidParameter(
_request, jsonPath, 'Expected to contain key ${JSON.encode(key)}'));
}
}
abstract class RequestParams implements HasToJson {
/**
* Return a request whose parameters are taken from this object and that has
* the given [id].
*/
Request toRequest(String id);
}
/**
* JsonDecoder for decoding responses from the server. This is intended to be
* used only for testing. Errors are reported using bare [Exception] objects.
*/
class ResponseDecoder extends JsonDecoder {
final RefactoringKind refactoringKind;
ResponseDecoder(this.refactoringKind);
@override
dynamic mismatch(String jsonPath, String expected, [Object actual]) {
StringBuffer buffer = new StringBuffer();
buffer.write('Expected ');
buffer.write(expected);
if (actual != null) {
buffer.write(' found "');
buffer.write(JSON.encode(actual));
buffer.write('"');
}
buffer.write(' at ');
buffer.write(jsonPath);
return new Exception(buffer.toString());
}
@override
dynamic missingKey(String jsonPath, String key) {
return new Exception('Missing key $key at $jsonPath');
}
}
/**
* The result data associated with a response.
*/
abstract class ResponseResult implements HasToJson {
/**
* Return a response whose result data is this object for the request with the
* given [id].
*/
Response toResponse(String id);
}