2023-01-30 16:51:24 -05:00
|
|
|
// Packages represented on ASG
|
|
|
|
//
|
|
|
|
// Copyright (C) 2014-2023 Ryan Specialty, LLC.
|
|
|
|
//
|
|
|
|
// This file is part of TAME.
|
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
//
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
//! Package object on the ASG.
|
|
|
|
|
2023-04-20 14:55:20 -04:00
|
|
|
use super::{prelude::*, Doc, Ident, NameableMissingObject, Tpl};
|
2023-04-06 22:49:13 -04:00
|
|
|
use crate::{
|
|
|
|
f::Functor,
|
2023-05-02 15:21:03 -04:00
|
|
|
fmt::{DisplayWrapper, TtQuote},
|
2023-04-06 22:49:13 -04:00
|
|
|
parse::{util::SPair, Token},
|
|
|
|
span::Span,
|
2023-01-31 16:37:25 -05:00
|
|
|
};
|
2023-02-03 15:53:50 -05:00
|
|
|
use std::fmt::Display;
|
|
|
|
|
|
|
|
#[cfg(doc)]
|
|
|
|
use super::ObjectKind;
|
2023-01-30 16:51:24 -05:00
|
|
|
|
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
2023-04-06 22:49:13 -04:00
|
|
|
pub struct Pkg(Span, PathSpec);
|
|
|
|
|
2023-01-30 16:51:24 -05:00
|
|
|
impl Pkg {
|
2023-04-06 22:49:13 -04:00
|
|
|
/// Create a new package intended to serve as the compilation unit,
|
|
|
|
/// with an empty pathspec.
|
2023-01-30 16:51:24 -05:00
|
|
|
pub fn new<S: Into<Span>>(span: S) -> Self {
|
2023-05-02 15:21:03 -04:00
|
|
|
Self(span.into(), PathSpec::Unnamed)
|
2023-04-06 22:49:13 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Represent a package imported according to the provided
|
|
|
|
/// [`PathSpec`].
|
2023-05-02 15:21:03 -04:00
|
|
|
pub fn new_imported(pathspec: SPair) -> Self {
|
|
|
|
Self(pathspec.span(), PathSpec::TodoImport(pathspec))
|
2023-01-30 16:51:24 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn span(&self) -> Span {
|
|
|
|
match self {
|
2023-04-06 22:49:13 -04:00
|
|
|
Self(span, _) => *span,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-02 15:21:03 -04:00
|
|
|
/// Attempt to assign a canonical name to this package.
|
|
|
|
///
|
|
|
|
/// Only [`PathSpec::Unnamed`] packages may have a named assigned,
|
|
|
|
/// otherwise an [`AsgError::PkgRename`] [`Err`] will be returned.
|
|
|
|
pub fn assign_canonical_name(
|
|
|
|
self,
|
|
|
|
name: SPair,
|
|
|
|
) -> Result<Self, (Self, AsgError)> {
|
|
|
|
match self {
|
|
|
|
Self(span, PathSpec::Unnamed) => {
|
|
|
|
Ok(Self(span, PathSpec::Canonical(name)))
|
|
|
|
}
|
|
|
|
Self(_, PathSpec::Canonical(orig) | PathSpec::TodoImport(orig)) => {
|
|
|
|
Err((self, AsgError::PkgRename(orig, name)))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The canonical name for this package assigned by
|
|
|
|
/// [`Self::assign_canonical_name`],
|
|
|
|
/// if any.
|
|
|
|
pub fn canonical_name(&self) -> Option<SPair> {
|
|
|
|
match self {
|
|
|
|
Self(_, pathspec) => pathspec.canonical_name(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// The import path to this package as provided by
|
|
|
|
/// [`Self::new_imported`],
|
|
|
|
/// if any.
|
|
|
|
pub fn import_path(&self) -> Option<SPair> {
|
2023-04-06 22:49:13 -04:00
|
|
|
match self {
|
2023-05-02 15:21:03 -04:00
|
|
|
Self(_, pathspec) => pathspec.import_path(),
|
2023-01-30 16:51:24 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Display for Pkg {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
|
|
|
write!(f, "package")
|
|
|
|
}
|
|
|
|
}
|
2023-01-31 16:37:25 -05:00
|
|
|
|
2023-04-20 14:55:20 -04:00
|
|
|
impl NameableMissingObject for Pkg {
|
|
|
|
fn missing(pathspec: SPair) -> Self {
|
|
|
|
Self::new_imported(pathspec)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-07 12:19:27 -05:00
|
|
|
impl Functor<Span> for Pkg {
|
|
|
|
fn map(self, f: impl FnOnce(Span) -> Span) -> Self::Target {
|
|
|
|
match self {
|
2023-04-06 22:49:13 -04:00
|
|
|
Self(span, path) => Self(f(span), path),
|
2023-02-07 12:19:27 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-05-02 15:21:03 -04:00
|
|
|
#[derive(Debug, PartialEq, Eq)]
|
|
|
|
enum PathSpec {
|
|
|
|
/// Unnamed package.
|
|
|
|
Unnamed,
|
|
|
|
|
|
|
|
/// Canonical package name.
|
|
|
|
///
|
|
|
|
/// This is the name of the package relative to the project root.
|
|
|
|
/// This is like the module name after `crate::` in Rust,
|
|
|
|
/// but with `/` package separators in place of `::`.
|
|
|
|
Canonical(SPair),
|
|
|
|
|
|
|
|
/// Import path relative to the current package
|
|
|
|
/// (which is likely the compilation unit).
|
|
|
|
///
|
|
|
|
/// TODO: This will be replaced with [`Self::Canonical`] once that is
|
|
|
|
/// working and relative paths can be resolved against the active
|
|
|
|
/// package.
|
|
|
|
TodoImport(SPair),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl PathSpec {
|
|
|
|
fn canonical_name(&self) -> Option<SPair> {
|
|
|
|
use PathSpec::*;
|
|
|
|
|
|
|
|
match self {
|
|
|
|
Unnamed => None,
|
|
|
|
Canonical(spair) => Some(*spair),
|
|
|
|
TodoImport(_) => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn import_path(&self) -> Option<SPair> {
|
|
|
|
use PathSpec::*;
|
|
|
|
|
|
|
|
match self {
|
|
|
|
Unnamed => None,
|
|
|
|
// TODO: Let's wait to see if we actually need this,
|
|
|
|
// since we'll need to allocate and intern a `/`-prefixed
|
|
|
|
// symbol.
|
|
|
|
Canonical(_) => None,
|
|
|
|
TodoImport(spair) => Some(*spair),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Display for PathSpec {
|
|
|
|
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
|
|
|
|
use PathSpec::*;
|
|
|
|
|
|
|
|
match self {
|
|
|
|
Unnamed => {
|
|
|
|
write!(f, "unnamed package")
|
|
|
|
}
|
|
|
|
Canonical(spair) => write!(f, "package {}", TtQuote::wrap(spair)),
|
|
|
|
TodoImport(spair) => write!(
|
|
|
|
f,
|
|
|
|
"package import {} relative to compilation unit",
|
|
|
|
TtQuote::wrap(spair)
|
|
|
|
),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-25 23:56:05 -05:00
|
|
|
object_rel! {
|
2023-02-03 15:53:50 -05:00
|
|
|
/// Packages serve as a root for all identifiers defined therein,
|
|
|
|
/// and so an edge to [`Ident`] will never be a cross edge.
|
|
|
|
///
|
|
|
|
/// Imported [`Ident`]s do not have edges from this package.
|
2023-02-25 23:56:05 -05:00
|
|
|
Pkg -> {
|
2023-04-06 22:49:13 -04:00
|
|
|
// Package import
|
|
|
|
cross Pkg,
|
|
|
|
|
|
|
|
// Identified objects owned by this package.
|
2023-02-25 23:56:05 -05:00
|
|
|
tree Ident,
|
tamer: Very basic support for template application NIR -> xmli
This this a big change that's difficult to break up, and I don't have the
energy after it.
This introduces nullary template application, short- and long-form. Note
that a body of the short form is a `@values@` argument, so that's not
supported yet.
This continues to formalize the idea of what "template application" and
"template expansion" mean in TAMER. It makes a separate `TplApply`
unnecessary, because now application is simply a reference to a
template. Expansion and application are one and the same: when a template
expands, it'll re-bind metavariables to the parent context. So in a
template context, this amounts to application.
But applying a closed template will have nothing to bind, and so is
equivalent to expansion. And since `Meta` objects are not valid outside of
a `Tpl` context, applying a non-closed template outside of another template
will be invalid.
So we get all of this with a single primitive (getting the "value" of a
template).
The expansion is conceptually like `,@` in Lisp, where we're splicing trees.
It's a mess in some spots, but I want to get this committed before I do a
little bit of cleanup.
2023-03-17 10:25:56 -04:00
|
|
|
|
|
|
|
// Anonymous templates are used for expansion.
|
|
|
|
tree Tpl,
|
2023-04-12 10:03:37 -04:00
|
|
|
|
|
|
|
// Arbitrary blocks of text serving as documentation.
|
|
|
|
tree Doc,
|
2023-01-31 16:37:25 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ObjectIndex<Pkg> {
|
2023-02-07 12:19:27 -05:00
|
|
|
/// Complete the definition of a package.
|
|
|
|
pub fn close(self, asg: &mut Asg, span: Span) -> Self {
|
|
|
|
self.map_obj(asg, Pkg::fmap(|open| open.merge(span).unwrap_or(open)))
|
|
|
|
}
|
2023-04-06 22:49:13 -04:00
|
|
|
|
2023-05-02 15:21:03 -04:00
|
|
|
/// Attempt to assign a canonical name to this package.
|
|
|
|
///
|
2023-05-02 16:07:25 -04:00
|
|
|
/// This assignment will fail if the package either already has a name
|
|
|
|
/// or if a package of the same name has already been declared.
|
2023-05-02 15:21:03 -04:00
|
|
|
pub fn assign_canonical_name(
|
|
|
|
self,
|
|
|
|
asg: &mut Asg,
|
|
|
|
name: SPair,
|
|
|
|
) -> Result<Self, AsgError> {
|
2023-05-02 16:07:25 -04:00
|
|
|
let oi_root = asg.root(name);
|
|
|
|
|
|
|
|
asg.try_index(oi_root, name, self).map_err(|oi_prev| {
|
|
|
|
let prev = oi_prev.resolve(asg);
|
|
|
|
|
|
|
|
// unwrap note: a canonical name must exist for this error to
|
|
|
|
// have been thrown,
|
|
|
|
// but this will at least not blow up if something really
|
|
|
|
// odd happens.
|
|
|
|
AsgError::PkgRedeclare(prev.canonical_name().unwrap_or(name), name)
|
|
|
|
})?;
|
|
|
|
|
2023-05-02 15:21:03 -04:00
|
|
|
self.try_map_obj(asg, |pkg| pkg.assign_canonical_name(name))
|
|
|
|
}
|
|
|
|
|
2023-04-06 22:49:13 -04:00
|
|
|
/// Indicate that a package should be imported at the provided
|
|
|
|
/// pathspec.
|
|
|
|
///
|
|
|
|
/// This simply adds the import to the graph;
|
|
|
|
/// package loading must be performed by another subsystem.
|
|
|
|
pub fn import(self, asg: &mut Asg, pathspec: SPair) -> Self {
|
|
|
|
let oi_import = asg.create(Pkg::new_imported(pathspec));
|
|
|
|
self.add_edge_to(asg, oi_import, Some(pathspec.span()))
|
|
|
|
}
|
2023-04-12 10:03:37 -04:00
|
|
|
|
|
|
|
/// Arbitrary text serving as documentation in a literate style.
|
|
|
|
pub fn append_doc_text(&self, asg: &mut Asg, text: SPair) -> Self {
|
|
|
|
let oi_doc = asg.create(Doc::new_text(text));
|
|
|
|
self.add_edge_to(asg, oi_doc, None)
|
|
|
|
}
|
2023-01-31 16:37:25 -05:00
|
|
|
}
|