blob: 5bc5afa0b2d783753c12f7a36aecfa94b2304de1 [file] [log] [blame]
// Copyright (c) 2018, 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:async';
import 'dart:core';
import 'package:analysis_server/src/protocol_server.dart' hide Element;
import 'package:analysis_server/src/services/correction/status.dart';
import 'package:analysis_server/src/services/refactoring/refactoring.dart';
import 'package:analysis_server/src/services/refactoring/refactoring_internal.dart';
import 'package:analysis_server/src/services/search/search_engine.dart';
import 'package:analyzer/dart/analysis/results.dart';
import 'package:analyzer/dart/ast/ast.dart';
import 'package:analyzer/dart/element/element.dart';
import 'package:analyzer/file_system/file_system.dart';
import 'package:analyzer/src/dart/analysis/driver.dart';
import 'package:analyzer/src/generated/source.dart';
import 'package:analyzer_plugin/utilities/change_builder/change_builder_core.dart';
import 'package:analyzer_plugin/utilities/range_factory.dart';
import 'package:path/path.dart' as pathos;
/**
* [MoveFileRefactoring] implementation.
*/
class MoveFileRefactoringImpl extends RefactoringImpl
implements MoveFileRefactoring {
final ResourceProvider resourceProvider;
final pathos.Context pathContext;
final RefactoringWorkspace refactoringWorkspace;
final ResolvedUnitResult resolvedUnit;
AnalysisDriver driver;
String oldFile;
String newFile;
final packagePrefixedStringPattern = new RegExp(r'''^r?['"]+package:''');
MoveFileRefactoringImpl(this.resourceProvider, this.refactoringWorkspace,
this.resolvedUnit, this.oldFile)
: this.pathContext = resourceProvider.pathContext;
@override
String get refactoringName => 'Move File';
@override
Future<RefactoringStatus> checkFinalConditions() async {
final drivers = refactoringWorkspace.driversContaining(oldFile);
if (drivers.length != 1) {
if (refactoringWorkspace.drivers
.any((d) => pathContext.equals(d.contextRoot.root, oldFile))) {
return new RefactoringStatus.fatal(
'Renaming an analysis root is not supported ($oldFile)');
} else {
return new RefactoringStatus.fatal(
'$oldFile does not belong to an analysis root.');
}
}
driver = drivers.first;
if (!driver.resourceProvider.getFile(oldFile).exists) {
return new RefactoringStatus.fatal('$oldFile does not exist.');
}
return new RefactoringStatus();
}
@override
Future<RefactoringStatus> checkInitialConditions() async {
return new RefactoringStatus();
}
@override
Future<SourceChange> createChange() async {
final ChangeBuilder changeBuilder = new ChangeBuilder();
final CompilationUnitElement element = resolvedUnit.unit.declaredElement;
if (element == null) {
return changeBuilder.sourceChange;
}
var libraryElement = element.library;
var libraryPath = libraryElement.source.fullName;
// If this element is a library, update outgoing references inside the file.
if (element == libraryElement.definingCompilationUnit) {
// Handle part-of directives in this library
final ResolvedLibraryResult libraryResult = await driver.currentSession
.getResolvedLibraryByElement(libraryElement);
ResolvedUnitResult definingUnitResult;
for (ResolvedUnitResult result in libraryResult.units) {
if (result.isPart) {
Iterable<PartOfDirective> partOfs = result.unit.directives
.whereType<PartOfDirective>()
.where(
(po) => po.uri != null && _isRelativeUri(po.uri.stringValue));
if (partOfs.isNotEmpty) {
await changeBuilder.addFileEdit(
result.unit.declaredElement.source.fullName, (builder) {
partOfs.forEach((po) {
final oldDir = pathContext.dirname(oldFile);
final newDir = pathContext.dirname(newFile);
String newLocation =
pathContext.join(newDir, pathos.basename(newFile));
String newUri = _getRelativeUri(newLocation, oldDir);
builder.addSimpleReplacement(
new SourceRange(po.uri.offset, po.uri.length), "'$newUri'");
});
});
}
}
if (result.path == libraryPath) {
definingUnitResult = result;
}
}
await changeBuilder.addFileEdit(definingUnitResult.path, (builder) {
final String oldDir = pathContext.dirname(oldFile);
final String newDir = pathContext.dirname(newFile);
for (var directive in definingUnitResult.unit.directives) {
if (directive is UriBasedDirective) {
_updateUriReference(builder, directive, oldDir, newDir);
}
}
});
} else {
// Otherwise, we need to update any relative part-of references.
Iterable<PartOfDirective> partOfs = resolvedUnit.unit.directives
.whereType<PartOfDirective>()
.where((po) => po.uri != null && _isRelativeUri(po.uri.stringValue));
if (partOfs.isNotEmpty) {
await changeBuilder.addFileEdit(element.source.fullName, (builder) {
partOfs.forEach((po) {
final oldDir = pathContext.dirname(oldFile);
final newDir = pathContext.dirname(newFile);
String oldLocation = pathContext.join(oldDir, po.uri.stringValue);
String newUri = _getRelativeUri(oldLocation, newDir);
builder.addSimpleReplacement(
new SourceRange(po.uri.offset, po.uri.length), "'$newUri'");
});
});
}
}
// Update incoming references to this file
List<SearchMatch> matches =
await refactoringWorkspace.searchEngine.searchReferences(element);
List<SourceReference> references = getSourceReferences(matches);
for (SourceReference reference in references) {
await changeBuilder.addFileEdit(reference.file, (builder) {
String newUri = _computeNewUri(reference);
builder.addSimpleReplacement(reference.range, "'$newUri'");
});
}
return changeBuilder.sourceChange;
}
/**
* Computes the URI to use to reference [newFile] from [reference].
*/
String _computeNewUri(SourceReference reference) {
String refDir = pathContext.dirname(reference.file);
// Try to keep package: URI
if (_isPackageReference(reference)) {
Source newSource = new NonExistingSource(
newFile, pathos.toUri(newFile), UriKind.FILE_URI);
Uri restoredUri = driver.sourceFactory.restoreUri(newSource);
if (restoredUri != null) {
return restoredUri.toString();
}
}
return _getRelativeUri(newFile, refDir);
}
String _getRelativeUri(String path, String from) {
String uri = pathContext.relative(path, from: from);
List<String> parts = pathContext.split(uri);
return pathos.posix.joinAll(parts);
}
bool _isPackageReference(SourceReference reference) {
final Source source = reference.element.source;
final String quotedImportUri = source.contents.data.substring(
reference.range.offset,
reference.range.offset + reference.range.length);
return packagePrefixedStringPattern.hasMatch(quotedImportUri);
}
/**
* Checks if the given [path] represents a relative URI.
*
* The following URI's are not relative:
* `/absolute/path/file.dart`
* `dart:math`
*/
bool _isRelativeUri(String path) {
// absolute URI
if (Uri.parse(path).isAbsolute) {
return false;
}
// absolute path
if (pathContext.isAbsolute(path)) {
return false;
}
// OK
return true;
}
void _updateUriReference(FileEditBuilder builder, UriBasedDirective directive,
String oldDir, String newDir) {
var uriNode = directive.uri;
var uriValue = uriNode.stringValue;
if (_isRelativeUri(uriValue)) {
String elementPath = pathContext.join(oldDir, uriValue);
String newUri = _getRelativeUri(elementPath, newDir);
builder.addSimpleReplacement(range.node(uriNode), "'$newUri'");
}
}
}