blob: 499eb8b4402d980f1181c1205d910c4d1ceb0175 [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 'package:analyzer/file_system/file_system.dart';
import 'package:analyzer/source/source.dart';
import 'package:analyzer/src/context/source.dart';
import 'package:analyzer/src/generated/engine.dart';
import 'package:analyzer/src/generated/sdk.dart' show DartSdk;
import 'package:path/path.dart' as path;
/// Base class providing implementations for the methods in [Source] that don't
/// require filesystem access.
abstract class BasicSource extends Source {
@override
final Uri uri;
BasicSource(this.uri);
@override
String get fullName => '$uri';
@override
int get hashCode => uri.hashCode;
@override
String get shortName => path.basename(fullName);
@override
bool operator ==(Object other) => other is Source && other.uri == uri;
}
/// Instances of the class `DartUriResolver` resolve `dart` URI's.
class DartUriResolver extends UriResolver {
/// The name of the `dart` scheme.
static String DART_SCHEME = "dart";
/// The Dart SDK against which URI's are to be resolved.
final DartSdk _sdk;
/// Initialize a newly created resolver to resolve Dart URI's against the
/// given platform within the given Dart SDK.
DartUriResolver(this._sdk);
/// Return the [DartSdk] against which URIs are to be resolved.
///
/// @return the [DartSdk] against which URIs are to be resolved.
DartSdk get dartSdk => _sdk;
@override
Uri? pathToUri(String path) {
return _sdk.pathToUri(path);
}
@override
Source? resolveAbsolute(Uri uri) {
if (!isDartUri(uri)) {
return null;
}
return _sdk.mapDartUri(uri.toString());
}
/// Return `true` if the given URI is a `dart:` URI.
///
/// @param uri the URI being tested
/// @return `true` if the given URI is a `dart:` URI
static bool isDartUri(Uri uri) => uri.isScheme(DART_SCHEME);
}
/// An implementation of an non-existing [Source].
class NonExistingSource extends Source {
static final unknown = NonExistingSource(
'/unknown.dart',
path.toUri('/unknown.dart'),
);
@override
final String fullName;
@override
final Uri uri;
NonExistingSource(this.fullName, this.uri);
@override
TimestampedData<String> get contents {
throw UnsupportedError('$fullName does not exist.');
}
@override
int get hashCode => fullName.hashCode;
@override
String get shortName => path.basename(fullName);
@override
bool operator ==(Object other) {
if (other is NonExistingSource) {
return other.uri == uri && other.fullName == fullName;
}
return false;
}
@override
bool exists() => false;
@override
String toString() => 'NonExistingSource($uri, $fullName)';
}
/// Instances of the class `SourceFactory` resolve possibly relative URI's
/// against an existing [Source].
abstract class SourceFactory {
/// Initialize a newly created source factory with the given absolute URI
/// [resolvers].
factory SourceFactory(List<UriResolver> resolvers) = SourceFactoryImpl;
/// Return the [DartSdk] associated with this [SourceFactory], or `null` if
/// there is no such SDK.
///
/// @return the [DartSdk] associated with this [SourceFactory], or `null` if
/// there is no such SDK
DartSdk? get dartSdk;
/// A table mapping package names to paths of directories containing the
/// package (or `null` if there is no registered package URI resolver).
Map<String, List<Folder>>? get packageMap;
/// Return a source object representing the given absolute URI, or `null` if
/// the URI is not a valid URI or if it is not an absolute URI.
///
/// @param absoluteUri the absolute URI to be resolved
/// @return a source object representing the absolute URI
Source? forUri(String absoluteUri);
/// Return a source object representing the given absolute URI, or `null` if
/// the URI is not an absolute URI.
///
/// @param absoluteUri the absolute URI to be resolved
/// @return a source object representing the absolute URI
Source? forUri2(Uri absoluteUri);
/// Return the URI that should be used to reference the file at the absolute
/// [path], or `null` if there is no valid way to reference the file.
/// The file at that path is not required to exist.
///
/// Throws an [ArgumentError] if the [path] is not a valid path.
Uri? pathToUri(String path);
/// Return a source representing the URI that results from resolving the given
/// (possibly relative) [containedUri] against the URI associated with the
/// [containingSource], whether or not the resulting source exists, or `null`
/// if either the [containedUri] is invalid or if it cannot be resolved
/// against the [containingSource]'s URI.
Source? resolveUri(Source? containingSource, String? containedUri);
}
/// The enumeration `SourceKind` defines the different kinds of sources that are
/// known to the analysis engine.
class SourceKind implements Comparable<SourceKind> {
/// A source containing HTML. The HTML might or might not contain Dart
/// scripts.
static const SourceKind HTML = SourceKind('HTML', 0);
/// A Dart compilation unit that is not a part of another library. Libraries
/// might or might not contain any directives, including a library directive.
static const SourceKind LIBRARY = SourceKind('LIBRARY', 1);
/// A Dart compilation unit that is part of another library. Parts contain a
/// part-of directive.
static const SourceKind PART = SourceKind('PART', 2);
/// An unknown kind of source. Used both when it is not possible to identify
/// the kind of a source and also when the kind of a source is not known
/// without performing a computation and the client does not want to spend the
/// time to identify the kind.
static const SourceKind UNKNOWN = SourceKind('UNKNOWN', 3);
static const List<SourceKind> values = [HTML, LIBRARY, PART, UNKNOWN];
/// The name of this source kind.
final String name;
/// The ordinal value of the source kind.
final int ordinal;
const SourceKind(this.name, this.ordinal);
@override
int get hashCode => ordinal;
@override
int compareTo(SourceKind other) => ordinal - other.ordinal;
@override
String toString() => name;
}
/// The abstract class `UriResolver` defines the behavior of objects that are
/// used to resolve URI's for a source factory. Subclasses of this class are
/// expected to resolve a single scheme of absolute URI.
abstract class UriResolver {
/// Return the absolute URI that should be used to reference the file at the
/// absolute [path], or `null` if this resolver cannot reference this file.
/// The file at that path is not required to exist.
///
/// Throws an [ArgumentError] if the [path] is not a valid path.
Uri? pathToUri(String path);
/// Resolve the given absolute [uri]. Return a [Source] representing the file
/// to which it was resolved, whether or not the resulting source exists, or
/// `null` if it could not be resolved because the URI is invalid.
Source? resolveAbsolute(Uri uri);
}