From 7f6ded7bc7be9ea2d3551b4e372b9b762add14f4 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 08:22:14 +0000 Subject: [PATCH 01/31] ErrorsAlgebra -> ReportAlgebra --- .../scala/aqua/lsp/LocationsInterpreter.scala | 3 ++- .../main/scala/aqua/lsp/LspSemantics.scala | 2 +- .../scala/aqua/semantics/CompilerState.scala | 2 +- .../main/scala/aqua/semantics/Semantics.scala | 22 +++++++++---------- .../semantics/rules/StackInterpreter.scala | 2 +- .../aqua/semantics/rules/ValuesAlgebra.scala | 6 ++--- .../abilities/AbilitiesInterpreter.scala | 2 +- .../definitions/DefinitionsInterpreter.scala | 2 +- .../rules/names/NamesInterpreter.scala | 2 +- .../ReportAlgebra.scala} | 4 ++-- .../{errors => report}/ReportErrors.scala | 4 ++-- .../rules/types/TypesInterpreter.scala | 2 +- .../src/test/scala/aqua/semantics/Utils.scala | 2 +- 13 files changed, 28 insertions(+), 27 deletions(-) rename semantics/src/main/scala/aqua/semantics/rules/{errors/ErrorsAlgebra.scala => report/ReportAlgebra.scala} (72%) rename semantics/src/main/scala/aqua/semantics/rules/{errors => report}/ReportErrors.scala (70%) diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala index 556a52e7d..27c17983c 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala @@ -2,8 +2,9 @@ package aqua.lsp import aqua.parser.lexer.Token import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.locations.{LocationsAlgebra, LocationsState} + import cats.data.State import monocle.Lens import monocle.macros.GenLens diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala index 49b89994f..0808e4415 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala @@ -3,7 +3,7 @@ package aqua.lsp import aqua.parser.Ast import aqua.parser.head.{ImportExpr, ImportFromExpr, UseExpr, UseFromExpr} import aqua.parser.lexer.{LiteralToken, Token} -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.locations.LocationsState import aqua.semantics.{CompilerState, RawSemantics, RulesViolated, SemanticError, Semantics} import cats.data.Validated.{Invalid, Valid} diff --git a/semantics/src/main/scala/aqua/semantics/CompilerState.scala b/semantics/src/main/scala/aqua/semantics/CompilerState.scala index bc52ca490..42f6a72ca 100644 --- a/semantics/src/main/scala/aqua/semantics/CompilerState.scala +++ b/semantics/src/main/scala/aqua/semantics/CompilerState.scala @@ -9,7 +9,7 @@ import aqua.semantics.rules.locations.LocationsState import aqua.semantics.rules.names.NamesState import aqua.semantics.rules.types.TypesState import aqua.semantics.rules.mangler.ManglerState -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import cats.Semigroup import cats.data.{Chain, State} diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index 74686eff6..c159879cb 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -14,8 +14,8 @@ import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgeb import aqua.semantics.rules.names.{NamesAlgebra, NamesInterpreter} import aqua.semantics.rules.mangler.{ManglerAlgebra, ManglerInterpreter} import aqua.semantics.rules.types.{TypesAlgebra, TypesInterpreter} -import aqua.semantics.rules.errors.ReportErrors -import aqua.semantics.rules.errors.ErrorsAlgebra +import aqua.semantics.rules.report.ReportErrors +import aqua.semantics.rules.report.ReportAlgebra import aqua.raw.ops.* import cats.arrow.FunctionK @@ -105,22 +105,22 @@ object RawSemantics extends Logging { private def elseWithoutIf[S[_], G[_]]( token: Token[S] - )(using E: ErrorsAlgebra[S, G]): G[Unit] = + )(using E: ReportAlgebra[S, G]): G[Unit] = E.report(token, "Unexpected `else` without `if`" :: Nil) private def catchWithoutTry[S[_], G[_]]( token: Token[S] - )(using E: ErrorsAlgebra[S, G]): G[Unit] = + )(using E: ReportAlgebra[S, G]): G[Unit] = E.report(token, "Unexpected `catch` without `try`" :: Nil) private def otherwiseWithoutPrev[S[_], G[_]]( token: Token[S] - )(using E: ErrorsAlgebra[S, G]): G[Unit] = + )(using E: ReportAlgebra[S, G]): G[Unit] = E.report(token, "Unexpected `otherwise` without previous instruction" :: Nil) private def parWithoutPrev[S[_], G[_]]( token: Token[S] - )(using E: ErrorsAlgebra[S, G]): G[Unit] = + )(using E: ReportAlgebra[S, G]): G[Unit] = E.report(token, "Unexpected `par` without previous instruction" :: Nil) /** @@ -140,7 +140,7 @@ object RawSemantics extends Logging { private def rawTagCombine[S[_], G[_]: Monad]( prev: RawTagWithToken[S], next: RawTagWithToken[S] - )(using E: ErrorsAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = + )(using E: ReportAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = (prev.tag, next.tag) match { case (_: IfTag, IfTag.Else) => prev.append(next).some.pure @@ -197,7 +197,7 @@ object RawSemantics extends Logging { */ private def rawTagSingleCheck[S[_], G[_]: Monad]( single: RawTagWithToken[S] - )(using E: ErrorsAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = + )(using E: ReportAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = single.tag match { case IfTag.Else => elseWithoutIf(single.token).as(none) case TryTag.Catch => catchWithoutTry(single.token).as(none) @@ -238,7 +238,7 @@ object RawSemantics extends Logging { */ def step[G[_]: Monad]( next: RawWithToken[S] - )(using ErrorsAlgebra[S, G]): G[InnersFoldState[S]] = + )(using ReportAlgebra[S, G]): G[InnersFoldState[S]] = last.fold(copy(last = next.some).pure)(prev => (prev.toTag, next.toTag) .traverseN(rawTagCombine) @@ -262,7 +262,7 @@ object RawSemantics extends Logging { * Produce result of folding */ def result[G[_]: Monad](using - ErrorsAlgebra[S, G] + ReportAlgebra[S, G] ): G[Option[Raw]] = if (acc.isEmpty) // Hack to report error if single tag in block is incorrect @@ -288,7 +288,7 @@ object RawSemantics extends Logging { T: TypesAlgebra[S, G], D: DefinitionsAlgebra[S, G], L: LocationsAlgebra[S, G], - E: ErrorsAlgebra[S, G] + E: ReportAlgebra[S, G] ): (Expr[S], Chain[G[RawWithToken[S]]]) => Eval[G[RawWithToken[S]]] = (expr, inners) => Eval later ExprSem .getProg[S, G](expr) diff --git a/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala index dd7e79f75..6d81f7baa 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala @@ -1,7 +1,7 @@ package aqua.semantics.rules import aqua.parser.lexer.Token -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import cats.data.State import cats.syntax.functor.* diff --git a/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala b/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala index a1410b4d9..d4edf63cd 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala @@ -7,7 +7,7 @@ import aqua.raw.value.* import aqua.semantics.rules.abilities.AbilitiesAlgebra import aqua.semantics.rules.names.NamesAlgebra import aqua.semantics.rules.types.TypesAlgebra -import aqua.semantics.rules.errors.ErrorsAlgebra +import aqua.semantics.rules.report.ReportAlgebra import aqua.types.* import cats.Monad @@ -30,7 +30,7 @@ import scala.collection.immutable.SortedMap class ValuesAlgebra[S[_], Alg[_]: Monad](using N: NamesAlgebra[S, Alg], T: TypesAlgebra[S, Alg], - E: ErrorsAlgebra[S, Alg], + E: ReportAlgebra[S, Alg], A: AbilitiesAlgebra[S, Alg] ) extends Logging { @@ -419,7 +419,7 @@ object ValuesAlgebra { N: NamesAlgebra[S, Alg], T: TypesAlgebra[S, Alg], A: AbilitiesAlgebra[S, Alg], - E: ErrorsAlgebra[S, Alg] + E: ReportAlgebra[S, Alg] ): ValuesAlgebra[S, Alg] = new ValuesAlgebra[S, Alg] } diff --git a/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala index 69275e8f5..080925842 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala @@ -4,7 +4,7 @@ import aqua.parser.lexer.{Name, NamedTypeToken, Token, ValueToken} import aqua.raw.value.ValueRaw import aqua.raw.{RawContext, ServiceRaw} import aqua.semantics.Levenshtein -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.mangler.ManglerAlgebra import aqua.semantics.rules.locations.LocationsAlgebra import aqua.semantics.rules.{abilities, StackInterpreter} diff --git a/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala index 1089a55fa..188f9f5d1 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala @@ -2,7 +2,7 @@ package aqua.semantics.rules.definitions import aqua.parser.lexer.{Name, NamedTypeToken, Token} import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.abilities.AbilitiesState import aqua.semantics.rules.locations.{LocationsAlgebra, LocationsState} import aqua.semantics.rules.types.TypesState diff --git a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala index 7a0c9121a..d37360c08 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala @@ -3,7 +3,7 @@ package aqua.semantics.rules.names import aqua.parser.lexer.{Name, Token} import aqua.semantics.Levenshtein import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.locations.LocationsAlgebra import aqua.types.{AbilityType, ArrowType, StreamType, Type} diff --git a/semantics/src/main/scala/aqua/semantics/rules/errors/ErrorsAlgebra.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala similarity index 72% rename from semantics/src/main/scala/aqua/semantics/rules/errors/ErrorsAlgebra.scala rename to semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala index ddd5dfc71..8a3db353b 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/errors/ErrorsAlgebra.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala @@ -1,8 +1,8 @@ -package aqua.semantics.rules.errors +package aqua.semantics.rules.report import aqua.parser.lexer.Token -trait ErrorsAlgebra[S[_], Alg[_]] { +trait ReportAlgebra[S[_], Alg[_]] { def report(token: Token[S], hints: List[String]): Alg[Unit] def report(token: Token[S], hint: String): Alg[Unit] = diff --git a/semantics/src/main/scala/aqua/semantics/rules/errors/ReportErrors.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala similarity index 70% rename from semantics/src/main/scala/aqua/semantics/rules/errors/ReportErrors.scala rename to semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala index 83fd2a2bd..f5ffdfcaa 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/errors/ReportErrors.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala @@ -1,10 +1,10 @@ -package aqua.semantics.rules.errors +package aqua.semantics.rules.report import aqua.parser.lexer.Token import cats.data.State -trait ReportErrors[S[_], X] extends ErrorsAlgebra[S, State[X, *]] { +trait ReportErrors[S[_], X] extends ReportAlgebra[S, State[X, *]] { def apply(st: X, token: Token[S], hints: List[String]): X def report(token: Token[S], hints: List[String]): State[X, Unit] = diff --git a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala index 14b32ca29..c66a53406 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala @@ -12,7 +12,7 @@ import aqua.raw.value.{ } import aqua.semantics.rules.locations.LocationsAlgebra import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.types.TypesStateHelper.{TypeResolution, TypeResolutionError} import aqua.types.* diff --git a/semantics/src/test/scala/aqua/semantics/Utils.scala b/semantics/src/test/scala/aqua/semantics/Utils.scala index 2605c8d63..17e8b3cf3 100644 --- a/semantics/src/test/scala/aqua/semantics/Utils.scala +++ b/semantics/src/test/scala/aqua/semantics/Utils.scala @@ -5,7 +5,7 @@ import aqua.parser.lexer.{Name, Token} import aqua.parser.lift.Span import aqua.raw.{Raw, RawContext} import aqua.semantics.expr.func.ClosureSem -import aqua.semantics.rules.errors.ReportErrors +import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.abilities.{AbilitiesAlgebra, AbilitiesInterpreter, AbilitiesState} import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgebra} import aqua.semantics.rules.names.{NamesAlgebra, NamesInterpreter, NamesState} From dac15b1678fa229e471c6767b601b7206fffb5dd Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 09:17:12 +0000 Subject: [PATCH 02/31] Refactor ReportAlgebra --- .../scala/aqua/lsp/LocationsInterpreter.scala | 8 +- .../main/scala/aqua/lsp/LspSemantics.scala | 10 +- .../scala/aqua/semantics/CompilerState.scala | 25 +- .../main/scala/aqua/semantics/Semantics.scala | 27 +- .../semantics/rules/StackInterpreter.scala | 12 +- .../aqua/semantics/rules/ValuesAlgebra.scala | 6 +- .../abilities/AbilitiesInterpreter.scala | 40 +-- .../definitions/DefinitionsInterpreter.scala | 13 +- .../rules/names/NamesInterpreter.scala | 38 +-- .../rules/report/ReportAlgebra.scala | 6 +- .../semantics/rules/report/ReportErrors.scala | 12 - .../rules/report/ReportInterpreter.scala | 18 ++ .../semantics/rules/report/ReportState.scala | 25 ++ .../rules/types/TypesInterpreter.scala | 253 ++++++++++-------- .../src/test/scala/aqua/semantics/Utils.scala | 8 +- .../aqua/semantics/ValuesAlgebraSpec.scala | 4 +- 16 files changed, 282 insertions(+), 223 deletions(-) delete mode 100644 semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala create mode 100644 semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala create mode 100644 semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala index 27c17983c..342b202b5 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LocationsInterpreter.scala @@ -2,7 +2,6 @@ package aqua.lsp import aqua.parser.lexer.Token import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.locations.{LocationsAlgebra, LocationsState} import cats.data.State @@ -10,9 +9,8 @@ import monocle.Lens import monocle.macros.GenLens import scribe.Logging -class LocationsInterpreter[S[_], X](implicit - lens: Lens[X, LocationsState[S]], - error: ReportErrors[S, X] +class LocationsInterpreter[S[_], X](using + lens: Lens[X, LocationsState[S]] ) extends LocationsAlgebra[S, State[X, *]] with Logging { type SX[A] = State[X, A] @@ -21,7 +19,7 @@ class LocationsInterpreter[S[_], X](implicit GenLens[LocationsState[S]](_.stack) ) - import stack.{getState, mapStackHead, modify, report} + import stack.* override def addToken(name: String, token: Token[S]): State[X, Unit] = modify { st => st.copy(tokens = st.tokens.updated(name, token)) diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala index 0808e4415..eb1c42845 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala @@ -3,7 +3,6 @@ package aqua.lsp import aqua.parser.Ast import aqua.parser.head.{ImportExpr, ImportFromExpr, UseExpr, UseFromExpr} import aqua.parser.lexer.{LiteralToken, Token} -import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.locations.LocationsState import aqua.semantics.{CompilerState, RawSemantics, RulesViolated, SemanticError, Semantics} import cats.data.Validated.{Invalid, Valid} @@ -53,15 +52,10 @@ class LspSemantics[S[_]] extends Semantics[S, LspContext[S]] { val importTokens = getImportTokens(ast) - implicit val ls: Lens[CompilerState[S], LocationsState[S]] = + given Lens[CompilerState[S], LocationsState[S]] = GenLens[CompilerState[S]](_.locations) - import monocle.syntax.all.* - implicit val re: ReportErrors[S, CompilerState[S]] = - (st: CompilerState[S], token: Token[S], hints: List[String]) => - st.focus(_.errors).modify(_.append(RulesViolated(token, hints))) - - implicit val locationsInterpreter: LocationsInterpreter[S, CompilerState[S]] = + given LocationsInterpreter[S, CompilerState[S]] = new LocationsInterpreter[S, CompilerState[S]]() RawSemantics diff --git a/semantics/src/main/scala/aqua/semantics/CompilerState.scala b/semantics/src/main/scala/aqua/semantics/CompilerState.scala index 42f6a72ca..3774467b7 100644 --- a/semantics/src/main/scala/aqua/semantics/CompilerState.scala +++ b/semantics/src/main/scala/aqua/semantics/CompilerState.scala @@ -9,7 +9,7 @@ import aqua.semantics.rules.locations.LocationsState import aqua.semantics.rules.names.NamesState import aqua.semantics.rules.types.TypesState import aqua.semantics.rules.mangler.ManglerState -import aqua.semantics.rules.report.ReportErrors +import aqua.semantics.rules.report.ReportState import cats.Semigroup import cats.data.{Chain, State} @@ -19,14 +19,16 @@ import monocle.Lens import monocle.macros.GenLens case class CompilerState[S[_]]( - errors: Chain[SemanticError[S]] = Chain.empty[SemanticError[S]], + report: ReportState[S] = ReportState[S](), mangler: ManglerState = ManglerState(), names: NamesState[S] = NamesState[S](), abilities: AbilitiesState[S] = AbilitiesState[S](), types: TypesState[S] = TypesState[S](), definitions: DefinitionsState[S] = DefinitionsState[S](), locations: LocationsState[S] = LocationsState[S]() -) +) { + lazy val errors: Chain[SemanticError[S]] = report.errors +} object CompilerState { type St[S[_]] = State[CompilerState[S], Raw] @@ -38,6 +40,9 @@ object CompilerState { types = TypesState.init[F](ctx) ) + given [S[_]]: Lens[CompilerState[S], ReportState[S]] = + GenLens[CompilerState[S]](_.report) + given [S[_]]: Lens[CompilerState[S], NamesState[S]] = GenLens[CompilerState[S]](_.names) @@ -53,18 +58,6 @@ object CompilerState { given [S[_]]: Lens[CompilerState[S], DefinitionsState[S]] = GenLens[CompilerState[S]](_.definitions) - given [S[_]]: ReportErrors[S, CompilerState[S]] = - new ReportErrors[S, CompilerState[S]] { - import monocle.syntax.all.* - - override def apply( - st: CompilerState[S], - token: Token[S], - hints: List[String] - ): CompilerState[S] = - st.focus(_.errors).modify(_.append(RulesViolated(token, hints))) - } - given [S[_]]: Monoid[St[S]] with { override def empty: St[S] = State.pure(Raw.Empty("compiler state monoid empty")) @@ -73,7 +66,7 @@ object CompilerState { b <- y.get _ <- State.set( CompilerState[S]( - a.errors ++ b.errors, + a.report |+| b.report, a.mangler |+| b.mangler, a.names |+| b.names, a.abilities |+| b.abilities, diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index c159879cb..61fc85db7 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -14,8 +14,7 @@ import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgeb import aqua.semantics.rules.names.{NamesAlgebra, NamesInterpreter} import aqua.semantics.rules.mangler.{ManglerAlgebra, ManglerInterpreter} import aqua.semantics.rules.types.{TypesAlgebra, TypesInterpreter} -import aqua.semantics.rules.report.ReportErrors -import aqua.semantics.rules.report.ReportAlgebra +import aqua.semantics.rules.report.{ReportAlgebra, ReportInterpreter} import aqua.raw.ops.* import cats.arrow.FunctionK @@ -55,7 +54,7 @@ class RawSemantics[S[_]](implicit p: Picker[RawContext]) extends Semantics[S, Ra init: RawContext ): ValidatedNec[SemanticError[S], RawContext] = { - implicit val locationsInterpreter: DummyLocationsInterpreter[S, CompilerState[S]] = + given LocationsAlgebra[S, State[CompilerState[S], *]] = new DummyLocationsInterpreter[S, CompilerState[S]]() RawSemantics @@ -105,23 +104,23 @@ object RawSemantics extends Logging { private def elseWithoutIf[S[_], G[_]]( token: Token[S] - )(using E: ReportAlgebra[S, G]): G[Unit] = - E.report(token, "Unexpected `else` without `if`" :: Nil) + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `else` without `if`" :: Nil) private def catchWithoutTry[S[_], G[_]]( token: Token[S] - )(using E: ReportAlgebra[S, G]): G[Unit] = - E.report(token, "Unexpected `catch` without `try`" :: Nil) + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `catch` without `try`" :: Nil) private def otherwiseWithoutPrev[S[_], G[_]]( token: Token[S] - )(using E: ReportAlgebra[S, G]): G[Unit] = - E.report(token, "Unexpected `otherwise` without previous instruction" :: Nil) + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `otherwise` without previous instruction" :: Nil) private def parWithoutPrev[S[_], G[_]]( token: Token[S] - )(using E: ReportAlgebra[S, G]): G[Unit] = - E.report(token, "Unexpected `par` without previous instruction" :: Nil) + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `par` without previous instruction" :: Nil) /** * Optionally combine two [[RawTag.Tree]] into one. @@ -307,6 +306,8 @@ object RawSemantics extends Logging { LocationsAlgebra[S, Interpreter[S, *]] ): Interpreter[S, Raw] = { + given ReportAlgebra[S, Interpreter[S, *]] = + new ReportInterpreter[S, CompilerState[S]] given TypesAlgebra[S, Interpreter[S, *]] = new TypesInterpreter[S, CompilerState[S]] given ManglerAlgebra[Interpreter[S, *]] = @@ -334,8 +335,8 @@ object RawSemantics extends Logging { ast: Ast[S], initState: CompilerState[S], init: RawContext - )(implicit - locations: LocationsAlgebra[S, Interpreter[S, *]] + )(using + LocationsAlgebra[S, Interpreter[S, *]] ): Eval[(CompilerState[S], RawContext)] = astToState[S](ast) .run(initState) diff --git a/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala index 6d81f7baa..052c0b4d8 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/StackInterpreter.scala @@ -1,7 +1,6 @@ package aqua.semantics.rules import aqua.parser.lexer.Token -import aqua.semantics.rules.report.ReportErrors import cats.data.State import cats.syntax.functor.* @@ -10,21 +9,12 @@ import monocle.Lens case class StackInterpreter[S[_], X, St, Fr]( stackLens: Lens[St, List[Fr]] -)(using - lens: Lens[X, St], - error: ReportErrors[S, X] -) { +)(using lens: Lens[X, St]) { type SX[A] = State[X, A] def getState: SX[St] = State.get.map(lens.get) def setState(st: St): SX[Unit] = State.modify(s => lens.replace(st)(s)) - def reportError(t: Token[S], hints: List[String]): SX[Unit] = - State.modify(error(_, t, hints)) - - def report(t: Token[S], hint: String): SX[Unit] = - State.modify(error(_, t, hint :: Nil)) - def modify(f: St => St): SX[Unit] = State.modify(lens.modify(f)) diff --git a/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala b/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala index d4edf63cd..b3d0ffa4a 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/ValuesAlgebra.scala @@ -30,8 +30,8 @@ import scala.collection.immutable.SortedMap class ValuesAlgebra[S[_], Alg[_]: Monad](using N: NamesAlgebra[S, Alg], T: TypesAlgebra[S, Alg], - E: ReportAlgebra[S, Alg], - A: AbilitiesAlgebra[S, Alg] + A: AbilitiesAlgebra[S, Alg], + report: ReportAlgebra[S, Alg] ) extends Logging { private def resolveSingleProperty(rootType: Type, op: PropertyOp[S]): Alg[Option[PropertyRaw]] = @@ -305,7 +305,7 @@ class ValuesAlgebra[S[_], Alg[_]: Monad](using _.flatTraverse { case ca: CallArrowRaw => ca.some.pure[Alg] // TODO: better error message (`raw` formatting) - case raw => E.report(v, s"Expected arrow call, got $raw").as(none) + case raw => report.error(v, s"Expected arrow call, got $raw").as(none) } ) diff --git a/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala index 080925842..e5c89c664 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/abilities/AbilitiesInterpreter.scala @@ -4,7 +4,7 @@ import aqua.parser.lexer.{Name, NamedTypeToken, Token, ValueToken} import aqua.raw.value.ValueRaw import aqua.raw.{RawContext, ServiceRaw} import aqua.semantics.Levenshtein -import aqua.semantics.rules.report.ReportErrors +import aqua.semantics.rules.report.ReportAlgebra import aqua.semantics.rules.mangler.ManglerAlgebra import aqua.semantics.rules.locations.LocationsAlgebra import aqua.semantics.rules.{abilities, StackInterpreter} @@ -22,7 +22,7 @@ import monocle.macros.GenLens class AbilitiesInterpreter[S[_], X](using lens: Lens[X, AbilitiesState[S]], - error: ReportErrors[S, X], + report: ReportAlgebra[S, State[X, *]], mangler: ManglerAlgebra[State[X, *]], locations: LocationsAlgebra[S, State[X, *]] ) extends AbilitiesAlgebra[S, State[X, *]] { @@ -33,7 +33,7 @@ class AbilitiesInterpreter[S[_], X](using GenLens[AbilitiesState[S]](_.stack) ) - import stackInt.{getState, mapStackHead, mapStackHeadM, modify, report} + import stackInt.* override def defineService( name: NamedTypeToken[S], @@ -45,10 +45,12 @@ class AbilitiesInterpreter[S[_], X](using getState .map(_.definitions.get(name.value).exists(_ == name)) .flatMap(exists => - report( - name, - "Service with this name was already defined" - ).whenA(!exists) + report + .error( + name, + "Service with this name was already defined" + ) + .whenA(!exists) ) .as(false) case false => @@ -74,21 +76,23 @@ class AbilitiesInterpreter[S[_], X](using abCtx.funcs .get(arrow.value) .fold( - report( - arrow, - Levenshtein.genMessage( - s"Ability is found, but arrow '${arrow.value}' isn't found in scope", - arrow.value, - abCtx.funcs.keys.toList + report + .error( + arrow, + Levenshtein.genMessage( + s"Ability is found, but arrow '${arrow.value}' isn't found in scope", + arrow.value, + abCtx.funcs.keys.toList + ) ) - ).as(none) + .as(none) ) { fn => // TODO: add name and arrow separately // TODO: find tokens somewhere addServiceArrowLocation(name, arrow).as(fn.arrow.`type`.some) } case None => - report(name, "Ability with this name is undefined").as(none) + report.error(name, "Ability with this name is undefined").as(none) } override def renameService(name: NamedTypeToken[S]): SX[Option[String]] = @@ -102,7 +106,7 @@ class AbilitiesInterpreter[S[_], X](using .map(newName => h.setServiceRename(name.value, newName) -> newName) ).map(_.some) case false => - report(name, "Service with this name is not registered").as(none) + report.error(name, "Service with this name is not registered").as(none) } override def getServiceRename(name: NamedTypeToken[S]): State[X, Option[String]] = @@ -111,8 +115,8 @@ class AbilitiesInterpreter[S[_], X](using getState.map(_.getServiceRename(name.value)) ).flatMapN { case (true, Some(rename)) => rename.some.pure - case (false, _) => report(name, "Service with this name is undefined").as(none) - case (_, None) => report(name, "Service ID is undefined").as(none) + case (false, _) => report.error(name, "Service with this name is undefined").as(none) + case (_, None) => report.error(name, "Service ID is undefined").as(none) } override def beginScope(token: Token[S]): SX[Unit] = diff --git a/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala index 188f9f5d1..0103b5986 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/definitions/DefinitionsInterpreter.scala @@ -2,7 +2,7 @@ package aqua.semantics.rules.definitions import aqua.parser.lexer.{Name, NamedTypeToken, Token} import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.report.ReportErrors +import aqua.semantics.rules.report.ReportAlgebra import aqua.semantics.rules.abilities.AbilitiesState import aqua.semantics.rules.locations.{LocationsAlgebra, LocationsState} import aqua.semantics.rules.types.TypesState @@ -21,7 +21,7 @@ import scala.collection.immutable.SortedMap class DefinitionsInterpreter[S[_], X](implicit lens: Lens[X, DefinitionsState[S]], - error: ReportErrors[S, X], + report: ReportAlgebra[S, State[X, *]], locations: LocationsAlgebra[S, State[X, *]] ) extends DefinitionsAlgebra[S, State[X, *]] { type SX[A] = State[X, A] @@ -31,9 +31,6 @@ class DefinitionsInterpreter[S[_], X](implicit private def modify(f: DefinitionsState[S] => DefinitionsState[S]): SX[Unit] = State.modify(lens.modify(f)) - def report(t: Token[S], hint: String): SX[Unit] = - State.modify(error(_, t, hint :: Nil)) - def define(name: Name[S], `type`: Type, defName: String): SX[Boolean] = getState.map(_.definitions.get(name.value)).flatMap { case None => @@ -47,7 +44,8 @@ class DefinitionsInterpreter[S[_], X](implicit ) .as(true) case Some(_) => - report(name, s"Cannot define $defName `${name.value}`, it was already defined above") + report + .error(name, s"Cannot define $defName `${name.value}`, it was already defined above") .as(false) } @@ -82,7 +80,8 @@ class DefinitionsInterpreter[S[_], X](implicit st.copy(definitions = Map.empty) }.as(arrs.some) case None => - report(token, "Cannot purge arrows, no arrows provided") + report + .error(token, "Cannot purge arrows, no arrows provided") .as(none) } } diff --git a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala index d37360c08..bd718aea1 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala @@ -3,7 +3,7 @@ package aqua.semantics.rules.names import aqua.parser.lexer.{Name, Token} import aqua.semantics.Levenshtein import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.report.ReportErrors +import aqua.semantics.rules.report.ReportAlgebra import aqua.semantics.rules.locations.LocationsAlgebra import aqua.types.{AbilityType, ArrowType, StreamType, Type} @@ -15,9 +15,9 @@ import cats.syntax.all.* import monocle.Lens import monocle.macros.GenLens -class NamesInterpreter[S[_], X](implicit +class NamesInterpreter[S[_], X](using lens: Lens[X, NamesState[S]], - error: ReportErrors[S, X], + report: ReportAlgebra[S, State[X, *]], locations: LocationsAlgebra[S, State[X, *]] ) extends NamesAlgebra[S, State[X, *]] { @@ -25,7 +25,7 @@ class NamesInterpreter[S[_], X](implicit GenLens[NamesState[S]](_.stack) ) - import stackInt.{getState, mapStackHead, mapStackHeadM, mapStackHead_, modify, report} + import stackInt.* type SX[A] = State[X, A] @@ -44,7 +44,7 @@ class NamesInterpreter[S[_], X](implicit .flatTap { case None if mustBeDefined => getState.flatMap(st => - report( + report.error( name, Levenshtein .genMessage( @@ -73,14 +73,15 @@ class NamesInterpreter[S[_], X](implicit locations.pointLocation(name.value, name).map(_ => Option(at)) case _ => getState.flatMap(st => - report( - name, - Levenshtein.genMessage( - s"Name '${name.value}' not found in scope", - name.value, - st.allNames.toList + report + .error( + name, + Levenshtein.genMessage( + s"Name '${name.value}' not found in scope", + name.value, + st.allNames.toList + ) ) - ) .as(Option.empty[ArrowType]) ) } @@ -98,11 +99,11 @@ class NamesInterpreter[S[_], X](implicit case Some(_) => getState.map(_.definitions.get(name.value).exists(_ == name)).flatMap { case true => State.pure(false) - case false => report(name, "This name was already defined in the scope").as(false) + case false => report.error(name, "This name was already defined in the scope").as(false) } case None => - mapStackHeadM(report(name, "Cannot define a variable in the root scope").as(false))(fr => - (fr.addName(name, `type`) -> true).pure + mapStackHeadM(report.error(name, "Cannot define a variable in the root scope").as(false))( + fr => (fr.addName(name, `type`) -> true).pure ) <* locations.addToken(name.value, name) } @@ -121,7 +122,7 @@ class NamesInterpreter[S[_], X](implicit override def defineConstant(name: Name[S], `type`: Type): SX[Boolean] = readName(name.value).flatMap { case Some(_) => - report(name, "This name was already defined in the scope").as(false) + report.error(name, "This name was already defined in the scope").as(false) case None => modify(st => st.copy( @@ -135,7 +136,7 @@ class NamesInterpreter[S[_], X](implicit case Some(_) => getState.map(_.definitions.get(name.value).exists(_ == name)).flatMap { case true => State.pure(false) - case false => report(name, "This arrow was already defined in the scope").as(false) + case false => report.error(name, "This arrow was already defined in the scope").as(false) } case None => @@ -149,7 +150,8 @@ class NamesInterpreter[S[_], X](implicit ) .as(true) else - report(name, "Cannot define a variable in the root scope") + report + .error(name, "Cannot define a variable in the root scope") .as(false) )(fr => (fr.addArrow(name, arrowType) -> true).pure) }.flatTap(_ => locations.addToken(name.value, name)) diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala index 8a3db353b..2af68e187 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala @@ -3,8 +3,8 @@ package aqua.semantics.rules.report import aqua.parser.lexer.Token trait ReportAlgebra[S[_], Alg[_]] { - def report(token: Token[S], hints: List[String]): Alg[Unit] + def error(token: Token[S], hints: List[String]): Alg[Unit] - def report(token: Token[S], hint: String): Alg[Unit] = - report(token, hint :: Nil) + def error(token: Token[S], hint: String): Alg[Unit] = + error(token, hint :: Nil) } diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala deleted file mode 100644 index f5ffdfcaa..000000000 --- a/semantics/src/main/scala/aqua/semantics/rules/report/ReportErrors.scala +++ /dev/null @@ -1,12 +0,0 @@ -package aqua.semantics.rules.report - -import aqua.parser.lexer.Token - -import cats.data.State - -trait ReportErrors[S[_], X] extends ReportAlgebra[S, State[X, *]] { - def apply(st: X, token: Token[S], hints: List[String]): X - - def report(token: Token[S], hints: List[String]): State[X, Unit] = - State.modify(apply(_, token, hints)) -} diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala new file mode 100644 index 000000000..d2b6b23dd --- /dev/null +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala @@ -0,0 +1,18 @@ +package aqua.semantics.rules.report + +import aqua.parser.lexer.Token + +import cats.data.State +import monocle.Lens + +class ReportInterpreter[S[_], X](using + lens: Lens[X, ReportState[S]] +) extends ReportAlgebra[S, State[X, *]] { + + override def error(token: Token[S], hints: List[String]): State[X, Unit] = + State.modify( + lens.modify( + _.reportError(token, hints) + ) + ) +} diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala new file mode 100644 index 000000000..751124f51 --- /dev/null +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala @@ -0,0 +1,25 @@ +package aqua.semantics.rules.report + +import aqua.semantics.{RulesViolated, SemanticError} +import aqua.parser.lexer.Token + +import cats.data.Chain +import cats.kernel.Monoid + +final case class ReportState[S[_]]( + errors: Chain[SemanticError[S]] = Chain.empty[SemanticError[S]] +) { + + def reportError(token: Token[S], hints: List[String]): ReportState[S] = + copy(errors = errors.append(RulesViolated(token, hints))) +} + +object ReportState { + + given [S[_]]: Monoid[ReportState[S]] with { + override val empty: ReportState[S] = ReportState() + + override def combine(x: ReportState[S], y: ReportState[S]): ReportState[S] = + ReportState(errors = x.errors ++ y.errors) + } +} diff --git a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala index c66a53406..f5c78b764 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala @@ -12,7 +12,7 @@ import aqua.raw.value.{ } import aqua.semantics.rules.locations.LocationsAlgebra import aqua.semantics.rules.StackInterpreter -import aqua.semantics.rules.report.ReportErrors +import aqua.semantics.rules.report.ReportAlgebra import aqua.semantics.rules.types.TypesStateHelper.{TypeResolution, TypeResolutionError} import aqua.types.* @@ -31,9 +31,9 @@ import monocle.macros.GenLens import scala.collection.immutable.SortedMap -class TypesInterpreter[S[_], X](implicit +class TypesInterpreter[S[_], X](using lens: Lens[X, TypesState[S]], - error: ReportErrors[S, X], + report: ReportAlgebra[S, State[X, *]], locations: LocationsAlgebra[S, State[X, *]] ) extends TypesAlgebra[S, State[X, *]] { @@ -55,7 +55,7 @@ class TypesInterpreter[S[_], X](implicit locations.pointLocations(tokensLocs).as(typ.some) case None => // TODO: Give more specific error message - report(token, s"Unresolved type").as(None) + report.error(token, s"Unresolved type").as(None) } override def resolveArrowDef(arrowDef: ArrowTypeToken[S]): State[X, Option[ArrowType]] = @@ -65,7 +65,7 @@ class TypesInterpreter[S[_], X](implicit locations.pointLocations(tokensLocs).as(tt.some) case Invalid(errs) => errs.traverse_ { case TypeResolutionError(token, hint) => - report(token, hint) + report.error(token, hint) }.as(none) } @@ -74,9 +74,9 @@ class TypesInterpreter[S[_], X](implicit case Some(serviceType: ServiceType) => serviceType.some.pure case Some(t) => - report(name, s"Type `$t` is not a service").as(none) + report.error(name, s"Type `$t` is not a service").as(none) case None => - report(name, s"Type `${name.value}` is not defined").as(none) + report.error(name, s"Type `${name.value}` is not defined").as(none) } override def defineAbilityType( @@ -87,10 +87,11 @@ class TypesInterpreter[S[_], X](implicit val types = fields.view.mapValues { case (_, t) => t }.toMap NonEmptyMap .fromMap(SortedMap.from(types)) - .fold(report(name, s"Ability `${name.value}` has no fields").as(none))(nonEmptyFields => - val `type` = AbilityType(name.value, nonEmptyFields) + .fold(report.error(name, s"Ability `${name.value}` has no fields").as(none))( + nonEmptyFields => + val `type` = AbilityType(name.value, nonEmptyFields) - modify(_.defineType(name, `type`)).as(`type`.some) + modify(_.defineType(name, `type`)).as(`type`.some) ) } @@ -104,20 +105,22 @@ class TypesInterpreter[S[_], X](implicit OptionT .when(t.codomain.length <= 1)(field -> t) .flatTapNone( - report(fieldName, "Service functions cannot have multiple results") + report.error(fieldName, "Service functions cannot have multiple results") ) case (field, (fieldName, t)) => OptionT( - report( - fieldName, - s"Field '$field' has unacceptable for service field type '$t'" - ).as(none) + report + .error( + fieldName, + s"Field '$field' has unacceptable for service field type '$t'" + ) + .as(none) ) }.flatMapF(arrows => NonEmptyMap .fromMap(SortedMap.from(arrows)) .fold( - report(name, s"Service `${name.value}` has no fields").as(none) + report.error(name, s"Service `${name.value}` has no fields").as(none) )(_.some.pure) ).semiflatMap(nonEmptyArrows => val `type` = ServiceType(name.value, nonEmptyArrows) @@ -134,20 +137,23 @@ class TypesInterpreter[S[_], X](implicit fields.toList.traverse { case (field, (fieldName, t: DataType)) => t match { - case _: StreamType => report(fieldName, s"Field '$field' has stream type").as(none) + case _: StreamType => + report.error(fieldName, s"Field '$field' has stream type").as(none) case _ => (field -> t).some.pure[ST] } case (field, (fieldName, t)) => - report( - fieldName, - s"Field '$field' has unacceptable for struct field type '$t'" - ).as(none) + report + .error( + fieldName, + s"Field '$field' has unacceptable for struct field type '$t'" + ) + .as(none) }.map(_.sequence.map(_.toMap)) .flatMap( _.map(SortedMap.from) .flatMap(NonEmptyMap.fromMap) .fold( - report(name, s"Struct `${name.value}` has no fields").as(none) + report.error(name, s"Struct `${name.value}` has no fields").as(none) )(nonEmptyFields => val `type` = StructType(name.value, nonEmptyFields) @@ -159,7 +165,7 @@ class TypesInterpreter[S[_], X](implicit override def defineAlias(name: NamedTypeToken[S], target: Type): State[X, Boolean] = getState.map(_.definitions.get(name.value)).flatMap { case Some(n) if n == name => State.pure(false) - case Some(_) => report(name, s"Type `${name.value}` was already defined").as(false) + case Some(_) => report.error(name, s"Type `${name.value}` was already defined").as(false) case None => modify(_.defineType(name, target)) .productL(locations.addToken(name.value, name)) @@ -171,10 +177,12 @@ class TypesInterpreter[S[_], X](implicit case nt: NamedType => nt.fields(op.value) .fold( - report( - op, - s"Field `${op.value}` not found in type `${nt.name}`, available: ${nt.fields.toNel.toList.map(_._1).mkString(", ")}" - ).as(None) + report + .error( + op, + s"Field `${op.value}` not found in type `${nt.name}`, available: ${nt.fields.toNel.toList.map(_._1).mkString(", ")}" + ) + .as(None) ) { t => locations.pointFieldLocation(nt.name, op.value, op).as(Some(IntoFieldRaw(op.value, t))) } @@ -182,10 +190,12 @@ class TypesInterpreter[S[_], X](implicit t.properties .get(op.value) .fold( - report( - op, - s"Expected data type to resolve a field '${op.value}' or a type with this property. Got: $rootT" - ).as(None) + report + .error( + op, + s"Expected data type to resolve a field '${op.value}' or a type with this property. Got: $rootT" + ) + .as(None) )(t => State.pure(Some(FunctorRaw(op.value, t)))) } @@ -199,10 +209,12 @@ class TypesInterpreter[S[_], X](implicit rootT match { case AbilityType(name, fieldsAndArrows) => fieldsAndArrows(op.name.value).fold( - report( - op, - s"Arrow `${op.name.value}` not found in type `$name`, available: ${fieldsAndArrows.toNel.toList.map(_._1).mkString(", ")}" - ).as(None) + report + .error( + op, + s"Arrow `${op.name.value}` not found in type `$name`, available: ${fieldsAndArrows.toNel.toList.map(_._1).mkString(", ")}" + ) + .as(None) ) { t => val resolvedType = t match { // TODO: is it a correct way to resolve `IntoArrow` type? @@ -217,10 +229,12 @@ class TypesInterpreter[S[_], X](implicit t.properties .get(op.name.value) .fold( - report( - op, - s"Expected scope type to resolve an arrow '${op.name.value}' or a type with this property. Got: $rootT" - ).as(None) + report + .error( + op, + s"Expected scope type to resolve an arrow '${op.name.value}' or a type with this property. Got: $rootT" + ) + .as(None) )(t => State.pure(Some(FunctorRaw(op.name.value, t)))) } @@ -238,12 +252,12 @@ class TypesInterpreter[S[_], X](implicit st.fields.lookup(fieldName) match { case Some(t) => ensureTypeMatches(op.fields.lookup(fieldName).getOrElse(op), t, value.`type`) - case None => report(op, s"No field with name '$fieldName' in $rootT").as(false) + case None => report.error(op, s"No field with name '$fieldName' in $rootT").as(false) } }.map(res => if (res.forall(identity)) Some(IntoCopyRaw(st, fields)) else None) case _ => - report(op, s"Expected $rootT to be a data type").as(None) + report.error(op, s"Expected $rootT to be a data type").as(None) } // TODO actually it's stateless, exists there just for reporting needs @@ -253,17 +267,17 @@ class TypesInterpreter[S[_], X](implicit idx: ValueRaw ): State[X, Option[PropertyRaw]] = if (!ScalarType.i64.acceptsValueOf(idx.`type`)) - report(op, s"Expected numeric index, got $idx").as(None) + report.error(op, s"Expected numeric index, got $idx").as(None) else rootT match { case ot: OptionType => op.idx.fold( State.pure(Some(IntoIndexRaw(idx, ot.element))) - )(v => report(v, s"Options might have only one element, use ! to get it").as(None)) + )(v => report.error(v, s"Options might have only one element, use ! to get it").as(None)) case rt: BoxType => State.pure(Some(IntoIndexRaw(idx, rt.element))) case _ => - report(op, s"Expected $rootT to be a collection type").as(None) + report.error(op, s"Expected $rootT to be a collection type").as(None) } override def ensureValuesComparable( @@ -299,7 +313,7 @@ class TypesInterpreter[S[_], X](implicit } if (isComparable(left, right)) State.pure(true) - else report(token, s"Cannot compare '$left' with '$right''").as(false) + else report.error(token, s"Cannot compare '$left' with '$right''").as(false) } override def ensureTypeMatches( @@ -315,10 +329,12 @@ class TypesInterpreter[S[_], X](implicit val typeFields = typeNamedType.fields // value can have more fields if (valueFields.length < typeFields.length) { - report( - token, - s"Number of fields doesn't match the data type, expected: $expected, given: $givenType" - ).as(false) + report + .error( + token, + s"Number of fields doesn't match the data type, expected: $expected, given: $givenType" + ) + .as(false) } else { valueFields.toSortedMap.toList.traverse { (name, `type`) => typeFields.lookup(name) match { @@ -333,10 +349,12 @@ class TypesInterpreter[S[_], X](implicit } ensureTypeMatches(nextToken, `type`, t) case None => - report( - token, - s"Wrong value type, expected: $expected, given: $givenType" - ).as(false) + report + .error( + token, + s"Wrong value type, expected: $expected, given: $givenType" + ) + .as(false) } }.map(_.forall(identity)) } @@ -349,10 +367,11 @@ class TypesInterpreter[S[_], X](implicit "You can extract value with `!`, but be aware it may trigger join behaviour." :: Nil else Nil - reportError( - token, - "Types mismatch." :: s"expected: $expected" :: s"given: $givenType" :: Nil ++ notes - ) + report + .error( + token, + "Types mismatch." :: s"expected: $expected" :: s"given: $givenType" :: Nil ++ notes + ) .as(false) } } @@ -361,10 +380,12 @@ class TypesInterpreter[S[_], X](implicit givenType match { case _: DataType => true.pure case _ => - report( - token, - s"Value of type '$givenType' could not be put into a collection" - ).as(false) + report + .error( + token, + s"Value of type '$givenType' could not be put into a collection" + ) + .as(false) } override def ensureTypeOneOf[T <: Type]( @@ -374,19 +395,23 @@ class TypesInterpreter[S[_], X](implicit ): State[X, Option[Type]] = expected .find(_ acceptsValueOf givenType) .fold( - reportError( - token, - "Types mismatch." :: - s"expected one of: ${expected.mkString(", ")}" :: - s"given: $givenType" :: Nil - ).as(none) + report + .error( + token, + "Types mismatch." :: + s"expected one of: ${expected.mkString(", ")}" :: + s"given: $givenType" :: Nil + ) + .as(none) )(_.some.pure) override def expectNoExport(token: Token[S]): State[X, Unit] = - report( - token, - "Types mismatch. Cannot assign to a variable the result of a call that returns nothing" - ).as(()) + report + .error( + token, + "Types mismatch. Cannot assign to a variable the result of a call that returns nothing" + ) + .as(()) override def checkArgumentsNumber( token: Token[S], @@ -395,10 +420,12 @@ class TypesInterpreter[S[_], X](implicit ): State[X, Boolean] = if (expected == givenNum) State.pure(true) else - report( - token, - s"Number of arguments doesn't match the function type, expected: ${expected}, given: $givenNum" - ).as(false) + report + .error( + token, + s"Number of arguments doesn't match the function type, expected: ${expected}, given: $givenNum" + ) + .as(false) override def beginArrowScope(token: ArrowTypeToken[S]): State[X, ArrowType] = Applicative[ST] @@ -434,37 +461,49 @@ class TypesInterpreter[S[_], X](implicit values: NonEmptyList[(ValueToken[S], ValueRaw)] ): State[X, Boolean] = mapStackHeadM[Boolean]( - report(values.head._1, "Fatal: checkArrowReturn has no matching beginArrowScope").as(false) + report + .error(values.head._1, "Fatal: checkArrowReturn has no matching beginArrowScope") + .as(false) )(frame => if (frame.retVals.nonEmpty) - report( - values.head._1, - "Return expression was already used in scope; you can use only one Return in an arrow declaration, use conditional return pattern if you need to return based on condition" - ).as(frame -> false) + report + .error( + values.head._1, + "Return expression was already used in scope; you can use only one Return in an arrow declaration, use conditional return pattern if you need to return based on condition" + ) + .as(frame -> false) else if (frame.token.res.isEmpty) - report( - values.head._1, - "No return type declared for this arrow, please remove `<- ...` expression or add `-> ...` return type(s) declaration to the arrow" - ).as(frame -> false) + report + .error( + values.head._1, + "No return type declared for this arrow, please remove `<- ...` expression or add `-> ...` return type(s) declaration to the arrow" + ) + .as(frame -> false) else if (frame.token.res.length > values.length) - report( - values.last._1, - s"Expected ${frame.token.res.length - values.length} more values to be returned, see return type declaration" - ).as(frame -> false) + report + .error( + values.last._1, + s"Expected ${frame.token.res.length - values.length} more values to be returned, see return type declaration" + ) + .as(frame -> false) else if (frame.token.res.length < values.length) - report( - values.toList.drop(frame.token.res.length).headOption.getOrElse(values.last)._1, - s"Too many values are returned from this arrow, this one is unexpected. Defined return type: ${frame.arrowType.codomain}" - ).as(frame -> false) + report + .error( + values.toList.drop(frame.token.res.length).headOption.getOrElse(values.last)._1, + s"Too many values are returned from this arrow, this one is unexpected. Defined return type: ${frame.arrowType.codomain}" + ) + .as(frame -> false) else frame.arrowType.codomain.toList .zip(values.toList) .traverse { case (returnType, (token, returnValue)) => if (!returnType.acceptsValueOf(returnValue.`type`)) - report( - token, - s"Wrong value type, expected: $returnType, given: ${returnValue.`type`}" - ).as(none) + report + .error( + token, + s"Wrong value type, expected: $returnType, given: ${returnValue.`type`}" + ) + .as(none) else returnValue.some.pure[SX] } .map(_.sequence) @@ -473,14 +512,16 @@ class TypesInterpreter[S[_], X](implicit override def endArrowScope(token: Token[S]): State[X, List[ValueRaw]] = mapStackHeadM( - report(token, "Fatal: endArrowScope has no matching beginArrowScope").as(Nil) + report.error(token, "Fatal: endArrowScope has no matching beginArrowScope").as(Nil) )(frame => if (frame.token.res.isEmpty) (frame -> Nil).pure else if (frame.retVals.isEmpty) - report( - frame.token.res.headOption.getOrElse(frame.token), - "Return type is defined for the arrow, but nothing returned. Use `<- value, ...` as the last expression inside function body." - ).as(frame -> Nil) + report + .error( + frame.token.res.headOption.getOrElse(frame.token), + "Return type is defined for the arrow, but nothing returned. Use `<- value, ...` as the last expression inside function body." + ) + .as(frame -> Nil) else (frame -> frame.retVals.getOrElse(Nil)).pure ) <* stack.endScope @@ -495,10 +536,12 @@ class TypesInterpreter[S[_], X](implicit .flatMap { case Some(_) => // TODO: Point to both locations here - report( - token, - s"Name `${name}` was already defined here" - ).as(ifDefined) + report + .error( + token, + s"Name `${name}` was already defined here" + ) + .as(ifDefined) case None => ifNotDefined } } diff --git a/semantics/src/test/scala/aqua/semantics/Utils.scala b/semantics/src/test/scala/aqua/semantics/Utils.scala index 17e8b3cf3..b00280be8 100644 --- a/semantics/src/test/scala/aqua/semantics/Utils.scala +++ b/semantics/src/test/scala/aqua/semantics/Utils.scala @@ -5,11 +5,12 @@ import aqua.parser.lexer.{Name, Token} import aqua.parser.lift.Span import aqua.raw.{Raw, RawContext} import aqua.semantics.expr.func.ClosureSem -import aqua.semantics.rules.report.ReportErrors import aqua.semantics.rules.abilities.{AbilitiesAlgebra, AbilitiesInterpreter, AbilitiesState} import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgebra} import aqua.semantics.rules.names.{NamesAlgebra, NamesInterpreter, NamesState} import aqua.semantics.rules.types.{TypesAlgebra, TypesInterpreter, TypesState} +import aqua.semantics.rules.mangler.{ManglerAlgebra, ManglerInterpreter} +import aqua.semantics.rules.report.{ReportAlgebra, ReportInterpreter} import aqua.types.* import cats.data.State @@ -17,11 +18,12 @@ import cats.{~>, Id} import monocle.Lens import monocle.macros.GenLens import monocle.syntax.all.* -import aqua.semantics.rules.mangler.ManglerAlgebra -import aqua.semantics.rules.mangler.ManglerInterpreter object Utils { + given ReportAlgebra[Id, State[CompilerState[Id], *]] = + new ReportInterpreter[Id, CompilerState[Id]] + given ManglerAlgebra[State[CompilerState[Id], *]] = new ManglerInterpreter[CompilerState[Id]] diff --git a/semantics/src/test/scala/aqua/semantics/ValuesAlgebraSpec.scala b/semantics/src/test/scala/aqua/semantics/ValuesAlgebraSpec.scala index d0312c696..a8a4bd9f4 100644 --- a/semantics/src/test/scala/aqua/semantics/ValuesAlgebraSpec.scala +++ b/semantics/src/test/scala/aqua/semantics/ValuesAlgebraSpec.scala @@ -7,6 +7,7 @@ import aqua.semantics.rules.definitions.{DefinitionsAlgebra, DefinitionsInterpre import aqua.semantics.rules.types.{TypesAlgebra, TypesInterpreter, TypesState} import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgebra} import aqua.semantics.rules.mangler.{ManglerAlgebra, ManglerInterpreter} +import aqua.semantics.rules.report.{ReportAlgebra, ReportInterpreter} import aqua.raw.value.{ApplyBinaryOpRaw, LiteralRaw} import aqua.raw.RawContext import aqua.types.* @@ -32,9 +33,10 @@ class ValuesAlgebraSpec extends AnyFlatSpec with Matchers with Inside { def algebra() = { type Interpreter[A] = State[TestState, A] + given ReportAlgebra[Id, Interpreter] = + new ReportInterpreter[Id, CompilerState[Id]] given LocationsAlgebra[Id, Interpreter] = new DummyLocationsInterpreter[Id, CompilerState[Id]] - given ManglerAlgebra[Interpreter] = new ManglerInterpreter[CompilerState[Id]] given TypesAlgebra[Id, Interpreter] = From 9673fc59072176af49e858a68bb7587006e4341d Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 10:05:12 +0000 Subject: [PATCH 03/31] Refactor --- api/api/.js/src/main/scala/api/AquaAPI.scala | 16 ++++++++-------- .../src/main/scala/aqua/api/APICompilation.scala | 9 +++++---- .../src/main/scala/aqua/run/FuncCompiler.scala | 2 +- io/src/main/scala/aqua/ErrorRendering.scala | 2 +- .../src/main/scala/aqua/lsp/LspSemantics.scala | 3 ++- .../main/scala/aqua/semantics/Semantics.scala | 4 +++- 6 files changed, 20 insertions(+), 16 deletions(-) diff --git a/api/api/.js/src/main/scala/api/AquaAPI.scala b/api/api/.js/src/main/scala/api/AquaAPI.scala index 7c4539d09..d77f317a8 100644 --- a/api/api/.js/src/main/scala/api/AquaAPI.scala +++ b/api/api/.js/src/main/scala/api/AquaAPI.scala @@ -1,7 +1,7 @@ package api import api.types.{AquaConfig, AquaFunction, CompilationResult, GeneratedSource, Input} -import aqua.ErrorRendering.showError +import aqua.ErrorRendering.given import aqua.raw.value.ValueRaw import aqua.api.{APICompilation, AquaAPIConfig} import aqua.api.TargetType.* @@ -18,13 +18,19 @@ import aqua.parser.lexer.{LiteralToken, Token} import aqua.parser.lift.FileSpan.F import aqua.parser.lift.{FileSpan, Span} import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} -import aqua.semantics.{CompilerState, HeaderError, RulesViolated, WrongAST} import aqua.{AquaIO, SpanParser} import aqua.model.transform.{Transform, TransformConfig} import aqua.backend.api.APIBackend import aqua.backend.js.JavaScriptBackend import aqua.backend.ts.TypeScriptBackend import aqua.definitions.FunctionDef +import aqua.js.{FunctionDefJs, ServiceDefJs, VarJson} +import aqua.model.AquaContext +import aqua.raw.ops.CallArrowRawTag +import aqua.raw.value.{LiteralRaw, VarRaw} +import aqua.res.AquaRes + +import cats.Applicative import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} import cats.data.Validated.{invalidNec, validNec, Invalid, Valid} import cats.syntax.applicative.* @@ -44,12 +50,6 @@ import scala.scalajs.js.{|, undefined, Promise, UndefOr} import scala.scalajs.js import scala.scalajs.js.JSConverters.* import scala.scalajs.js.annotation.* -import aqua.js.{FunctionDefJs, ServiceDefJs, VarJson} -import aqua.model.AquaContext -import aqua.raw.ops.CallArrowRawTag -import aqua.raw.value.{LiteralRaw, VarRaw} -import aqua.res.AquaRes -import cats.Applicative @JSExportTopLevel("Aqua") object AquaAPI extends App with Logging { diff --git a/api/api/src/main/scala/aqua/api/APICompilation.scala b/api/api/src/main/scala/aqua/api/APICompilation.scala index da3aedc83..fae2e7e16 100644 --- a/api/api/src/main/scala/aqua/api/APICompilation.scala +++ b/api/api/src/main/scala/aqua/api/APICompilation.scala @@ -1,6 +1,6 @@ package aqua.api -import aqua.ErrorRendering.showError +import aqua.ErrorRendering.given import aqua.raw.value.ValueRaw import aqua.api.AquaAPIConfig import aqua.backend.{AirFunction, Backend, Generated} @@ -15,16 +15,16 @@ import aqua.parser.lexer.{LiteralToken, Token} import aqua.parser.lift.FileSpan.F import aqua.parser.lift.{FileSpan, Span} import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} -import aqua.semantics.{CompilerState, HeaderError, RulesViolated, WrongAST} import aqua.{AquaIO, SpanParser} import aqua.model.transform.{Transform, TransformConfig} import aqua.backend.api.APIBackend import aqua.definitions.FunctionDef import aqua.model.AquaContext import aqua.res.AquaRes + import cats.Applicative import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} -import cats.data.Validated.{Invalid, Valid, invalid, invalidNec, validNec} +import cats.data.Validated.{invalid, invalidNec, validNec, Invalid, Valid} import cats.syntax.applicative.* import cats.syntax.apply.* import cats.syntax.flatMap.* @@ -158,7 +158,8 @@ object APICompilation { override def generate(aqua: AquaRes): Seq[Generated] = backend.generate(aqua) , config - ).map(_.leftMap(_.map(_.show).distinct)) + ) + .map(_.leftMap(_.map(_.show).distinct)) }.map(_.leftMap(NonEmptyChain.fromNonEmptyList).andThen(identity)) } } diff --git a/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala b/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala index 5bd3747ec..cfbe8be2f 100644 --- a/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala +++ b/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala @@ -1,6 +1,6 @@ package aqua.run -import aqua.ErrorRendering.showError +import aqua.ErrorRendering.given import aqua.compiler.{AquaCompiler, AquaCompilerConf, CompilerAPI} import aqua.files.{AquaFileSources, FileModuleId} import aqua.{AquaIO, SpanParser} diff --git a/io/src/main/scala/aqua/ErrorRendering.scala b/io/src/main/scala/aqua/ErrorRendering.scala index 6950c177e..9a36b2642 100644 --- a/io/src/main/scala/aqua/ErrorRendering.scala +++ b/io/src/main/scala/aqua/ErrorRendering.scala @@ -29,7 +29,7 @@ object ErrorRendering { ) ) + Console.RESET + "\n" - implicit val showError: Show[AquaError[FileModuleId, AquaFileError, FileSpan.F]] = Show.show { + given Show[AquaError[FileModuleId, AquaFileError, FileSpan.F]] = Show.show { case ParserErr(err) => err match { case BlockIndentError(indent, message) => diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala index eb1c42845..d79e00e7a 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala @@ -4,7 +4,8 @@ import aqua.parser.Ast import aqua.parser.head.{ImportExpr, ImportFromExpr, UseExpr, UseFromExpr} import aqua.parser.lexer.{LiteralToken, Token} import aqua.semantics.rules.locations.LocationsState -import aqua.semantics.{CompilerState, RawSemantics, RulesViolated, SemanticError, Semantics} +import aqua.semantics.{CompilerState, RawSemantics, SemanticError, Semantics} + import cats.data.Validated.{Invalid, Valid} import cats.syntax.applicative.* import cats.syntax.apply.* diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index 61fc85db7..fe32294b9 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -47,7 +47,9 @@ trait Semantics[S[_], C] { ): ValidatedNec[SemanticError[S], C] } -class RawSemantics[S[_]](implicit p: Picker[RawContext]) extends Semantics[S, RawContext] { +class RawSemantics[S[_]](using + Picker[RawContext] +) extends Semantics[S, RawContext] { def process( ast: Ast[S], From d2ebae7fb8d7063365c28f91c48d42f216e01b13 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 10:35:01 +0000 Subject: [PATCH 04/31] Refactor AquaError --- .../main/scala/aqua/api/APICompilation.scala | 4 +- .../scala/aqua/compiler/AquaCompiler.scala | 3 +- .../main/scala/aqua/compiler/AquaError.scala | 26 ++++++------ .../main/scala/aqua/compiler/AquaParser.scala | 15 +++---- .../scala/aqua/compiler/CompilerAPI.scala | 6 ++- io/src/main/scala/aqua/ErrorRendering.scala | 10 +++-- .../.js/src/main/scala/aqua/lsp/AquaLSP.scala | 42 +++++++++---------- 7 files changed, 56 insertions(+), 50 deletions(-) diff --git a/api/api/src/main/scala/aqua/api/APICompilation.scala b/api/api/src/main/scala/aqua/api/APICompilation.scala index fae2e7e16..d0744b646 100644 --- a/api/api/src/main/scala/aqua/api/APICompilation.scala +++ b/api/api/src/main/scala/aqua/api/APICompilation.scala @@ -45,7 +45,8 @@ object APICompilation { aquaConfig: AquaAPIConfig, fillWithTypes: List[ValueRaw] => ValidatedNec[String, List[ValueRaw]] ): IO[ValidatedNec[String, (FunctionDef, String)]] = { - implicit val aio: AquaIO[IO] = new AquaFilesIO[IO] + given AquaIO[IO] = new AquaFilesIO[IO] + ( LogLevels.levelFromString(aquaConfig.logLevel), Constants.parse(aquaConfig.constants) @@ -80,7 +81,6 @@ object APICompilation { } } }.leftMap(_.map(_.show).distinct) - } } match { case Valid(pr) => pr diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index 382d9dbf2..9babf6213 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -1,5 +1,6 @@ package aqua.compiler +import aqua.compiler.AquaError.{ParserError as AquaParserError, *} import aqua.backend.Backend import aqua.linker.{AquaModule, Linker, Modules} import aqua.model.AquaContext @@ -90,7 +91,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( .map { rc => NonEmptyMap.one(mod.id, rc) } } // The whole chain returns a semantics error finally - .leftMap(_.map[Err](CompileError(_))) + .leftMap(_.map[Err](CompileError.apply)) } ) .map( diff --git a/compiler/src/main/scala/aqua/compiler/AquaError.scala b/compiler/src/main/scala/aqua/compiler/AquaError.scala index bd15d0661..17c936a2b 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaError.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaError.scala @@ -1,20 +1,20 @@ package aqua.compiler -import aqua.parser.ParserError +import aqua.parser import aqua.parser.lexer.Token -import aqua.semantics.SemanticError -import cats.data.NonEmptyChain +import aqua.semantics -trait AquaError[I, E, S[_]] -case class SourcesErr[I, E, S[_]](err: E) extends AquaError[I, E, S] -case class ParserErr[I, E, S[_]](err: ParserError[S]) extends AquaError[I, E, S] +import cats.data.NonEmptyChain -case class ResolveImportsErr[I, E, S[_]](fromFile: I, token: Token[S], err: E) - extends AquaError[I, E, S] -case class ImportErr[I, E, S[_]](token: Token[S]) extends AquaError[I, E, S] +enum AquaError[I, E, S[_]] { + case SourcesError(err: E) + case ParserError(err: parser.ParserError[S]) -case class CycleError[I, E, S[_]](modules: NonEmptyChain[I]) extends AquaError[I, E, S] + case ResolveImportsError(fromFile: I, token: Token[S], err: E) + case ImportError(token: Token[S]) extends AquaError[I, E, S] + case CycleError(modules: NonEmptyChain[I]) -case class CompileError[I, E, S[_]](err: SemanticError[S]) extends AquaError[I, E, S] -case class OutputError[I, E, S[_]](compiled: AquaCompiled[I], err: E) extends AquaError[I, E, S] -case class AirValidationError[I, E, S[_]](errors: NonEmptyChain[String]) extends AquaError[I, E, S] + case CompileError(err: semantics.SemanticError[S]) + case OutputError(compiled: AquaCompiled[I], err: E) + case AirValidationError(errors: NonEmptyChain[String]) +} diff --git a/compiler/src/main/scala/aqua/compiler/AquaParser.scala b/compiler/src/main/scala/aqua/compiler/AquaParser.scala index 5940647ea..46c1bbb48 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaParser.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaParser.scala @@ -1,9 +1,11 @@ package aqua.compiler +import aqua.compiler.AquaError.{ParserError as AquaParserError, *} import aqua.linker.{AquaModule, Modules} import aqua.parser.head.{FilenameExpr, ImportExpr} import aqua.parser.lift.{LiftParser, Span} import aqua.parser.{Ast, ParserError} + import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} import cats.parse.Parser0 import cats.syntax.applicative.* @@ -31,10 +33,10 @@ class AquaParser[F[_], E, I, S[_]: Comonad]( // Parse all the source files def parseSources: F[ValidatedNec[Err, Chain[(I, Body)]]] = sources.sources.map( - _.leftMap(_.map[Err](SourcesErr(_))).andThen(_.map { case (i, s) => + _.leftMap(_.map(SourcesError.apply)).andThen(_.map { case (i, s) => parser(i)(s) .bimap( - _.map[Err](ParserErr(_)), + _.map(AquaParserError.apply), ast => Chain.one(i -> ast) ) }.foldA) @@ -50,9 +52,8 @@ class AquaParser[F[_], E, I, S[_]: Comonad]( .resolveImport(id, fe.fileValue) )( _.bimap( - _.map[Err](ResolveImportsErr(id, fe.filename, _)), - importId => - Chain.one[(I, (String, Err))](importId -> (fe.fileValue, ImportErr(fe.filename))) + _.map(ResolveImportsError(id, fe.filename, _) : Err), + importId => Chain.one(importId -> (fe.fileValue, ImportError(fe.filename): Err)) ) ) } @@ -88,8 +89,8 @@ class AquaParser[F[_], E, I, S[_]: Comonad]( def loadModule(imp: I): F[ValidatedNec[Err, AquaModule[I, Err, Body]]] = sources .load(imp) - .map(_.leftMap(_.map[Err](SourcesErr(_))).andThen { src => - parser(imp)(src).leftMap(_.map[Err](ParserErr(_))) + .map(_.leftMap(_.map(SourcesError.apply)).andThen { src => + parser(imp)(src).leftMap(_.map[Err](AquaParserError.apply)) }) .flatMap { case Validated.Valid(ast) => diff --git a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala index 979d25695..77f74c868 100644 --- a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala +++ b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala @@ -1,5 +1,6 @@ package aqua.compiler +import aqua.compiler.AquaError.* import aqua.backend.{AirFunction, Backend} import aqua.linker.{AquaModule, Linker, Modules} import aqua.model.AquaContext @@ -10,8 +11,9 @@ import aqua.raw.{RawContext, RawPart} import aqua.res.AquaRes import aqua.semantics.header.{HeaderHandler, HeaderSem} import aqua.semantics.{CompilerState, RawSemantics, Semantics} + import cats.data.* -import cats.data.Validated.{Invalid, Valid, invalid, validNec} +import cats.data.Validated.{invalid, validNec, Invalid, Valid} import cats.parse.Parser0 import cats.syntax.applicative.* import cats.syntax.flatMap.* @@ -20,7 +22,7 @@ import cats.syntax.functor.* import cats.syntax.monoid.* import cats.syntax.semigroup.* import cats.syntax.traverse.* -import cats.{Comonad, Monad, Monoid, Order, ~>} +import cats.{~>, Comonad, Monad, Monoid, Order} import scribe.Logging import scala.collection.MapView diff --git a/io/src/main/scala/aqua/ErrorRendering.scala b/io/src/main/scala/aqua/ErrorRendering.scala index 9a36b2642..59ac2fab4 100644 --- a/io/src/main/scala/aqua/ErrorRendering.scala +++ b/io/src/main/scala/aqua/ErrorRendering.scala @@ -1,11 +1,13 @@ package aqua +import aqua.compiler.AquaError.{ParserError as AquaParserError, *} import aqua.compiler.* import aqua.files.FileModuleId import aqua.io.AquaFileError import aqua.parser.lift.{FileSpan, Span} import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} import aqua.semantics.{HeaderError, RulesViolated, WrongAST} + import cats.parse.LocationMap import cats.parse.Parser.Expectation import cats.parse.Parser.Expectation.* @@ -30,7 +32,7 @@ object ErrorRendering { ) + Console.RESET + "\n" given Show[AquaError[FileModuleId, AquaFileError, FileSpan.F]] = Show.show { - case ParserErr(err) => + case AquaParserError(err) => err match { case BlockIndentError(indent, message) => showForConsole("Syntax error", indent._1, message :: Nil) @@ -63,15 +65,15 @@ object ErrorRendering { .reverse .mkString("\n") } - case SourcesErr(err) => + case SourcesError(err) => Console.RED + err.showForConsole + Console.RESET case AirValidationError(errors) => Console.RED + errors.toChain.toList.mkString("\n") + Console.RESET - case ResolveImportsErr(_, token, err) => + case ResolveImportsError(_, token, err) => val span = token.unit._1 showForConsole("Cannot resolve imports", span, err.showForConsole :: Nil) - case ImportErr(token) => + case ImportError(token) => val span = token.unit._1 showForConsole("Cannot resolve import", span, "Cannot resolve import" :: Nil) case CycleError(modules) => diff --git a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala index 4032f4799..b259a6901 100644 --- a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala +++ b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala @@ -1,6 +1,7 @@ package aqua.lsp import aqua.compiler.* +import aqua.compiler.AquaError.{ParserError as AquaParserError, *} import aqua.files.{AquaFileSources, AquaFilesIO, FileModuleId} import aqua.io.* import aqua.parser.lexer.{LiteralToken, Token} @@ -10,7 +11,7 @@ import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} import aqua.raw.ConstantRaw import aqua.semantics.{HeaderError, RulesViolated, WrongAST} import aqua.{AquaIO, SpanParser} -import cats.data.Validated.{Invalid, Valid, invalidNec, validNec} +import cats.data.Validated.{invalidNec, validNec, Invalid, Valid} import cats.data.{NonEmptyChain, Validated} import cats.effect.IO import cats.effect.unsafe.implicits.global @@ -22,7 +23,7 @@ import scala.concurrent.Future import scala.scalajs.js import scala.scalajs.js.JSConverters.* import scala.scalajs.js.annotation.* -import scala.scalajs.js.{UndefOr, undefined} +import scala.scalajs.js.{undefined, UndefOr} @JSExportAll case class CompilationResult( @@ -77,7 +78,7 @@ object AquaLSP extends App with Logging { def errorToInfo(error: AquaError[FileModuleId, AquaFileError, FileSpan.F]): List[ErrorInfo] = { error match { - case ParserErr(err) => + case AquaParserError(err) => err match { case BlockIndentError(indent, message) => ErrorInfo(indent._1, message) :: Nil @@ -98,11 +99,11 @@ object AquaLSP extends App with Logging { .map(_._2) .reverse } - case SourcesErr(err) => + case SourcesError(err) => ErrorInfo.applyOp(0, 0, err.showForConsole, None) :: Nil - case ResolveImportsErr(_, token, err) => + case ResolveImportsError(_, token, err) => ErrorInfo(token.unit._1, err.showForConsole) :: Nil - case ImportErr(token) => + case ImportError(token) => ErrorInfo(token.unit._1, "Cannot resolve import") :: Nil case CycleError(modules) => ErrorInfo.applyOp( @@ -156,7 +157,7 @@ object AquaLSP extends App with Logging { _.getOrElse( pathId, invalidNec( - SourcesErr(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) + SourcesError(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) ) ) ) @@ -165,7 +166,7 @@ object AquaLSP extends App with Logging { .map(l => validNec(l)) .getOrElse( invalidNec( - SourcesErr(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) + SourcesError(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) ) ) ) @@ -176,21 +177,20 @@ object AquaLSP extends App with Logging { locations: List[(Token[FileSpan.F], Token[FileSpan.F])] ): js.Array[TokenLink] = { locations.flatMap { case (from, to) => - - val fromOp = TokenLocation.fromSpan(from.unit._1) - val toOp = TokenLocation.fromSpan(to.unit._1) - - val link = for { - from <- fromOp - to <- toOp - } yield { - TokenLink(from, to) - } + val fromOp = TokenLocation.fromSpan(from.unit._1) + val toOp = TokenLocation.fromSpan(to.unit._1) + + val link = for { + from <- fromOp + to <- toOp + } yield { + TokenLink(from, to) + } - if (link.isEmpty) - logger.warn(s"Incorrect coordinates for token '${from.unit._1.name}'") + if (link.isEmpty) + logger.warn(s"Incorrect coordinates for token '${from.unit._1.name}'") - link.toList + link.toList }.toJSArray } From 7965d5527bac8bcc33856cb2718193f49fcc77be Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 10:45:14 +0000 Subject: [PATCH 05/31] Fixes --- compiler/src/main/scala/aqua/compiler/AquaCompiled.scala | 7 ++++++- compiler/src/main/scala/aqua/compiler/AquaError.scala | 4 ++-- compiler/src/main/scala/aqua/compiler/AquaParser.scala | 2 +- .../.js/src/main/scala/aqua/lsp/AquaLSP.scala | 2 +- 4 files changed, 10 insertions(+), 5 deletions(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiled.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiled.scala index 3c8b4d1cc..75f7af0f2 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiled.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiled.scala @@ -2,4 +2,9 @@ package aqua.compiler import aqua.backend.Generated -case class AquaCompiled[I](sourceId: I, compiled: Seq[Generated], funcsCount: Int, servicesCount: Int) +case class AquaCompiled[+I]( + sourceId: I, + compiled: Seq[Generated], + funcsCount: Int, + servicesCount: Int +) diff --git a/compiler/src/main/scala/aqua/compiler/AquaError.scala b/compiler/src/main/scala/aqua/compiler/AquaError.scala index 17c936a2b..3407b7ba6 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaError.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaError.scala @@ -6,12 +6,12 @@ import aqua.semantics import cats.data.NonEmptyChain -enum AquaError[I, E, S[_]] { +enum AquaError[+I, +E, S[_]] { case SourcesError(err: E) case ParserError(err: parser.ParserError[S]) case ResolveImportsError(fromFile: I, token: Token[S], err: E) - case ImportError(token: Token[S]) extends AquaError[I, E, S] + case ImportError(token: Token[S]) case CycleError(modules: NonEmptyChain[I]) case CompileError(err: semantics.SemanticError[S]) diff --git a/compiler/src/main/scala/aqua/compiler/AquaParser.scala b/compiler/src/main/scala/aqua/compiler/AquaParser.scala index 46c1bbb48..792a8e4f0 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaParser.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaParser.scala @@ -52,7 +52,7 @@ class AquaParser[F[_], E, I, S[_]: Comonad]( .resolveImport(id, fe.fileValue) )( _.bimap( - _.map(ResolveImportsError(id, fe.filename, _) : Err), + _.map(ResolveImportsError(id, fe.filename, _): Err), importId => Chain.one(importId -> (fe.fileValue, ImportError(fe.filename): Err)) ) ) diff --git a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala index b259a6901..969cca515 100644 --- a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala +++ b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala @@ -134,7 +134,7 @@ object AquaLSP extends App with Logging { ): scalajs.js.Promise[CompilationResult] = { logger.debug(s"Compiling '$pathStr' with imports: $imports") - implicit val aio: AquaIO[IO] = new AquaFilesIO[IO] + given AquaIO[IO] = new AquaFilesIO[IO] val path = Path(pathStr) val pathId = FileModuleId(path) From d1d5f60a545e9d56fc1c7af78c4d62462f7fd2e9 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 15:00:49 +0000 Subject: [PATCH 06/31] Add warnings, refactor --- .../main/scala/aqua/api/APICompilation.scala | 6 +- .../main/scala/aqua/compiler/AquaParser.scala | 81 +++++++++---------- parser/src/main/scala/aqua/parser/Ast.scala | 9 +++ .../scala/aqua/semantics/CompilerState.scala | 2 + .../aqua/semantics/SemanticWarning.scala | 8 ++ .../main/scala/aqua/semantics/Semantics.scala | 5 +- .../rules/report/ReportAlgebra.scala | 5 ++ .../rules/report/ReportInterpreter.scala | 7 ++ .../semantics/rules/report/ReportState.scala | 13 ++- 9 files changed, 87 insertions(+), 49 deletions(-) create mode 100644 semantics/src/main/scala/aqua/semantics/SemanticWarning.scala diff --git a/api/api/src/main/scala/aqua/api/APICompilation.scala b/api/api/src/main/scala/aqua/api/APICompilation.scala index d0744b646..36b6045ee 100644 --- a/api/api/src/main/scala/aqua/api/APICompilation.scala +++ b/api/api/src/main/scala/aqua/api/APICompilation.scala @@ -94,7 +94,8 @@ object APICompilation { aquaConfig: AquaAPIConfig, backend: Backend ): IO[ValidatedNec[String, Chain[AquaCompiled[FileModuleId]]]] = { - implicit val aio: AquaIO[IO] = new AquaFilesIO[IO] + given AquaIO[IO] = new AquaFilesIO[IO] + val path = Path(pathStr) val sources = new AquaFileSources[IO](path, imports.map(Path.apply)) compileRaw( @@ -110,7 +111,8 @@ object APICompilation { aquaConfig: AquaAPIConfig, backend: Backend ): IO[ValidatedNec[String, Chain[AquaCompiled[FileModuleId]]]] = { - implicit val aio: AquaIO[IO] = new AquaFilesIO[IO] + given AquaIO[IO] = new AquaFilesIO[IO] + val path = Path("") val strSources: AquaFileSources[IO] = diff --git a/compiler/src/main/scala/aqua/compiler/AquaParser.scala b/compiler/src/main/scala/aqua/compiler/AquaParser.scala index 792a8e4f0..2c218c8a1 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaParser.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaParser.scala @@ -8,11 +8,13 @@ import aqua.parser.{Ast, ParserError} import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} import cats.parse.Parser0 +import cats.syntax.either.* import cats.syntax.applicative.* import cats.syntax.flatMap.* import cats.syntax.functor.* import cats.syntax.monad.* import cats.syntax.foldable.* +import cats.syntax.traverse.* import cats.syntax.validated.* import cats.data.Chain.* import cats.data.Validated.* @@ -21,72 +23,67 @@ import cats.{~>, Comonad, Monad} import scribe.Logging // TODO: add tests -class AquaParser[F[_], E, I, S[_]: Comonad]( +class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( sources: AquaSources[F, E, I], parser: I => String => ValidatedNec[ParserError[S], Ast[S]] -)(implicit F: Monad[F]) - extends Logging { +) extends Logging { type Body = Ast[S] type Err = AquaError[I, E, S] // Parse all the source files - def parseSources: F[ValidatedNec[Err, Chain[(I, Body)]]] = + private def parseSources: F[ValidatedNec[Err, Chain[(I, Body)]]] = sources.sources.map( _.leftMap(_.map(SourcesError.apply)).andThen(_.map { case (i, s) => - parser(i)(s) - .bimap( - _.map(AquaParserError.apply), - ast => Chain.one(i -> ast) - ) + parser(i)(s).bimap( + _.map(AquaParserError.apply), + ast => Chain.one(i -> ast) + ) }.foldA) ) // Resolve imports (not parse, just resolve) of the given file - def resolveImports(id: I, ast: Body): F[ValidatedNec[Err, AquaModule[I, Err, Body]]] = - ast.head.tailForced - .map(_.head) - .collect { case fe: FilenameExpr[F] => - F.map( - sources - .resolveImport(id, fe.fileValue) - )( + private def resolveImports(id: I, ast: Body): F[ValidatedNec[Err, AquaModule[I, Err, Body]]] = + ast.collectHead { case fe: FilenameExpr[S] => + fe.fileValue -> fe.token + }.value.traverse { case (filename, token) => + sources + .resolveImport(id, filename) + .map( _.bimap( - _.map(ResolveImportsError(id, fe.filename, _): Err), - importId => Chain.one(importId -> (fe.fileValue, ImportError(fe.filename): Err)) + _.map(ResolveImportsError(id, token, _): Err), + importId => Chain.one(importId -> (filename, ImportError(token): Err)) ) ) + }.map( + _.foldA.map { collected => + AquaModule[I, Err, Body]( + id, + // How filenames correspond to the resolved IDs + collected.map { case (i, (fn, _)) => + fn -> i + }.toList.toMap[String, I], + // Resolved IDs to errors that point to the import in source code + collected.map { case (i, (_, err)) => + i -> err + }.toList.toMap[I, Err], + ast + ) } - .sequence - .map( - _.foldA.map { collected => - AquaModule[I, Err, Body]( - id, - // How filenames correspond to the resolved IDs - collected.map { case (i, (fn, _)) => - fn -> i - }.toList.toMap[String, I], - // Resolved IDs to errors that point to the import in source code - collected.map { case (i, (_, err)) => - i -> err - }.toList.toMap[I, Err], - ast - ) - } - ) + ) // Parse sources, convert to modules - def sourceModules: F[ValidatedNec[Err, Modules[I, Err, Body]]] = + private def sourceModules: F[ValidatedNec[Err, Modules[I, Err, Body]]] = parseSources.flatMap { case Validated.Valid(srcs) => srcs.traverse { case (id, ast) => - resolveImports(id, ast).map(_.map(Chain.one)) - }.map(_.foldA) + resolveImports(id, ast) + }.map(_.sequence) case Validated.Invalid(errs) => errs.invalid.pure[F] }.map(_.map(_.foldLeft(Modules[I, Err, Body]())(_.add(_, toExport = true)))) - def loadModule(imp: I): F[ValidatedNec[Err, AquaModule[I, Err, Body]]] = + private def loadModule(imp: I): F[ValidatedNec[Err, AquaModule[I, Err, Body]]] = sources .load(imp) .map(_.leftMap(_.map(SourcesError.apply)).andThen { src => @@ -99,7 +96,7 @@ class AquaParser[F[_], E, I, S[_]: Comonad]( errs.invalid.pure[F] } - def resolveModules( + private def resolveModules( modules: Modules[I, Err, Body] ): F[ValidatedNec[Err, Modules[I, Err, Ast[S]]]] = modules.dependsOn.map { case (moduleId, unresolvedErrors) => @@ -119,7 +116,7 @@ class AquaParser[F[_], E, I, S[_]: Comonad]( err.pure[F] } - def resolveSources: F[ValidatedNec[Err, Modules[I, Err, Ast[S]]]] = + private def resolveSources: F[ValidatedNec[Err, Modules[I, Err, Ast[S]]]] = sourceModules.flatMap { case Validated.Valid(ms) => resolveModules(ms) case err => err.pure[F] diff --git a/parser/src/main/scala/aqua/parser/Ast.scala b/parser/src/main/scala/aqua/parser/Ast.scala index 7bfe51a5f..e392c793b 100644 --- a/parser/src/main/scala/aqua/parser/Ast.scala +++ b/parser/src/main/scala/aqua/parser/Ast.scala @@ -7,6 +7,7 @@ import aqua.parser.lift.LiftParser.* import aqua.helpers.Tree import cats.data.{Chain, Validated, ValidatedNec} +import cats.syntax.flatMap.* import cats.free.Cofree import cats.{Comonad, Eval} import cats.~> @@ -19,6 +20,14 @@ case class Ast[S[_]](head: Ast.Head[S], tree: Ast.Tree[S]) { def cataHead[T](folder: (HeaderExpr[S], Chain[T]) => Eval[T]): Eval[T] = Cofree.cata[Chain, HeaderExpr[S], T](head)(folder) + + def collectHead[T](pf: PartialFunction[HeaderExpr[S], T]): Eval[Chain[T]] = + cataHead((e, acc: Chain[Chain[T]]) => + Eval.later { + val flatAcc = acc.flatten + if (pf.isDefinedAt(e)) flatAcc :+ pf(e) else flatAcc + } + ) } object Ast { diff --git a/semantics/src/main/scala/aqua/semantics/CompilerState.scala b/semantics/src/main/scala/aqua/semantics/CompilerState.scala index 3774467b7..84c57a4dd 100644 --- a/semantics/src/main/scala/aqua/semantics/CompilerState.scala +++ b/semantics/src/main/scala/aqua/semantics/CompilerState.scala @@ -27,7 +27,9 @@ case class CompilerState[S[_]]( definitions: DefinitionsState[S] = DefinitionsState[S](), locations: LocationsState[S] = LocationsState[S]() ) { + lazy val errors: Chain[SemanticError[S]] = report.errors + lazy val warnings: Chain[SemanticWarning[S]] = report.warnings } object CompilerState { diff --git a/semantics/src/main/scala/aqua/semantics/SemanticWarning.scala b/semantics/src/main/scala/aqua/semantics/SemanticWarning.scala new file mode 100644 index 000000000..f23d5ca3f --- /dev/null +++ b/semantics/src/main/scala/aqua/semantics/SemanticWarning.scala @@ -0,0 +1,8 @@ +package aqua.semantics + +import aqua.parser.lexer.Token + +final case class SemanticWarning[S[_]]( + token: Token[S], + hints: List[String] +) diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index fe32294b9..f24927e89 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -283,7 +283,7 @@ object RawSemantics extends Logging { .pure } - private def folder[S[_], G[_]: Monad](implicit + private def folder[S[_], G[_]: Monad](using A: AbilitiesAlgebra[S, G], N: NamesAlgebra[S, G], T: TypesAlgebra[S, G], @@ -327,7 +327,7 @@ object RawSemantics extends Logging { .map(_.raw) } - private def astToState[S[_]](ast: Ast[S])(implicit + private def astToState[S[_]](ast: Ast[S])(using locations: LocationsAlgebra[S, Interpreter[S, *]] ): Interpreter[S, Raw] = transpile[S](ast) @@ -365,6 +365,7 @@ object RawSemantics extends Logging { ) { case (ctx, p) => ctx.copy(parts = ctx.parts :+ (ctx -> p)) } + case (_, m) => internalError( s"Unexpected Raw ($m)" diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala index 2af68e187..c3d069620 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportAlgebra.scala @@ -7,4 +7,9 @@ trait ReportAlgebra[S[_], Alg[_]] { def error(token: Token[S], hint: String): Alg[Unit] = error(token, hint :: Nil) + + def warning(token: Token[S], hints: List[String]): Alg[Unit] + + def warning(token: Token[S], hint: String): Alg[Unit] = + warning(token, hint :: Nil) } diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala index d2b6b23dd..d87be5012 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportInterpreter.scala @@ -15,4 +15,11 @@ class ReportInterpreter[S[_], X](using _.reportError(token, hints) ) ) + + override def warning(token: Token[S], hints: List[String]): State[X, Unit] = + State.modify( + lens.modify( + _.reportWarning(token, hints) + ) + ) } diff --git a/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala b/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala index 751124f51..518671339 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/report/ReportState.scala @@ -1,17 +1,21 @@ package aqua.semantics.rules.report -import aqua.semantics.{RulesViolated, SemanticError} +import aqua.semantics.{RulesViolated, SemanticError, SemanticWarning} import aqua.parser.lexer.Token import cats.data.Chain import cats.kernel.Monoid final case class ReportState[S[_]]( - errors: Chain[SemanticError[S]] = Chain.empty[SemanticError[S]] + errors: Chain[SemanticError[S]] = Chain.empty[SemanticError[S]], + warnings: Chain[SemanticWarning[S]] = Chain.empty[SemanticWarning[S]] ) { def reportError(token: Token[S], hints: List[String]): ReportState[S] = copy(errors = errors.append(RulesViolated(token, hints))) + + def reportWarning(token: Token[S], hints: List[String]): ReportState[S] = + copy(warnings = warnings.append(SemanticWarning(token, hints))) } object ReportState { @@ -20,6 +24,9 @@ object ReportState { override val empty: ReportState[S] = ReportState() override def combine(x: ReportState[S], y: ReportState[S]): ReportState[S] = - ReportState(errors = x.errors ++ y.errors) + ReportState( + errors = x.errors ++ y.errors, + warnings = x.warnings ++ y.warnings + ) } } From e2323c2c2b0e1fea61e2f44c8dd7ddb0dcf0ce9a Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 15:18:04 +0000 Subject: [PATCH 07/31] Refactor parser --- .../main/scala/aqua/compiler/AquaParser.scala | 82 ++++++++++--------- 1 file changed, 43 insertions(+), 39 deletions(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaParser.scala b/compiler/src/main/scala/aqua/compiler/AquaParser.scala index 2c218c8a1..79621254d 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaParser.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaParser.scala @@ -34,12 +34,14 @@ class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( // Parse all the source files private def parseSources: F[ValidatedNec[Err, Chain[(I, Body)]]] = sources.sources.map( - _.leftMap(_.map(SourcesError.apply)).andThen(_.map { case (i, s) => - parser(i)(s).bimap( - _.map(AquaParserError.apply), - ast => Chain.one(i -> ast) - ) - }.foldA) + _.leftMap(_.map(SourcesError.apply)).andThen( + _.traverse { case (i, s) => + parser(i)(s).bimap( + _.map(AquaParserError.apply), + ast => i -> ast + ) + } + ) ) // Resolve imports (not parse, just resolve) of the given file @@ -52,25 +54,23 @@ class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( .map( _.bimap( _.map(ResolveImportsError(id, token, _): Err), - importId => Chain.one(importId -> (filename, ImportError(token): Err)) + importId => importId -> (filename, ImportError(token): Err) ) ) - }.map( - _.foldA.map { collected => - AquaModule[I, Err, Body]( - id, - // How filenames correspond to the resolved IDs - collected.map { case (i, (fn, _)) => - fn -> i - }.toList.toMap[String, I], - // Resolved IDs to errors that point to the import in source code - collected.map { case (i, (_, err)) => - i -> err - }.toList.toMap[I, Err], - ast - ) - } - ) + }.map(_.sequence.map { collected => + AquaModule[I, Err, Body]( + id, + // How filenames correspond to the resolved IDs + collected.map { case (i, (fn, _)) => + fn -> i + }.toList.toMap[String, I], + // Resolved IDs to errors that point to the import in source code + collected.map { case (i, (_, err)) => + i -> err + }.toList.toMap[I, Err], + ast + ) + }) // Parse sources, convert to modules private def sourceModules: F[ValidatedNec[Err, Modules[I, Err, Body]]] = @@ -81,13 +81,19 @@ class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( }.map(_.sequence) case Validated.Invalid(errs) => errs.invalid.pure[F] - }.map(_.map(_.foldLeft(Modules[I, Err, Body]())(_.add(_, toExport = true)))) + }.map( + _.map( + _.foldLeft(Modules[I, Err, Body]())( + _.add(_, toExport = true) + ) + ) + ) private def loadModule(imp: I): F[ValidatedNec[Err, AquaModule[I, Err, Body]]] = sources .load(imp) .map(_.leftMap(_.map(SourcesError.apply)).andThen { src => - parser(imp)(src).leftMap(_.map[Err](AquaParserError.apply)) + parser(imp)(src).leftMap(_.map(AquaParserError.apply)) }) .flatMap { case Validated.Valid(ast) => @@ -99,22 +105,20 @@ class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( private def resolveModules( modules: Modules[I, Err, Body] ): F[ValidatedNec[Err, Modules[I, Err, Ast[S]]]] = - modules.dependsOn.map { case (moduleId, unresolvedErrors) => + modules.dependsOn.toList.traverse { case (moduleId, unresolvedErrors) => loadModule(moduleId).map(_.leftMap(_ ++ unresolvedErrors)) - }.toList.sequence - .map( - _.foldLeft(modules.validNec[Err]) { case (mods, m) => - mods.andThen(ms => m.map(ms.add(_))) - } + }.map( + _.sequence.map( + _.foldLeft(modules)(_ add _) ) - .flatMap { - case Validated.Valid(ms) if ms.isResolved => - ms.validNec.pure[F] - case Validated.Valid(ms) => - resolveModules(ms) - case err => - err.pure[F] - } + ).flatMap { + case Validated.Valid(ms) if ms.isResolved => + ms.validNec.pure[F] + case Validated.Valid(ms) => + resolveModules(ms) + case err => + err.pure[F] + } private def resolveSources: F[ValidatedNec[Err, Modules[I, Err, Ast[S]]]] = sourceModules.flatMap { From 61042ee8cebb432718d535490bfa9a69225165b7 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 15:37:02 +0000 Subject: [PATCH 08/31] Move semantics --- .../scala/aqua/semantics/RawSemantics.scala | 355 +++++++++++++++++ .../main/scala/aqua/semantics/Semantics.scala | 366 +----------------- 2 files changed, 358 insertions(+), 363 deletions(-) create mode 100644 semantics/src/main/scala/aqua/semantics/RawSemantics.scala diff --git a/semantics/src/main/scala/aqua/semantics/RawSemantics.scala b/semantics/src/main/scala/aqua/semantics/RawSemantics.scala new file mode 100644 index 000000000..9363190b0 --- /dev/null +++ b/semantics/src/main/scala/aqua/semantics/RawSemantics.scala @@ -0,0 +1,355 @@ +package aqua.semantics + +import aqua.errors.Errors.internalError +import aqua.raw.ops.* +import aqua.semantics.rules.abilities.{AbilitiesAlgebra, AbilitiesInterpreter, AbilitiesState} +import aqua.semantics.rules.definitions.{DefinitionsAlgebra, DefinitionsInterpreter} +import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgebra} +import aqua.semantics.rules.names.{NamesAlgebra, NamesInterpreter} +import aqua.semantics.rules.mangler.{ManglerAlgebra, ManglerInterpreter} +import aqua.semantics.rules.types.{TypesAlgebra, TypesInterpreter} +import aqua.semantics.rules.report.{ReportAlgebra, ReportInterpreter} +import aqua.semantics.header.Picker +import aqua.semantics.header.Picker.* +import aqua.raw.{Raw, RawContext, RawPart} +import aqua.parser.{Ast, Expr} +import aqua.parser.lexer.{LiteralToken, Token} + +import cats.{Eval, Monad} +import cats.data.{Chain, NonEmptyChain, State, StateT, ValidatedNec} +import cats.syntax.applicative.* +import cats.syntax.option.* +import cats.syntax.apply.* +import cats.syntax.flatMap.* +import cats.syntax.functor.* +import cats.syntax.foldable.* +import cats.syntax.reducible.* +import cats.syntax.traverse.* +import cats.syntax.semigroup.* +import scribe.Logging + +class RawSemantics[S[_]](using + Picker[RawContext] +) extends Semantics[S, RawContext] { + + def process( + ast: Ast[S], + init: RawContext + ): ValidatedNec[SemanticError[S], RawContext] = { + + given LocationsAlgebra[S, State[CompilerState[S], *]] = + new DummyLocationsInterpreter[S, CompilerState[S]]() + + RawSemantics + .interpret(ast, CompilerState.init(init), init) + .map { case (state, ctx) => + NonEmptyChain + .fromChain(state.errors) + .toInvalid(ctx) + } + // TODO: return as Eval + .value + } +} + +object RawSemantics extends Logging { + + /** + * [[RawTag.Tree]] with [[Token]] used for error reporting + */ + private final case class RawTagWithToken[S[_]]( + tree: RawTag.Tree, + token: Token[S] + ) { + lazy val tag: RawTag = tree.head + + private def modifyTree(f: RawTag.Tree => RawTag.Tree): RawTagWithToken[S] = + copy(tree = f(tree)) + + /** + * Wrap tail of @param next in [[SeqTag]] + * and append it to current tree tail + */ + def append(next: RawTagWithToken[S]): RawTagWithToken[S] = modifyTree(tree => + tree.copy( + tail = ( + tree.tail, + // SeqTag.wrap will return single node as is + next.tree.tail.map(SeqTag.wrap) + ).mapN(_ :+ _) + ) + ) + + def wrapIn(tag: GroupTag): RawTagWithToken[S] = modifyTree(tree => tag.wrap(tree)) + + def toRaw: RawWithToken[S] = RawWithToken(FuncOp(tree), token) + } + + private def elseWithoutIf[S[_], G[_]]( + token: Token[S] + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `else` without `if`" :: Nil) + + private def catchWithoutTry[S[_], G[_]]( + token: Token[S] + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `catch` without `try`" :: Nil) + + private def otherwiseWithoutPrev[S[_], G[_]]( + token: Token[S] + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `otherwise` without previous instruction" :: Nil) + + private def parWithoutPrev[S[_], G[_]]( + token: Token[S] + )(using report: ReportAlgebra[S, G]): G[Unit] = + report.error(token, "Unexpected `par` without previous instruction" :: Nil) + + /** + * Optionally combine two [[RawTag.Tree]] into one. + * Used to combine `if` and `else`, + * `try` and `catch` (`otherwise`); + * to create [[ParTag]] from `par`, + * [[TryTag]] from `otherwise` + * + * @param prev Previous tag + * @param next Next tag + * @param E Algebra for error reporting + * @return [[Some]] with result of combination + * [[None]] if tags should not be combined + * or error occuried + */ + private def rawTagCombine[S[_], G[_]: Monad]( + prev: RawTagWithToken[S], + next: RawTagWithToken[S] + )(using E: ReportAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = + (prev.tag, next.tag) match { + case (_: IfTag, IfTag.Else) => + prev.append(next).some.pure + case (_, IfTag.Else) | (IfTag.Else, _) => + val token = prev.tag match { + case IfTag.Else => prev.token + case _ => next.token + } + + elseWithoutIf(token).as(none) + + case (TryTag, TryTag.Catch) => + prev.append(next).some.pure + case (_, TryTag.Catch) | (TryTag.Catch, _) => + val token = prev.tag match { + case TryTag.Catch => prev.token + case _ => next.token + } + + catchWithoutTry(token).as(none) + + case (TryTag.Otherwise, _) => + otherwiseWithoutPrev(prev.token).as(none) + case (TryTag, TryTag.Otherwise) => + prev.append(next).some.pure + case (_, TryTag.Otherwise) => + prev + .wrapIn(TryTag) + .append(next) + .some + .pure + + case (ParTag.Par, _) => + parWithoutPrev(prev.token).as(none) + case (ParTag, ParTag.Par) => + prev.append(next).some.pure + case (_, ParTag.Par) => + prev + .wrapIn(ParTag) + .append(next) + .some + .pure + + case _ => none.pure + } + + /** + * Check if tag is valid to be single + * + * @param single tag + * @param E Algebra for error reporting + * @return [[Some]] if tag is valid to be single + * [[None]] otherwise + */ + private def rawTagSingleCheck[S[_], G[_]: Monad]( + single: RawTagWithToken[S] + )(using E: ReportAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = + single.tag match { + case IfTag.Else => elseWithoutIf(single.token).as(none) + case TryTag.Catch => catchWithoutTry(single.token).as(none) + case TryTag.Otherwise => otherwiseWithoutPrev(single.token).as(none) + case ParTag.Par => parWithoutPrev(single.token).as(none) + case _ => single.some.pure + } + + /** + * [[Raw]] with [[Token]] used for error reporting + */ + private final case class RawWithToken[S[_]]( + raw: Raw, + token: Token[S] + ) { + + def toTag: Option[RawTagWithToken[S]] = + raw match { + case FuncOp(tree) => RawTagWithToken(tree, token).some + case _ => none + } + + } + + /** + * State for folding [[Raw]] results of children + * + * @param last Last seen [[Raw]] with [[Token]] + * @param acc All previous [[Raw]] + */ + private final case class InnersFoldState[S[_]]( + last: Option[RawWithToken[S]] = None, + acc: Chain[Raw] = Chain.empty + ) { + + /** + * Process new incoming [[Raw]] + */ + def step[G[_]: Monad]( + next: RawWithToken[S] + )(using ReportAlgebra[S, G]): G[InnersFoldState[S]] = + last.fold(copy(last = next.some).pure)(prev => + (prev.toTag, next.toTag) + .traverseN(rawTagCombine) + .map( + _.flatten.fold( + // No combination - just update last and acc + copy( + last = next.some, + acc = prev.raw +: acc + ) + )(combined => + // Result of combination is the new last + copy( + last = combined.toRaw.some + ) + ) + ) + ) + + /** + * Produce result of folding + */ + def result[G[_]: Monad](using + ReportAlgebra[S, G] + ): G[Option[Raw]] = + if (acc.isEmpty) + // Hack to report error if single tag in block is incorrect + last.flatTraverse(single => + single.toTag.fold(single.raw.some.pure)(singleTag => + for { + checked <- rawTagSingleCheck(singleTag) + maybeRaw = checked.map(_.toRaw.raw) + } yield maybeRaw + ) + ) + else + last + .fold(acc)(_.raw +: acc) + .reverse + .reduceLeftOption(_ |+| _) + .pure + } + + private def folder[S[_], G[_]: Monad](using + A: AbilitiesAlgebra[S, G], + N: NamesAlgebra[S, G], + T: TypesAlgebra[S, G], + D: DefinitionsAlgebra[S, G], + L: LocationsAlgebra[S, G], + E: ReportAlgebra[S, G] + ): (Expr[S], Chain[G[RawWithToken[S]]]) => Eval[G[RawWithToken[S]]] = (expr, inners) => + Eval later ExprSem + .getProg[S, G](expr) + .apply(for { + children <- inners.sequence + resultState <- children + .traverse(raw => StateT.modifyF((state: InnersFoldState[S]) => state.step(raw))) + .runS(InnersFoldState()) + result <- resultState.result + } yield result.getOrElse(Raw.empty("AST is empty"))) + .map(raw => RawWithToken(raw, expr.token)) + + type Interpreter[S[_], A] = State[CompilerState[S], A] + + def transpile[S[_]](ast: Ast[S])(using + LocationsAlgebra[S, Interpreter[S, *]] + ): Interpreter[S, Raw] = { + + given ReportAlgebra[S, Interpreter[S, *]] = + new ReportInterpreter[S, CompilerState[S]] + given TypesAlgebra[S, Interpreter[S, *]] = + new TypesInterpreter[S, CompilerState[S]] + given ManglerAlgebra[Interpreter[S, *]] = + new ManglerInterpreter[CompilerState[S]] + given AbilitiesAlgebra[S, Interpreter[S, *]] = + new AbilitiesInterpreter[S, CompilerState[S]] + given NamesAlgebra[S, Interpreter[S, *]] = + new NamesInterpreter[S, CompilerState[S]] + given DefinitionsAlgebra[S, Interpreter[S, *]] = + new DefinitionsInterpreter[S, CompilerState[S]] + + ast + .cata(folder[S, Interpreter[S, *]]) + .value + .map(_.raw) + } + + private def astToState[S[_]](ast: Ast[S])(using + locations: LocationsAlgebra[S, Interpreter[S, *]] + ): Interpreter[S, Raw] = + transpile[S](ast) + + // If there are any errors, they're inside CompilerState[S] + def interpret[S[_]]( + ast: Ast[S], + initState: CompilerState[S], + init: RawContext + )(using + LocationsAlgebra[S, Interpreter[S, *]] + ): Eval[(CompilerState[S], RawContext)] = + astToState[S](ast) + .run(initState) + .map { + case (state, _: Raw.Empty) => + // No `parts`, but has `init` + ( + state, + RawContext.blank.copy( + init = Some(init.copy(module = init.module.map(_ + "|init"))) + .filter(_ != RawContext.blank) + ) + ) + + case (state, part: (RawPart | RawPart.Parts)) => + state -> RawPart + .contextPart(part) + .parts + .foldLeft( + RawContext.blank.copy( + init = Some(init.copy(module = init.module.map(_ + "|init"))) + .filter(_ != RawContext.blank) + ) + ) { case (ctx, p) => + ctx.copy(parts = ctx.parts :+ (ctx -> p)) + } + + case (_, m) => + internalError( + s"Unexpected Raw ($m)" + ) + } +} diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index f24927e89..9c6d1f854 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -1,43 +1,9 @@ package aqua.semantics -import aqua.errors.Errors.internalError -import aqua.parser.head.{HeadExpr, HeaderExpr, ImportExpr, ImportFromExpr} -import aqua.parser.lexer.{LiteralToken, Token} -import aqua.parser.{Ast, Expr} -import aqua.raw.ops.{FuncOp, SeqGroupTag} -import aqua.raw.{Raw, RawContext, RawPart} -import aqua.semantics.header.Picker -import aqua.semantics.header.Picker.* -import aqua.semantics.rules.abilities.{AbilitiesAlgebra, AbilitiesInterpreter, AbilitiesState} -import aqua.semantics.rules.definitions.{DefinitionsAlgebra, DefinitionsInterpreter} -import aqua.semantics.rules.locations.{DummyLocationsInterpreter, LocationsAlgebra} -import aqua.semantics.rules.names.{NamesAlgebra, NamesInterpreter} -import aqua.semantics.rules.mangler.{ManglerAlgebra, ManglerInterpreter} -import aqua.semantics.rules.types.{TypesAlgebra, TypesInterpreter} -import aqua.semantics.rules.report.{ReportAlgebra, ReportInterpreter} -import aqua.raw.ops.* +import aqua.parser.Ast +import aqua.semantics.SemanticError -import cats.arrow.FunctionK -import cats.data.* -import cats.Reducible -import cats.data.Chain.* -import cats.data.Validated.{Invalid, Valid} -import cats.kernel.Monoid -import cats.syntax.applicative.* -import cats.syntax.option.* -import cats.syntax.apply.* -import cats.syntax.flatMap.* -import cats.syntax.functor.* -import cats.syntax.foldable.* -import cats.syntax.reducible.* -import cats.syntax.traverse.* -import cats.free.CofreeInstances -import cats.syntax.semigroup.* -import cats.{Eval, Monad, Semigroup} -import monocle.Lens -import monocle.macros.GenLens -import scribe.{log, Logging} -import cats.free.Cofree +import cats.data.ValidatedNec trait Semantics[S[_], C] { @@ -46,329 +12,3 @@ trait Semantics[S[_], C] { init: C ): ValidatedNec[SemanticError[S], C] } - -class RawSemantics[S[_]](using - Picker[RawContext] -) extends Semantics[S, RawContext] { - - def process( - ast: Ast[S], - init: RawContext - ): ValidatedNec[SemanticError[S], RawContext] = { - - given LocationsAlgebra[S, State[CompilerState[S], *]] = - new DummyLocationsInterpreter[S, CompilerState[S]]() - - RawSemantics - .interpret(ast, CompilerState.init(init), init) - .map { case (state, ctx) => - NonEmptyChain - .fromChain(state.errors) - .toInvalid(ctx) - } - // TODO: return as Eval - .value - } -} - -object RawSemantics extends Logging { - - /** - * [[RawTag.Tree]] with [[Token]] used for error reporting - */ - private final case class RawTagWithToken[S[_]]( - tree: RawTag.Tree, - token: Token[S] - ) { - lazy val tag: RawTag = tree.head - - private def modifyTree(f: RawTag.Tree => RawTag.Tree): RawTagWithToken[S] = - copy(tree = f(tree)) - - /** - * Wrap tail of @param next in [[SeqTag]] - * and append it to current tree tail - */ - def append(next: RawTagWithToken[S]): RawTagWithToken[S] = modifyTree(tree => - tree.copy( - tail = ( - tree.tail, - // SeqTag.wrap will return single node as is - next.tree.tail.map(SeqTag.wrap) - ).mapN(_ :+ _) - ) - ) - - def wrapIn(tag: GroupTag): RawTagWithToken[S] = modifyTree(tree => tag.wrap(tree)) - - def toRaw: RawWithToken[S] = RawWithToken(FuncOp(tree), token) - } - - private def elseWithoutIf[S[_], G[_]]( - token: Token[S] - )(using report: ReportAlgebra[S, G]): G[Unit] = - report.error(token, "Unexpected `else` without `if`" :: Nil) - - private def catchWithoutTry[S[_], G[_]]( - token: Token[S] - )(using report: ReportAlgebra[S, G]): G[Unit] = - report.error(token, "Unexpected `catch` without `try`" :: Nil) - - private def otherwiseWithoutPrev[S[_], G[_]]( - token: Token[S] - )(using report: ReportAlgebra[S, G]): G[Unit] = - report.error(token, "Unexpected `otherwise` without previous instruction" :: Nil) - - private def parWithoutPrev[S[_], G[_]]( - token: Token[S] - )(using report: ReportAlgebra[S, G]): G[Unit] = - report.error(token, "Unexpected `par` without previous instruction" :: Nil) - - /** - * Optionally combine two [[RawTag.Tree]] into one. - * Used to combine `if` and `else`, - * `try` and `catch` (`otherwise`); - * to create [[ParTag]] from `par`, - * [[TryTag]] from `otherwise` - * - * @param prev Previous tag - * @param next Next tag - * @param E Algebra for error reporting - * @return [[Some]] with result of combination - * [[None]] if tags should not be combined - * or error occuried - */ - private def rawTagCombine[S[_], G[_]: Monad]( - prev: RawTagWithToken[S], - next: RawTagWithToken[S] - )(using E: ReportAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = - (prev.tag, next.tag) match { - case (_: IfTag, IfTag.Else) => - prev.append(next).some.pure - case (_, IfTag.Else) | (IfTag.Else, _) => - val token = prev.tag match { - case IfTag.Else => prev.token - case _ => next.token - } - - elseWithoutIf(token).as(none) - - case (TryTag, TryTag.Catch) => - prev.append(next).some.pure - case (_, TryTag.Catch) | (TryTag.Catch, _) => - val token = prev.tag match { - case TryTag.Catch => prev.token - case _ => next.token - } - - catchWithoutTry(token).as(none) - - case (TryTag.Otherwise, _) => - otherwiseWithoutPrev(prev.token).as(none) - case (TryTag, TryTag.Otherwise) => - prev.append(next).some.pure - case (_, TryTag.Otherwise) => - prev - .wrapIn(TryTag) - .append(next) - .some - .pure - - case (ParTag.Par, _) => - parWithoutPrev(prev.token).as(none) - case (ParTag, ParTag.Par) => - prev.append(next).some.pure - case (_, ParTag.Par) => - prev - .wrapIn(ParTag) - .append(next) - .some - .pure - - case _ => none.pure - } - - /** - * Check if tag is valid to be single - * - * @param single tag - * @param E Algebra for error reporting - * @return [[Some]] if tag is valid to be single - * [[None]] otherwise - */ - private def rawTagSingleCheck[S[_], G[_]: Monad]( - single: RawTagWithToken[S] - )(using E: ReportAlgebra[S, G]): G[Option[RawTagWithToken[S]]] = - single.tag match { - case IfTag.Else => elseWithoutIf(single.token).as(none) - case TryTag.Catch => catchWithoutTry(single.token).as(none) - case TryTag.Otherwise => otherwiseWithoutPrev(single.token).as(none) - case ParTag.Par => parWithoutPrev(single.token).as(none) - case _ => single.some.pure - } - - /** - * [[Raw]] with [[Token]] used for error reporting - */ - private final case class RawWithToken[S[_]]( - raw: Raw, - token: Token[S] - ) { - - def toTag: Option[RawTagWithToken[S]] = - raw match { - case FuncOp(tree) => RawTagWithToken(tree, token).some - case _ => none - } - - } - - /** - * State for folding [[Raw]] results of children - * - * @param last Last seen [[Raw]] with [[Token]] - * @param acc All previous [[Raw]] - */ - private final case class InnersFoldState[S[_]]( - last: Option[RawWithToken[S]] = None, - acc: Chain[Raw] = Chain.empty - ) { - - /** - * Process new incoming [[Raw]] - */ - def step[G[_]: Monad]( - next: RawWithToken[S] - )(using ReportAlgebra[S, G]): G[InnersFoldState[S]] = - last.fold(copy(last = next.some).pure)(prev => - (prev.toTag, next.toTag) - .traverseN(rawTagCombine) - .map( - _.flatten.fold( - // No combination - just update last and acc - copy( - last = next.some, - acc = prev.raw +: acc - ) - )(combined => - // Result of combination is the new last - copy( - last = combined.toRaw.some - ) - ) - ) - ) - - /** - * Produce result of folding - */ - def result[G[_]: Monad](using - ReportAlgebra[S, G] - ): G[Option[Raw]] = - if (acc.isEmpty) - // Hack to report error if single tag in block is incorrect - last.flatTraverse(single => - single.toTag.fold(single.raw.some.pure)(singleTag => - for { - checked <- rawTagSingleCheck(singleTag) - maybeRaw = checked.map(_.toRaw.raw) - } yield maybeRaw - ) - ) - else - last - .fold(acc)(_.raw +: acc) - .reverse - .reduceLeftOption(_ |+| _) - .pure - } - - private def folder[S[_], G[_]: Monad](using - A: AbilitiesAlgebra[S, G], - N: NamesAlgebra[S, G], - T: TypesAlgebra[S, G], - D: DefinitionsAlgebra[S, G], - L: LocationsAlgebra[S, G], - E: ReportAlgebra[S, G] - ): (Expr[S], Chain[G[RawWithToken[S]]]) => Eval[G[RawWithToken[S]]] = (expr, inners) => - Eval later ExprSem - .getProg[S, G](expr) - .apply(for { - children <- inners.sequence - resultState <- children - .traverse(raw => StateT.modifyF((state: InnersFoldState[S]) => state.step(raw))) - .runS(InnersFoldState()) - result <- resultState.result - } yield result.getOrElse(Raw.empty("AST is empty"))) - .map(raw => RawWithToken(raw, expr.token)) - - type Interpreter[S[_], A] = State[CompilerState[S], A] - - def transpile[S[_]](ast: Ast[S])(using - LocationsAlgebra[S, Interpreter[S, *]] - ): Interpreter[S, Raw] = { - - given ReportAlgebra[S, Interpreter[S, *]] = - new ReportInterpreter[S, CompilerState[S]] - given TypesAlgebra[S, Interpreter[S, *]] = - new TypesInterpreter[S, CompilerState[S]] - given ManglerAlgebra[Interpreter[S, *]] = - new ManglerInterpreter[CompilerState[S]] - given AbilitiesAlgebra[S, Interpreter[S, *]] = - new AbilitiesInterpreter[S, CompilerState[S]] - given NamesAlgebra[S, Interpreter[S, *]] = - new NamesInterpreter[S, CompilerState[S]] - given DefinitionsAlgebra[S, Interpreter[S, *]] = - new DefinitionsInterpreter[S, CompilerState[S]] - - ast - .cata(folder[S, Interpreter[S, *]]) - .value - .map(_.raw) - } - - private def astToState[S[_]](ast: Ast[S])(using - locations: LocationsAlgebra[S, Interpreter[S, *]] - ): Interpreter[S, Raw] = - transpile[S](ast) - - // If there are any errors, they're inside CompilerState[S] - def interpret[S[_]]( - ast: Ast[S], - initState: CompilerState[S], - init: RawContext - )(using - LocationsAlgebra[S, Interpreter[S, *]] - ): Eval[(CompilerState[S], RawContext)] = - astToState[S](ast) - .run(initState) - .map { - case (state, _: Raw.Empty) => - // No `parts`, but has `init` - ( - state, - RawContext.blank.copy( - init = Some(init.copy(module = init.module.map(_ + "|init"))) - .filter(_ != RawContext.blank) - ) - ) - - case (state, part: (RawPart | RawPart.Parts)) => - state -> RawPart - .contextPart(part) - .parts - .foldLeft( - RawContext.blank.copy( - init = Some(init.copy(module = init.module.map(_ + "|init"))) - .filter(_ != RawContext.blank) - ) - ) { case (ctx, p) => - ctx.copy(parts = ctx.parts :+ (ctx -> p)) - } - - case (_, m) => - internalError( - s"Unexpected Raw ($m)" - ) - } -} From e10bd62d07769944e655ced664bc750aaf629b55 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Wed, 20 Sep 2023 16:21:31 +0000 Subject: [PATCH 09/31] Savepoint --- .../scala/aqua/semantics/RawSemantics.scala | 20 ++++++++++++------- .../main/scala/aqua/semantics/Semantics.scala | 14 ++++++++++--- 2 files changed, 24 insertions(+), 10 deletions(-) diff --git a/semantics/src/main/scala/aqua/semantics/RawSemantics.scala b/semantics/src/main/scala/aqua/semantics/RawSemantics.scala index 9363190b0..653c7a6b8 100644 --- a/semantics/src/main/scala/aqua/semantics/RawSemantics.scala +++ b/semantics/src/main/scala/aqua/semantics/RawSemantics.scala @@ -16,7 +16,7 @@ import aqua.parser.{Ast, Expr} import aqua.parser.lexer.{LiteralToken, Token} import cats.{Eval, Monad} -import cats.data.{Chain, NonEmptyChain, State, StateT, ValidatedNec} +import cats.data.{Chain, EitherT, NonEmptyChain, State, StateT, ValidatedNec, Writer} import cats.syntax.applicative.* import cats.syntax.option.* import cats.syntax.apply.* @@ -30,12 +30,12 @@ import scribe.Logging class RawSemantics[S[_]](using Picker[RawContext] -) extends Semantics[S, RawContext] { +) extends Semantics[S, SemanticWarning[S], SemanticError[S], RawContext] { - def process( + override def process( ast: Ast[S], init: RawContext - ): ValidatedNec[SemanticError[S], RawContext] = { + ): ProcessResult = { given LocationsAlgebra[S, State[CompilerState[S], *]] = new DummyLocationsInterpreter[S, CompilerState[S]]() @@ -43,9 +43,15 @@ class RawSemantics[S[_]](using RawSemantics .interpret(ast, CompilerState.init(init), init) .map { case (state, ctx) => - NonEmptyChain - .fromChain(state.errors) - .toInvalid(ctx) + EitherT( + Writer + .tell(state.warnings) + .as( + NonEmptyChain + .fromChain(state.errors) + .toLeft(ctx) + ) + ) } // TODO: return as Eval .value diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index 9c6d1f854..d4ba2194c 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -3,12 +3,20 @@ package aqua.semantics import aqua.parser.Ast import aqua.semantics.SemanticError -import cats.data.ValidatedNec +import cats.data.{Chain, EitherNec, EitherT, NonEmptyChain, ValidatedNec, Writer} -trait Semantics[S[_], C] { +trait Semantics[S[_], W, E, C] { + + final type ProcessResult = Semantics.ProcessResult[W, E, C] def process( ast: Ast[S], init: C - ): ValidatedNec[SemanticError[S], C] + ): ProcessResult +} + +object Semantics { + + type ProcessResult[W, E, C] = + EitherT[Writer[Chain[W], *], NonEmptyChain[E], C] } From da8667fbd42ddf7422f1d2b23d6f1426be46ae57 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Thu, 21 Sep 2023 08:40:42 +0000 Subject: [PATCH 10/31] Refactor semantics and compiler --- .../scala/aqua/compiler/AquaCompiler.scala | 59 ++++++++++--------- .../main/scala/aqua/lsp/LspSemantics.scala | 28 ++++----- .../scala/aqua/semantics/RawSemantics.scala | 2 +- .../main/scala/aqua/semantics/Semantics.scala | 19 +++--- .../scala/aqua/semantics/SemanticsSpec.scala | 4 +- 5 files changed, 59 insertions(+), 53 deletions(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index 9babf6213..b7952ad13 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -11,6 +11,7 @@ import aqua.raw.{RawContext, RawPart} import aqua.res.AquaRes import aqua.semantics.{CompilerState, Semantics} import aqua.semantics.header.{HeaderHandler, HeaderSem, Picker} +import aqua.semantics.{SemanticError, SemanticWarning} import cats.data.* import cats.data.Validated.{validNec, Invalid, Valid} @@ -21,6 +22,7 @@ import cats.syntax.functor.* import cats.syntax.monoid.* import cats.syntax.traverse.* import cats.syntax.semigroup.* +import cats.syntax.either.* import cats.{~>, Comonad, Functor, Monad, Monoid, Order} import scribe.Logging @@ -62,37 +64,40 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( logger.trace("starting resolving sources...") new AquaParser[F, E, I, S](sources, parser) .resolve[ValidatedCtx](mod => - context => + context => { // Context with prepared imports context.andThen { ctx => - val imports = mod.imports.view - .mapValues(ctx(_)) - .collect { case (fn, Some(fc)) => fn -> fc } - .toMap + val imports = mod.imports.flatMap { case (fn, id) => + ctx.apply(id).map(fn -> _) + } val header = mod.body.head - // To manage imports, exports run HeaderHandler - headerHandler - .sem( - imports, - header - ) - .andThen { headerSem => - // Analyze the body, with prepared initial context - logger.trace("semantic processing...") - semantics - .process( - mod.body, - headerSem.initCtx - ) - // Handle exports, declares - finalize the resulting context - .andThen { ctx => - headerSem.finCtx(ctx) - } - .map { rc => NonEmptyMap.one(mod.id, rc) } - } - // The whole chain returns a semantics error finally - .leftMap(_.map[Err](CompileError.apply)) + + val process = for { + // To manage imports, exports run HeaderHandler + headerSem <- headerHandler + .sem(imports, header) + .toEither + .toEitherT[semantics.ProcessWarnings] + // Analyze the body, with prepared initial context + _ = logger.trace("semantic processing...") + processed <- semantics + .process( + mod.body, + headerSem.initCtx + ) + // Handle exports, declares - finalize the resulting context + rc <- headerSem + .finCtx(processed) + .toEither + .toEitherT[semantics.ProcessWarnings] + } yield NonEmptyMap.one(mod.id, rc) + + process + .leftMap(_.map(CompileError.apply)) + .toValidated + .value } + } ) .map( _.andThen { modules => linkModules(modules, cycle => CycleError[I, E, S](cycle.map(_.id))) } diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala index d79e00e7a..0544378a4 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala @@ -4,7 +4,7 @@ import aqua.parser.Ast import aqua.parser.head.{ImportExpr, ImportFromExpr, UseExpr, UseFromExpr} import aqua.parser.lexer.{LiteralToken, Token} import aqua.semantics.rules.locations.LocationsState -import aqua.semantics.{CompilerState, RawSemantics, SemanticError, Semantics} +import aqua.semantics.{CompilerState, RawSemantics, SemanticError, SemanticWarning, Semantics} import cats.data.Validated.{Invalid, Valid} import cats.syntax.applicative.* @@ -12,6 +12,7 @@ import cats.syntax.apply.* import cats.syntax.flatMap.* import cats.syntax.functor.* import cats.syntax.foldable.* +import cats.syntax.either.* import cats.syntax.reducible.* import cats.data.{NonEmptyChain, ValidatedNec} import monocle.Lens @@ -34,7 +35,7 @@ class LspSemantics[S[_]] extends Semantics[S, LspContext[S]] { def process( ast: Ast[S], init: LspContext[S] - ): ValidatedNec[SemanticError[S], LspContext[S]] = { + ): ProcessResult = { val rawState = CompilerState.init[S](init.raw) @@ -62,19 +63,16 @@ class LspSemantics[S[_]] extends Semantics[S, LspContext[S]] { RawSemantics .interpret(ast, initState, init.raw) .map { case (state, ctx) => - // TODO: better to change return type in `process` method - Valid( - LspContext( - raw = ctx, - rootArrows = state.names.rootArrows, - constants = state.names.constants, - abDefinitions = state.abilities.definitions, - locations = state.locations.allLocations, - importTokens = importTokens, - tokens = state.locations.tokens, - errors = state.errors.toList - ) - ) + LspContext( + raw = ctx, + rootArrows = state.names.rootArrows, + constants = state.names.constants, + abDefinitions = state.abilities.definitions, + locations = state.locations.allLocations, + importTokens = importTokens, + tokens = state.locations.tokens, + errors = state.errors.toList + ).asRight.toEitherT[ProcessWarnings] } // TODO: return as Eval .value diff --git a/semantics/src/main/scala/aqua/semantics/RawSemantics.scala b/semantics/src/main/scala/aqua/semantics/RawSemantics.scala index 653c7a6b8..6f6142129 100644 --- a/semantics/src/main/scala/aqua/semantics/RawSemantics.scala +++ b/semantics/src/main/scala/aqua/semantics/RawSemantics.scala @@ -30,7 +30,7 @@ import scribe.Logging class RawSemantics[S[_]](using Picker[RawContext] -) extends Semantics[S, SemanticWarning[S], SemanticError[S], RawContext] { +) extends Semantics[S, RawContext] { override def process( ast: Ast[S], diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index d4ba2194c..969505749 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -5,18 +5,21 @@ import aqua.semantics.SemanticError import cats.data.{Chain, EitherNec, EitherT, NonEmptyChain, ValidatedNec, Writer} -trait Semantics[S[_], W, E, C] { +trait Semantics[S[_], C] { - final type ProcessResult = Semantics.ProcessResult[W, E, C] + final type ProcessWarnings = [A] =>> Writer[ + Chain[SemanticWarning[S]], + A + ] + + final type ProcessResult = EitherT[ + ProcessWarnings, + NonEmptyChain[SemanticError[S]], + C + ] def process( ast: Ast[S], init: C ): ProcessResult } - -object Semantics { - - type ProcessResult[W, E, C] = - EitherT[Writer[Chain[W], *], NonEmptyChain[E], C] -} diff --git a/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala b/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala index 453e22999..35e25f731 100644 --- a/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala +++ b/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala @@ -35,7 +35,7 @@ class SemanticsSpec extends AnyFlatSpec with Matchers with Inside { def insideBody(script: String)(test: RawTag.Tree => Any): Unit = inside(parser(script)) { case Validated.Valid(ast) => val init = RawContext.blank - inside(semantics.process(ast, init)) { case Validated.Valid(ctx) => + inside(semantics.process(ast, init).value.value) { case Right(ctx) => inside(ctx.funcs.headOption) { case Some((_, func)) => test(func.arrow.body) } @@ -45,7 +45,7 @@ class SemanticsSpec extends AnyFlatSpec with Matchers with Inside { def insideSemErrors(script: String)(test: NonEmptyChain[SemanticError[Span.S]] => Any): Unit = inside(parser(script)) { case Validated.Valid(ast) => val init = RawContext.blank - inside(semantics.process(ast, init)) { case Validated.Invalid(errors) => + inside(semantics.process(ast, init).value.value) { case Left(errors) => test(errors) } } From 178c538379d9756db5709eca0d7acfd36d7fd3e6 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Thu, 21 Sep 2023 10:00:44 +0000 Subject: [PATCH 11/31] Refactor types --- .../scala/aqua/compiler/AquaCompiler.scala | 11 +++++++--- .../main/scala/aqua/compiler/AquaParser.scala | 22 ++++++++++++------- 2 files changed, 22 insertions(+), 11 deletions(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index b7952ad13..fec81d190 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -44,7 +44,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( ValidatedCtxT ], cycleError: Linker.DepCycle[AquaModule[I, Err, ValidatedCtxT]] => Err - ): ValidatedNec[Err, Map[I, ValidatedCtx]] = { + ): EitherNec[Err, Map[I, ValidatedCtx]] = { logger.trace("linking modules...") Linker @@ -54,6 +54,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( // By default, provide an empty context for this module's id i => validNec(NonEmptyMap.one(i, Monoid.empty[C])) ) + .toEither } def compileRaw( @@ -99,9 +100,13 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( } } ) - .map( - _.andThen { modules => linkModules(modules, cycle => CycleError[I, E, S](cycle.map(_.id))) } + .subflatMap(modules => + linkModules( + modules, + cycle => CycleError[I, E, S](cycle.map(_.id)) + ) ) + .toValidated } } diff --git a/compiler/src/main/scala/aqua/compiler/AquaParser.scala b/compiler/src/main/scala/aqua/compiler/AquaParser.scala index 79621254d..c2f35e75f 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaParser.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaParser.scala @@ -6,7 +6,7 @@ import aqua.parser.head.{FilenameExpr, ImportExpr} import aqua.parser.lift.{LiftParser, Span} import aqua.parser.{Ast, ParserError} -import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} +import cats.data.{Chain, EitherNec, EitherT, NonEmptyChain, Validated, ValidatedNec} import cats.parse.Parser0 import cats.syntax.either.* import cats.syntax.applicative.* @@ -31,6 +31,8 @@ class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( type Body = Ast[S] type Err = AquaError[I, E, S] + private type FE[A] = EitherT[F, NonEmptyChain[Err], A] + // Parse all the source files private def parseSources: F[ValidatedNec[Err, Chain[(I, Body)]]] = sources.sources.map( @@ -120,15 +122,19 @@ class AquaParser[F[_]: Monad, E, I, S[_]: Comonad]( err.pure[F] } - private def resolveSources: F[ValidatedNec[Err, Modules[I, Err, Ast[S]]]] = - sourceModules.flatMap { - case Validated.Valid(ms) => resolveModules(ms) - case err => err.pure[F] - } + private def resolveSources: FE[Modules[I, Err, Ast[S]]] = + for { + ms <- EitherT( + sourceModules.map(_.toEither) + ) + res <- EitherT( + resolveModules(ms).map(_.toEither) + ) + } yield res def resolve[T]( transpile: AquaModule[I, Err, Body] => T => T - ): F[ValidatedNec[Err, Modules[I, Err, T => T]]] = - resolveSources.map(_.map(_.mapModuleToBody(transpile))) + ): FE[Modules[I, Err, T => T]] = + resolveSources.map(_.mapModuleToBody(transpile)) } From 19b7ee336446fc1061f491651673fe8b52f61e27 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Thu, 21 Sep 2023 12:24:03 +0000 Subject: [PATCH 12/31] Refactor compiler --- .../scala/aqua/compiler/AquaCompiler.scala | 108 +++++++++++------- .../scala/aqua/compiler/AquaWarning.scala | 7 ++ .../src/main/scala/aqua/linker/Linker.scala | 2 +- 3 files changed, 72 insertions(+), 45 deletions(-) create mode 100644 compiler/src/main/scala/aqua/compiler/AquaWarning.scala diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index fec81d190..d118cf6a8 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -24,6 +24,7 @@ import cats.syntax.traverse.* import cats.syntax.semigroup.* import cats.syntax.either.* import cats.{~>, Comonad, Functor, Monad, Monoid, Order} +import cats.arrow.FunctionK import scribe.Logging class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( @@ -32,19 +33,47 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( ) extends Logging { type Err = AquaError[I, E, S] + type Warn = AquaWarning[S] type Ctx = NonEmptyMap[I, C] - type ValidatedCtx = ValidatedNec[Err, Ctx] - type ValidatedCtxT = ValidatedCtx => ValidatedCtx + type CompileWarnings = + [A] =>> Writer[Chain[Warn], A] + + type CompileResult = + [A] =>> EitherT[CompileWarnings, NonEmptyChain[Err], A] + + private val warningsK: semantics.ProcessWarnings ~> CompileWarnings = + new FunctionK[semantics.ProcessWarnings, CompileWarnings] { + + override def apply[A]( + fa: semantics.ProcessWarnings[A] + ): CompileWarnings[A] = + fa.mapWritten(_.map(AquaWarning.CompileWarning.apply)) + } + + extension (res: semantics.ProcessResult) { + + def toCompileResult: CompileResult[C] = + res + .leftMap(_.map(CompileError.apply)) + .mapK(warningsK) + } + + extension [A](res: ValidatedNec[SemanticError[S], A]) { + + def toCompileResult: CompileResult[A] = + res.toEither + .leftMap(_.map(CompileError.apply)) + .toEitherT[CompileWarnings] + } + + type CompiledCtx = CompileResult[Ctx] + type CompiledCtxT = CompiledCtx => CompiledCtx private def linkModules( - modules: Modules[ - I, - Err, - ValidatedCtxT - ], - cycleError: Linker.DepCycle[AquaModule[I, Err, ValidatedCtxT]] => Err - ): EitherNec[Err, Map[I, ValidatedCtx]] = { + modules: Modules[I, Err, CompiledCtxT], + cycleError: Linker.DepCycle[AquaModule[I, Err, CompiledCtxT]] => Err + ): EitherNec[Err, Map[I, CompiledCtx]] = { logger.trace("linking modules...") Linker @@ -52,7 +81,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( modules, cycleError, // By default, provide an empty context for this module's id - i => validNec(NonEmptyMap.one(i, Monoid.empty[C])) + i => NonEmptyMap.one(i, Monoid.empty[C]).asRight.toEitherT ) .toEither } @@ -60,45 +89,35 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( def compileRaw( sources: AquaSources[F, E, I], parser: I => String => ValidatedNec[ParserError[S], Ast[S]] - ): F[Validated[NonEmptyChain[Err], Map[I, ValidatedCtx]]] = { + ): F[ValidatedNec[Err, Map[I, ValidatedNec[Err, Ctx]]]] = { logger.trace("starting resolving sources...") new AquaParser[F, E, I, S](sources, parser) - .resolve[ValidatedCtx](mod => - context => { - // Context with prepared imports - context.andThen { ctx => - val imports = mod.imports.flatMap { case (fn, id) => + .resolve[CompiledCtx](mod => + context => + for { + // Context with prepared imports + ctx <- context + imports = mod.imports.flatMap { case (fn, id) => ctx.apply(id).map(fn -> _) } - val header = mod.body.head - - val process = for { - // To manage imports, exports run HeaderHandler - headerSem <- headerHandler - .sem(imports, header) - .toEither - .toEitherT[semantics.ProcessWarnings] - // Analyze the body, with prepared initial context - _ = logger.trace("semantic processing...") - processed <- semantics - .process( - mod.body, - headerSem.initCtx - ) - // Handle exports, declares - finalize the resulting context - rc <- headerSem - .finCtx(processed) - .toEither - .toEitherT[semantics.ProcessWarnings] - } yield NonEmptyMap.one(mod.id, rc) - - process - .leftMap(_.map(CompileError.apply)) - .toValidated - .value - } - } + header = mod.body.head + headerSem <- headerHandler + .sem(imports, header) + .toCompileResult + // Analyze the body, with prepared initial context + _ = logger.trace("semantic processing...") + processed <- semantics + .process( + mod.body, + headerSem.initCtx + ) + .toCompileResult + // Handle exports, declares - finalize the resulting context + rc <- headerSem + .finCtx(processed) + .toCompileResult + } yield NonEmptyMap.one(mod.id, rc) ) .subflatMap(modules => linkModules( @@ -106,6 +125,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( cycle => CycleError[I, E, S](cycle.map(_.id)) ) ) + .map(_.mapValues(_.toValidated.value).toMap) .toValidated } diff --git a/compiler/src/main/scala/aqua/compiler/AquaWarning.scala b/compiler/src/main/scala/aqua/compiler/AquaWarning.scala new file mode 100644 index 000000000..1e6a20214 --- /dev/null +++ b/compiler/src/main/scala/aqua/compiler/AquaWarning.scala @@ -0,0 +1,7 @@ +package aqua.compiler + +import aqua.semantics + +enum AquaWarning[S[_]] { + case CompileWarning(warning: semantics.SemanticWarning[S]) +} diff --git a/linker/src/main/scala/aqua/linker/Linker.scala b/linker/src/main/scala/aqua/linker/Linker.scala index 6d71b6ea9..2e6cc9611 100644 --- a/linker/src/main/scala/aqua/linker/Linker.scala +++ b/linker/src/main/scala/aqua/linker/Linker.scala @@ -105,7 +105,7 @@ object Linker extends Logging { val importKeys = m.dependsOn.keySet logger.debug(s"${m.id} dependsOn $importKeys") val deps: T => T = - importKeys.map(acc).foldLeft[T => T](identity) { case (fAcc, f) => + importKeys.map(acc).foldLeft(identity[T]) { case (fAcc, f) => logger.debug("COMBINING ONE TIME ") t => { logger.debug(s"call combine $t") From 2e0fa132db8c0c9ca272574dbfced045883f6e9e Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Thu, 21 Sep 2023 14:56:38 +0000 Subject: [PATCH 13/31] Refactor compiler --- .../scala/aqua/compiler/AquaCompiler.scala | 41 +++++++++++-------- .../scala/aqua/compiler/CompilerAPI.scala | 38 ++++++++--------- .../.js/src/main/scala/aqua/lsp/AquaLSP.scala | 38 +++++------------ .../src/main/scala/aqua/lsp/LSPCompiler.scala | 16 ++------ 4 files changed, 57 insertions(+), 76 deletions(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index d118cf6a8..86631d34b 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -73,25 +73,33 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( private def linkModules( modules: Modules[I, Err, CompiledCtxT], cycleError: Linker.DepCycle[AquaModule[I, Err, CompiledCtxT]] => Err - ): EitherNec[Err, Map[I, CompiledCtx]] = { + ): CompileResult[Map[I, C]] = { logger.trace("linking modules...") - Linker - .link( - modules, - cycleError, - // By default, provide an empty context for this module's id - i => NonEmptyMap.one(i, Monoid.empty[C]).asRight.toEitherT + for { + linked <- Linker + .link( + modules, + cycleError, + // By default, provide an empty context for this module's id + i => NonEmptyMap.one(i, Monoid.empty[C]).asRight.toEitherT + ) + .toEither + .toEitherT[CompileWarnings] + res <- EitherT( + linked.toList.traverse { case (id, ctx) => + ctx.map(_.apply(id).map(id -> _).get).toValidated + }.map(_.sequence.toEither) ) - .toEither + } yield res.toMap } def compileRaw( sources: AquaSources[F, E, I], parser: I => String => ValidatedNec[ParserError[S], Ast[S]] - ): F[ValidatedNec[Err, Map[I, ValidatedNec[Err, Ctx]]]] = { - + ): F[CompileResult[Map[I, C]]] = { logger.trace("starting resolving sources...") + new AquaParser[F, E, I, S](sources, parser) .resolve[CompiledCtx](mod => context => @@ -119,14 +127,15 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( .toCompileResult } yield NonEmptyMap.one(mod.id, rc) ) - .subflatMap(modules => - linkModules( - modules, - cycle => CycleError[I, E, S](cycle.map(_.id)) + .value + .map( + _.toEitherT[CompileWarnings].flatMap(modules => + linkModules( + modules, + cycle => CycleError[I, E, S](cycle.map(_.id)) + ) ) ) - .map(_.mapValues(_.toValidated.value).toMap) - .toValidated } } diff --git a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala index 77f74c868..1350be55d 100644 --- a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala +++ b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala @@ -22,6 +22,7 @@ import cats.syntax.functor.* import cats.syntax.monoid.* import cats.syntax.semigroup.* import cats.syntax.traverse.* +import cats.syntax.either.* import cats.{~>, Comonad, Monad, Monoid, Order} import scribe.Logging @@ -30,18 +31,13 @@ import scala.collection.MapView object CompilerAPI extends Logging { private def toAquaProcessed[I: Order, E, S[_]: Comonad]( - filesWithContext: Map[ - I, - ValidatedNec[AquaError[I, E, S], NonEmptyMap[I, RawContext]] - ] - ): ValidatedNec[AquaError[I, E, S], Chain[AquaProcessed[I]]] = { + filesWithContext: Map[I, RawContext] + ): Chain[AquaProcessed[I]] = { logger.trace("linking finished") - filesWithContext.values.toList - // Gather all RawContext in List inside ValidatedNec - .flatTraverse(_.map(_.toNel.toList)) + filesWithContext.toList // Process all contexts maintaining Cache - .traverse(_.traverse { case (i, rawContext) => + .traverse { case (i, rawContext) => for { cache <- State.get[AquaContext.Cache] _ = logger.trace(s"Going to prepare exports for $i...") @@ -49,9 +45,10 @@ object CompilerAPI extends Logging { _ = logger.trace(s"AquaProcessed prepared for $i") _ <- State.set(expCache) } yield AquaProcessed(i, exp) - }.runA(AquaContext.Cache())) + } + .runA(AquaContext.Cache()) // Convert result List to Chain - .map(_.map(Chain.fromSeq)) + .map(Chain.fromSeq) .value } @@ -82,7 +79,7 @@ object CompilerAPI extends Logging { for { compiledRaw <- compiler.compileRaw(sources, parser) - compiledV = compiledRaw.andThen(toAquaProcessed) + compiledV = compiledRaw.value.value.toValidated.map(toAquaProcessed) _ <- airValidator.init() result <- compiledV.traverse { compiled => compiled.traverse { ap => @@ -136,16 +133,15 @@ object CompilerAPI extends Logging { ): F[ValidatedNec[AquaError[I, E, S], Chain[AquaContext]]] = { val compiler = getAquaCompiler[F, E, I, S](config) - compiler - .compileRaw(sources, parser) - .map(_.andThen { filesWithContext => - toAquaProcessed(filesWithContext) - }) - .map(_.map { compiled => - compiled.map { ap => + val compiledRaw = compiler.compileRaw(sources, parser) + + compiledRaw.map( + _.value.value.toValidated + .map(toAquaProcessed) + .map(_.map { ap => logger.trace("generating output...") ap.context - } - }) + }) + ) } } diff --git a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala index 969cca515..9e0f92dc3 100644 --- a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala +++ b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala @@ -11,13 +11,14 @@ import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} import aqua.raw.ConstantRaw import aqua.semantics.{HeaderError, RulesViolated, WrongAST} import aqua.{AquaIO, SpanParser} + import cats.data.Validated.{invalidNec, validNec, Invalid, Valid} import cats.data.{NonEmptyChain, Validated} import cats.effect.IO +import cats.syntax.option.* import cats.effect.unsafe.implicits.global import fs2.io.file.{Files, Path} import scribe.Logging - import scala.concurrent.ExecutionContext.Implicits.global import scala.concurrent.Future import scala.scalajs.js @@ -142,34 +143,17 @@ object AquaLSP extends App with Logging { val config = AquaCompilerConf(ConstantRaw.defaultConstants(None)) val proc = for { - - res <- LSPCompiler - .compileToLsp[IO, AquaFileError, FileModuleId, FileSpan.F]( - sources, - SpanParser.parser, - config - ) + res <- LSPCompiler.compileToLsp[IO, AquaFileError, FileModuleId, FileSpan.F]( + sources, + SpanParser.parser, + config + ) } yield { - val fileRes: Validated[NonEmptyChain[ - AquaError[FileModuleId, AquaFileError, FileSpan.F] - ], LspContext[FileSpan.F]] = res - .andThen( - _.getOrElse( - pathId, - invalidNec( - SourcesError(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) - ) - ) - ) - .andThen( - _.get(pathId) - .map(l => validNec(l)) - .getOrElse( - invalidNec( - SourcesError(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) - ) - ) + val fileRes = res.andThen( + _.get(pathId).toValidNec( + SourcesError(Unresolvable(s"Unexpected. No file $pathStr in compiler results")) ) + ) logger.debug("Compilation done.") diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala index 8e0474e55..f4135bcdd 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala @@ -4,6 +4,7 @@ import aqua.compiler.{AquaCompiler, AquaCompilerConf, AquaError, AquaSources} import aqua.parser.{Ast, ParserError} import aqua.raw.RawContext import aqua.semantics.header.{HeaderHandler, HeaderSem} + import cats.data.Validated.validNec import cats.syntax.semigroup.* import cats.syntax.applicative.* @@ -11,6 +12,7 @@ import cats.syntax.flatMap.* import cats.syntax.functor.* import cats.syntax.monoid.* import cats.syntax.traverse.* +import cats.syntax.either.* import cats.{Comonad, Monad, Monoid, Order} import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} @@ -55,21 +57,11 @@ object LSPCompiler { sources: AquaSources[F, E, I], parser: I => String => ValidatedNec[ParserError[S], Ast[S]], config: AquaCompilerConf - ): F[Validated[NonEmptyChain[AquaError[I, E, S]], Map[I, Validated[NonEmptyChain[ - AquaError[I, E, S] - ], Map[I, LspContext[S]]]]]] = { + ): F[ValidatedNec[AquaError[I, E, S], Map[I, LspContext[S]]]] = { val compiler = getLspAquaCompiler[F, E, I, S](config) compiler .compileRaw(sources, parser) - .map { v => - v.map { innerMap => - innerMap.view.mapValues { vCtx => - vCtx.map { - _.toSortedMap.toMap - } - }.toMap - } - } + .map(_.value.value.toValidated) } } From 28d8d486de828b5a0e5885f74248cc867f883a4c Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Thu, 21 Sep 2023 15:17:28 +0000 Subject: [PATCH 14/31] Refactor types --- .../scala/aqua/compiler/AquaCompiler.scala | 36 +++++++++---------- .../main/scala/aqua/compiler/package.scala | 12 +++++++ 2 files changed, 28 insertions(+), 20 deletions(-) create mode 100644 compiler/src/main/scala/aqua/compiler/package.scala diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index 86631d34b..4faca1c75 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -33,27 +33,23 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( ) extends Logging { type Err = AquaError[I, E, S] - type Warn = AquaWarning[S] type Ctx = NonEmptyMap[I, C] - type CompileWarnings = - [A] =>> Writer[Chain[Warn], A] + type CompileWarns = [A] =>> CompileWarnings[S][A] + type CompileRes = [A] =>> CompileResult[I, E, S][A] - type CompileResult = - [A] =>> EitherT[CompileWarnings, NonEmptyChain[Err], A] - - private val warningsK: semantics.ProcessWarnings ~> CompileWarnings = - new FunctionK[semantics.ProcessWarnings, CompileWarnings] { + private val warningsK: semantics.ProcessWarnings ~> CompileWarns = + new FunctionK[semantics.ProcessWarnings, CompileWarns] { override def apply[A]( fa: semantics.ProcessWarnings[A] - ): CompileWarnings[A] = + ): CompileWarns[A] = fa.mapWritten(_.map(AquaWarning.CompileWarning.apply)) } extension (res: semantics.ProcessResult) { - def toCompileResult: CompileResult[C] = + def toCompileRes: CompileRes[C] = res .leftMap(_.map(CompileError.apply)) .mapK(warningsK) @@ -61,19 +57,19 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( extension [A](res: ValidatedNec[SemanticError[S], A]) { - def toCompileResult: CompileResult[A] = + def toCompileRes: CompileRes[A] = res.toEither .leftMap(_.map(CompileError.apply)) - .toEitherT[CompileWarnings] + .toEitherT[CompileWarns] } - type CompiledCtx = CompileResult[Ctx] + type CompiledCtx = CompileRes[Ctx] type CompiledCtxT = CompiledCtx => CompiledCtx private def linkModules( modules: Modules[I, Err, CompiledCtxT], cycleError: Linker.DepCycle[AquaModule[I, Err, CompiledCtxT]] => Err - ): CompileResult[Map[I, C]] = { + ): CompileRes[Map[I, C]] = { logger.trace("linking modules...") for { @@ -85,7 +81,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( i => NonEmptyMap.one(i, Monoid.empty[C]).asRight.toEitherT ) .toEither - .toEitherT[CompileWarnings] + .toEitherT[CompileWarns] res <- EitherT( linked.toList.traverse { case (id, ctx) => ctx.map(_.apply(id).map(id -> _).get).toValidated @@ -97,7 +93,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( def compileRaw( sources: AquaSources[F, E, I], parser: I => String => ValidatedNec[ParserError[S], Ast[S]] - ): F[CompileResult[Map[I, C]]] = { + ): F[CompileRes[Map[I, C]]] = { logger.trace("starting resolving sources...") new AquaParser[F, E, I, S](sources, parser) @@ -112,7 +108,7 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( header = mod.body.head headerSem <- headerHandler .sem(imports, header) - .toCompileResult + .toCompileRes // Analyze the body, with prepared initial context _ = logger.trace("semantic processing...") processed <- semantics @@ -120,16 +116,16 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( mod.body, headerSem.initCtx ) - .toCompileResult + .toCompileRes // Handle exports, declares - finalize the resulting context rc <- headerSem .finCtx(processed) - .toCompileResult + .toCompileRes } yield NonEmptyMap.one(mod.id, rc) ) .value .map( - _.toEitherT[CompileWarnings].flatMap(modules => + _.toEitherT[CompileWarns].flatMap(modules => linkModules( modules, cycle => CycleError[I, E, S](cycle.map(_.id)) diff --git a/compiler/src/main/scala/aqua/compiler/package.scala b/compiler/src/main/scala/aqua/compiler/package.scala new file mode 100644 index 000000000..4acac30b0 --- /dev/null +++ b/compiler/src/main/scala/aqua/compiler/package.scala @@ -0,0 +1,12 @@ +package aqua + +import cats.data.{Chain, EitherT, NonEmptyChain, Writer} + +package object compiler { + + type CompileWarnings[S[_]] = + [A] =>> Writer[Chain[AquaWarning[S]], A] + + type CompileResult[I, E, S[_]] = + [A] =>> EitherT[CompileWarnings[S], NonEmptyChain[AquaError[I, E, S]], A] +} From d936fd581da96cb446d358a87774366af4106bb9 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 08:37:53 +0000 Subject: [PATCH 15/31] Refactor retunr types --- .../main/scala/aqua/api/APICompilation.scala | 9 ++- .../main/scala/aqua/run/FuncCompiler.scala | 20 +++++-- .../scala/aqua/compiler/CompilerAPI.scala | 57 ++++++------------- .../aqua/compiler/AquaCompilerSpec.scala | 7 ++- .../main/scala/aqua/logging/LogLevels.scala | 2 +- 5 files changed, 44 insertions(+), 51 deletions(-) diff --git a/api/api/src/main/scala/aqua/api/APICompilation.scala b/api/api/src/main/scala/aqua/api/APICompilation.scala index 36b6045ee..f4c30dd91 100644 --- a/api/api/src/main/scala/aqua/api/APICompilation.scala +++ b/api/api/src/main/scala/aqua/api/APICompilation.scala @@ -33,6 +33,7 @@ import cats.effect.IO import cats.effect.unsafe.implicits.global import cats.syntax.show.* import cats.syntax.traverse.* +import cats.syntax.either.* import fs2.io.file.{Files, Path} import scribe.Logging @@ -98,6 +99,7 @@ object APICompilation { val path = Path(pathStr) val sources = new AquaFileSources[IO](path, imports.map(Path.apply)) + compileRaw( aquaConfig, sources, @@ -121,6 +123,7 @@ object APICompilation { IO.pure(Valid(Chain.one((FileModuleId(path), input)))) } } + compileRaw( aquaConfig, strSources, @@ -153,15 +156,15 @@ object APICompilation { override def validate(airs: List[AirFunction]): IO[ValidatedNec[String, Unit]] = Applicative[IO].pure(validNec(())) }, - new Backend.Transform: + new Backend.Transform { override def transform(ex: AquaContext): AquaRes = Transform.contextRes(ex, transformConfig) override def generate(aqua: AquaRes): Seq[Generated] = backend.generate(aqua) - , + }, config ) - .map(_.leftMap(_.map(_.show).distinct)) + .map(_.leftMap(_.map(_.show).distinct).value.value.toValidated) }.map(_.leftMap(NonEmptyChain.fromNonEmptyList).andThen(identity)) } } diff --git a/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala b/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala index cfbe8be2f..de2730885 100644 --- a/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala +++ b/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala @@ -1,7 +1,7 @@ package aqua.run import aqua.ErrorRendering.given -import aqua.compiler.{AquaCompiler, AquaCompilerConf, CompilerAPI} +import aqua.compiler.{AquaCompiler, AquaCompilerConf, CompileResult, CompilerAPI} import aqua.files.{AquaFileSources, FileModuleId} import aqua.{AquaIO, SpanParser} import aqua.io.{AquaFileError, AquaPath, PackagePath, Prelude} @@ -21,6 +21,8 @@ import cats.syntax.monad.* import cats.syntax.show.* import cats.syntax.traverse.* import cats.syntax.option.* +import cats.syntax.either.* +import cats.syntax.validated.* import fs2.io.file.{Files, Path} import scribe.Logging @@ -44,7 +46,13 @@ class FuncCompiler[F[_]: Files: AquaIO: Async]( SpanParser.parser, config ) - .map(_.leftMap(_.map(_.show))) + .map( + _.value.value + .leftMap( + _.map(_.show) + ) + .toValidated + ) } private def compileBuiltins() = { @@ -65,14 +73,14 @@ class FuncCompiler[F[_]: Files: AquaIO: Async]( // compile builtins and add it to context builtinsV <- if (withBuiltins) compileBuiltins() - else validNec[String, Chain[AquaContext]](Chain.empty).pure[F] - compileResult <- input.map { ap => + else Chain.empty.validNec.pure + compileResult <- input.traverse { ap => // compile only context to wrap and call function later Clock[F].timed( ap.getPath().flatMap(p => compileToContext(p, preludeImports ++ imports)) ) - }.getOrElse((Duration.Zero, validNec[String, Chain[AquaContext]](Chain.empty)).pure[F]) - (compileTime, contextV) = compileResult + } + (compileTime, contextV) = compileResult.orEmpty } yield { logger.debug(s"Compile time: ${compileTime.toMillis}ms") // add builtins to the end of context diff --git a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala index 1350be55d..b472d99d0 100644 --- a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala +++ b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala @@ -74,70 +74,49 @@ object CompilerAPI extends Logging { airValidator: AirValidator[F], backend: Backend.Transform, config: AquaCompilerConf - ): F[ValidatedNec[AquaError[I, E, S], Chain[AquaCompiled[I]]]] = { + ): F[CompileResult[I, E, S][Chain[AquaCompiled[I]]]] = { val compiler = getAquaCompiler[F, E, I, S](config) for { compiledRaw <- compiler.compileRaw(sources, parser) - compiledV = compiledRaw.value.value.toValidated.map(toAquaProcessed) + compiledV = compiledRaw.map(toAquaProcessed) _ <- airValidator.init() - result <- compiledV.traverse { compiled => + result <- compiledV.flatTraverse { compiled => compiled.traverse { ap => logger.trace("generating output...") val res = backend.transform(ap.context) - val compiled = backend.generate(res) + val generated = backend.generate(res) + val air = generated.toList.flatMap(_.air) + val compiled = AquaCompiled( + sourceId = ap.id, + compiled = generated, + funcsCount = res.funcs.length.toInt, + servicesCount = res.services.length.toInt + ) + airValidator - .validate( - compiled.toList.flatMap(_.air) - ) + .validate(air) .map( _.leftMap(errs => AirValidationError(errs): AquaError[I, E, S]) - .as( - AquaCompiled(ap.id, compiled, res.funcs.length.toInt, res.services.length.toInt) - ) + .as(compiled) .toValidatedNec ) - }.map(_.sequence) - }.map(_.andThen(identity)) // There is no flatTraverse for Validated + }.map(_.sequence.toEither.toEitherT) + } } yield result } - def compileTo[F[_]: Monad, E, I: Order, S[_]: Comonad, T]( - sources: AquaSources[F, E, I], - parser: I => String => ValidatedNec[ParserError[S], Ast[S]], - airValidator: AirValidator[F], - backend: Backend.Transform, - config: AquaCompilerConf, - write: AquaCompiled[I] => F[Seq[Validated[E, T]]] - ): F[ValidatedNec[AquaError[I, E, S], Chain[T]]] = - compile[F, E, I, S](sources, parser, airValidator, backend, config) - .flatMap( - _.traverse(compiled => - compiled.toList.flatTraverse { ac => - write(ac).map( - _.toList.map( - _.bimap( - e => OutputError(ac, e): AquaError[I, E, S], - Chain.one - ).toValidatedNec - ) - ) - }.map(_.foldA) - ).map(_.andThen(identity)) // There is no flatTraverse for Validated - ) - def compileToContext[F[_]: Monad, E, I: Order, S[_]: Comonad]( sources: AquaSources[F, E, I], parser: I => String => ValidatedNec[ParserError[S], Ast[S]], config: AquaCompilerConf - ): F[ValidatedNec[AquaError[I, E, S], Chain[AquaContext]]] = { + ): F[CompileResult[I, E, S][Chain[AquaContext]]] = { val compiler = getAquaCompiler[F, E, I, S](config) val compiledRaw = compiler.compileRaw(sources, parser) compiledRaw.map( - _.value.value.toValidated - .map(toAquaProcessed) + _.map(toAquaProcessed) .map(_.map { ap => logger.trace("generating output...") ap.context diff --git a/compiler/src/test/scala/aqua/compiler/AquaCompilerSpec.scala b/compiler/src/test/scala/aqua/compiler/AquaCompilerSpec.scala index dccf940f0..4022b7d5c 100644 --- a/compiler/src/test/scala/aqua/compiler/AquaCompilerSpec.scala +++ b/compiler/src/test/scala/aqua/compiler/AquaCompilerSpec.scala @@ -30,6 +30,7 @@ import cats.data.{Chain, NonEmptyChain, NonEmptyMap, Validated, ValidatedNec} import cats.instances.string.* import cats.syntax.show.* import cats.syntax.option.* +import cats.syntax.either.* class AquaCompilerSpec extends AnyFlatSpec with Matchers { import ModelBuilder.* @@ -59,8 +60,11 @@ class AquaCompilerSpec extends AnyFlatSpec with Matchers { id => txt => Parser.parse(Parser.parserSchema)(txt), AquaCompilerConf(ConstantRaw.defaultConstants(None)) ) + .value + .value + .toValidated - "aqua compiler" should "compile a simple snipped to the right context" in { + "aqua compiler" should "compile a simple snippet to the right context" in { val res = compileToContext( Map( @@ -93,7 +97,6 @@ class AquaCompilerSpec extends AnyFlatSpec with Matchers { val const = ctx.allValues.get("X") const.nonEmpty should be(true) const.get should be(LiteralModel.number(5)) - } def through(peer: ValueModel) = diff --git a/utils/logging/src/main/scala/aqua/logging/LogLevels.scala b/utils/logging/src/main/scala/aqua/logging/LogLevels.scala index 468e0f2ce..01edf71fc 100644 --- a/utils/logging/src/main/scala/aqua/logging/LogLevels.scala +++ b/utils/logging/src/main/scala/aqua/logging/LogLevels.scala @@ -32,7 +32,7 @@ object LogLevels { name: String, level: String, logLevels: LogLevels - ): Validated[NonEmptyList[String], LogLevels] = { + ): ValidatedNel[String, LogLevels] = { levelFromString(level).andThen { level => name.trim().toLowerCase() match { case "compiler" => From 76b2e2b24fea5a584a1ed301406c02421d3db0b5 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 10:36:17 +0000 Subject: [PATCH 16/31] Return warnings --- api/api/.js/src/main/scala/api/AquaAPI.scala | 121 ++++++++++-------- .../main/scala/api/types/OutputTypes.scala | 39 ++++-- .../.jvm/src/main/scala/aqua/api/Test.scala | 38 ++++-- .../main/scala/aqua/api/APICompilation.scala | 81 +++++++----- api/api/src/main/scala/aqua/api/package.scala | 17 +++ .../main/scala/aqua/run/FuncCompiler.scala | 38 +++--- .../{ErrorRendering.scala => Rendering.scala} | 18 ++- .../scala/aqua/parser/lift/FileSpan.scala | 4 +- 8 files changed, 219 insertions(+), 137 deletions(-) create mode 100644 api/api/src/main/scala/aqua/api/package.scala rename io/src/main/scala/aqua/{ErrorRendering.scala => Rendering.scala} (87%) diff --git a/api/api/.js/src/main/scala/api/AquaAPI.scala b/api/api/.js/src/main/scala/api/AquaAPI.scala index d77f317a8..ea25baf58 100644 --- a/api/api/.js/src/main/scala/api/AquaAPI.scala +++ b/api/api/.js/src/main/scala/api/AquaAPI.scala @@ -1,9 +1,9 @@ package api import api.types.{AquaConfig, AquaFunction, CompilationResult, GeneratedSource, Input} -import aqua.ErrorRendering.given +import aqua.Rendering.given import aqua.raw.value.ValueRaw -import aqua.api.{APICompilation, AquaAPIConfig} +import aqua.api.{APICompilation, APIResult, AquaAPIConfig} import aqua.api.TargetType.* import aqua.backend.air.AirBackend import aqua.backend.{AirFunction, Backend, Generated} @@ -41,6 +41,7 @@ import cats.effect.IO import cats.effect.unsafe.implicits.global import cats.syntax.show.* import cats.syntax.traverse.* +import cats.syntax.either.* import fs2.io.file.{Files, Path} import scribe.Logging @@ -100,15 +101,17 @@ object AquaAPI extends App with Logging { case JavaScriptType => JavaScriptBackend() } - extension (res: IO[ValidatedNec[String, Chain[AquaCompiled[FileModuleId]]]]) - def toResult: IO[CompilationResult] = res.map { compiledV => - compiledV.map { compiled => - config.targetType match { - case AirType => generatedToAirResult(compiled) - case TypeScriptType => compiledToTsSourceResult(compiled) - case JavaScriptType => compiledToJsSourceResult(compiled) - } - }.leftMap(errorsToResult).merge + extension (res: APIResult[Chain[AquaCompiled[FileModuleId]]]) + def toResult: CompilationResult = { + val (warnings, result) = res.value.run + + result.map { compiled => + (config.targetType match { + case AirType => generatedToAirResult + case TypeScriptType => compiledToTsSourceResult + case JavaScriptType => compiledToJsSourceResult + }).apply(compiled, warnings) + }.leftMap(errorsToResult(_, warnings)).merge } input match { @@ -120,7 +123,7 @@ object AquaAPI extends App with Logging { config, backend ) - .toResult + .map(_.toResult) case p: types.Path => APICompilation .compilePath( @@ -129,7 +132,7 @@ object AquaAPI extends App with Logging { config, backend ) - .toResult + .map(_.toResult) } } @@ -140,12 +143,17 @@ object AquaAPI extends App with Logging { case p: types.Path => p.path } - extension (res: IO[ValidatedNec[String, (FunctionDef, String)]]) - def callToResult: IO[CompilationResult] = res.map( - _.map { case (definitions, air) => - CompilationResult.result(call = Some(AquaFunction(FunctionDefJs(definitions), air))) - }.leftMap(errorsToResult).merge - ) + extension (res: APIResult[(FunctionDef, String)]) + def callToResult: CompilationResult = { + val (warnings, result) = res.value.run + + result.map { case (definitions, air) => + CompilationResult.result( + call = Some(AquaFunction(FunctionDefJs(definitions), air)), + warnings = warnings.toList + ) + }.leftMap(errorsToResult(_, warnings)).merge + } APICompilation .compileCall( @@ -155,56 +163,65 @@ object AquaAPI extends App with Logging { config, vr => VarJson.checkDataGetServices(vr, Some(call.arguments)).map(_._1) ) - .callToResult + .map(_.callToResult) } - private def errorsToResult(errors: NonEmptyChain[String]): CompilationResult = { - CompilationResult.errs(errors.toChain.toList) - } - - extension (res: List[GeneratedSource]) - - def toSourcesResult: CompilationResult = - CompilationResult.result(sources = res.toJSArray) + private def errorsToResult( + errors: NonEmptyChain[String], + warnings: Chain[String] + ): CompilationResult = + CompilationResult.errs( + errors.toChain.toList, + warnings.toList + ) private def compiledToTsSourceResult( - compiled: Chain[AquaCompiled[FileModuleId]] + compiled: Chain[AquaCompiled[FileModuleId]], + warnings: Chain[String] ): CompilationResult = - compiled.toList - .flatMap(c => - c.compiled - .find(_.suffix == TypeScriptBackend.ext) - .map(_.content) - .map(GeneratedSource.tsSource(c.sourceId.toString, _)) - ) - .toSourcesResult + CompilationResult.result( + sources = compiled.toList + .flatMap(c => + c.compiled + .find(_.suffix == TypeScriptBackend.ext) + .map(_.content) + .map(GeneratedSource.tsSource(c.sourceId.toString, _)) + ), + warnings = warnings.toList + ) private def compiledToJsSourceResult( - compiled: Chain[AquaCompiled[FileModuleId]] + compiled: Chain[AquaCompiled[FileModuleId]], + warnings: Chain[String] ): CompilationResult = - compiled.toList.flatMap { c => - for { - dtsContent <- c.compiled - .find(_.suffix == JavaScriptBackend.dtsExt) - .map(_.content) - jsContent <- c.compiled - .find(_.suffix == JavaScriptBackend.ext) - .map(_.content) - } yield GeneratedSource.jsSource(c.sourceId.toString, jsContent, dtsContent) - }.toSourcesResult + CompilationResult.result( + sources = compiled.toList.flatMap { c => + for { + dtsContent <- c.compiled + .find(_.suffix == JavaScriptBackend.dtsExt) + .map(_.content) + jsContent <- c.compiled + .find(_.suffix == JavaScriptBackend.ext) + .map(_.content) + } yield GeneratedSource.jsSource(c.sourceId.toString, jsContent, dtsContent) + }, + warnings = warnings.toList + ) private def generatedToAirResult( - compiled: Chain[AquaCompiled[FileModuleId]] + compiled: Chain[AquaCompiled[FileModuleId]], + warnings: Chain[String] ): CompilationResult = { val generated = compiled.toList.flatMap(_.compiled) val serviceDefs = generated.flatMap(_.services).map(s => s.name -> ServiceDefJs(s)) val functions = generated.flatMap( - _.air.map(as => (as.name, AquaFunction(FunctionDefJs(as.funcDef), as.air))) + _.air.map(as => as.name -> AquaFunction(FunctionDefJs(as.funcDef), as.air)) ) CompilationResult.result( - js.Dictionary.apply(serviceDefs: _*), - js.Dictionary.apply(functions: _*) + services = serviceDefs.toMap, + functions = functions.toMap, + warnings = warnings.toList ) } diff --git a/api/api/.js/src/main/scala/api/types/OutputTypes.scala b/api/api/.js/src/main/scala/api/types/OutputTypes.scala index 9954510f2..d0d8fbc21 100644 --- a/api/api/.js/src/main/scala/api/types/OutputTypes.scala +++ b/api/api/.js/src/main/scala/api/types/OutputTypes.scala @@ -2,13 +2,14 @@ package api.types import aqua.js.{FunctionDefJs, ServiceDefJs} import aqua.model.transform.TransformConfig -import cats.data.Validated.{Invalid, Valid, invalidNec, validNec} +import cats.data.Validated.{invalidNec, validNec, Invalid, Valid} import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} import scala.scalajs.js import scala.scalajs.js.JSConverters.* import scala.scalajs.js.annotation.{JSExport, JSExportTopLevel} import scala.scalajs.js.| +import aqua.parser.lexer.Token.func @JSExportTopLevel("AquaFunction") case class AquaFunction( @@ -32,7 +33,9 @@ case class GeneratedSource( object GeneratedSource { def tsSource(name: String, tsSource: String) = new GeneratedSource(name, tsSource, null, null) - def jsSource(name: String, jsSource: String, tsTypes: String) = new GeneratedSource(name, null, jsSource, tsTypes) + + def jsSource(name: String, jsSource: String, tsTypes: String) = + new GeneratedSource(name, null, jsSource, tsTypes) } @JSExportTopLevel("CompilationResult") @@ -46,21 +49,39 @@ class CompilationResult( @JSExport val generatedSources: js.Array[GeneratedSource], @JSExport - val errors: js.Array[String] + val errors: js.Array[String], + @JSExport + val warnings: js.Array[String] ) object CompilationResult { def result( - services: js.Dictionary[ServiceDefJs] = js.Dictionary(), - functions: js.Dictionary[AquaFunction] = js.Dictionary(), + services: Map[String, ServiceDefJs] = Map.empty, + functions: Map[String, AquaFunction] = Map.empty, call: Option[AquaFunction] = None, - sources: js.Array[GeneratedSource] = js.Array() + sources: List[GeneratedSource] = List.empty, + warnings: List[String] = List.empty ): CompilationResult = - new CompilationResult(services, functions, call.orNull, sources, js.Array()) + new CompilationResult( + services.toJSDictionary, + functions.toJSDictionary, + call.orNull, + sources.toJSArray, + js.Array(), + warnings.toJSArray + ) def errs( - errors: List[String] + errors: List[String] = List.empty, + warnings: List[String] = List.empty ): CompilationResult = - CompilationResult(js.Dictionary(), js.Dictionary(), null, null, errors.toJSArray) + new CompilationResult( + js.Dictionary.empty, + js.Dictionary.empty, + null, + null, + errors.toJSArray, + warnings.toJSArray + ) } diff --git a/api/api/.jvm/src/main/scala/aqua/api/Test.scala b/api/api/.jvm/src/main/scala/aqua/api/Test.scala index 4b3be82a2..23d8f8fe3 100644 --- a/api/api/.jvm/src/main/scala/aqua/api/Test.scala +++ b/api/api/.jvm/src/main/scala/aqua/api/Test.scala @@ -9,7 +9,7 @@ import cats.data.Chain import cats.data.Validated.{Invalid, Valid} import cats.effect.{IO, IOApp} import fs2.io.file.{Files, Path} -import fs2.{Stream, text} +import fs2.{text, Stream} object Test extends IOApp.Simple { @@ -21,19 +21,31 @@ object Test extends IOApp.Simple { AquaAPIConfig(targetType = TypeScriptType), TypeScriptBackend(false, "IFluenceClient$$") ) - .flatMap { - case Valid(res) => - val content = res.get(0).get.compiled.head.content - val targetPath = Path("./target/antithesis.ts") + .flatMap { res => + val (warnings, result) = res.value.run - Stream.emit(content) - .through(text.utf8.encode) - .through(Files[IO].writeAll(targetPath)) - .attempt - .compile - .last.flatMap(_ => IO.delay(println(s"File: ${targetPath.absolute.normalize}"))) - case Invalid(e) => - IO.delay(println(e)) + IO.delay { + warnings.toList.foreach(println) + } *> result.fold( + errors => + IO.delay { + errors.toChain.toList.foreach(println) + }, + compiled => { + val content = compiled.get(0).get.compiled.head.content + val targetPath = Path("./target/antithesis.ts") + + Stream + .emit(content) + .through(text.utf8.encode) + .through(Files[IO].writeAll(targetPath)) + .attempt + .compile + .last *> IO.delay( + println(s"File: ${targetPath.absolute.normalize}") + ) + } + ) } } diff --git a/api/api/src/main/scala/aqua/api/APICompilation.scala b/api/api/src/main/scala/aqua/api/APICompilation.scala index f4c30dd91..a2e4b2971 100644 --- a/api/api/src/main/scala/aqua/api/APICompilation.scala +++ b/api/api/src/main/scala/aqua/api/APICompilation.scala @@ -1,7 +1,8 @@ package aqua.api -import aqua.ErrorRendering.given +import aqua.Rendering.given import aqua.raw.value.ValueRaw +import aqua.raw.ConstantRaw import aqua.api.AquaAPIConfig import aqua.backend.{AirFunction, Backend, Generated} import aqua.compiler.* @@ -23,7 +24,8 @@ import aqua.model.AquaContext import aqua.res.AquaRes import cats.Applicative -import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} +import cats.~> +import cats.data.{Chain, EitherT, NonEmptyChain, Validated, ValidatedNec, Writer, ValidatedNel, NonEmptyList} import cats.data.Validated.{invalid, invalidNec, validNec, Invalid, Valid} import cats.syntax.applicative.* import cats.syntax.apply.* @@ -35,23 +37,43 @@ import cats.syntax.show.* import cats.syntax.traverse.* import cats.syntax.either.* import fs2.io.file.{Files, Path} -import scribe.Logging +import scribe.{Logging, Level} object APICompilation { + extension [A, C <: NonEmptyChain[String] | NonEmptyList[String]](v: Validated[C, A]) { + def toResult: APIResult[A] = + v.toEither.leftMap{ + case v: NonEmptyChain[String] => v + case v: NonEmptyList[String] => + NonEmptyChain.fromNonEmptyList(v) + }.toEitherT + } + + extension [A](v: CompileResult[FileModuleId, AquaFileError, FileSpan.F][A]) { + def toResult: APIResult[A] = + v.leftMap(_.map(_.show)).mapK( + new (CompileWarnings[FileSpan.F] ~> APIWarnings) { + override def apply[A](w: CompileWarnings[FileSpan.F][A]): APIWarnings[A] = + w.mapWritten(_.map(_.show)) + } + ) + } + def compileCall( functionStr: String, pathStr: String, imports: List[String], aquaConfig: AquaAPIConfig, fillWithTypes: List[ValueRaw] => ValidatedNec[String, List[ValueRaw]] - ): IO[ValidatedNec[String, (FunctionDef, String)]] = { + ): IO[APIResult[(FunctionDef, String)]] = { given AquaIO[IO] = new AquaFilesIO[IO] ( LogLevels.levelFromString(aquaConfig.logLevel), Constants.parse(aquaConfig.constants) - ).mapN { (level, constants) => + ).tupled.toResult.flatTraverse { + case (level, constants) => val transformConfig = aquaConfig.getTransformConfig.copy(constants = constants) @@ -62,30 +84,24 @@ object APICompilation { imports.map(Path.apply), transformConfig ).compile().map { contextV => - contextV.andThen { context => - CliFunc + for { + context <- contextV.toResult + cliFunc <- CliFunc .fromString(functionStr) - .leftMap(errs => NonEmptyChain.fromNonEmptyList(errs)) - .andThen { cliFunc => - FuncCompiler.findFunction(context, cliFunc).andThen { arrow => - fillWithTypes(cliFunc.args).andThen { argsWithTypes => - val func = cliFunc.copy(args = argsWithTypes) - val preparer = new RunPreparer( + .toResult + arrow <- FuncCompiler + .findFunction(context, cliFunc) + .toResult + argsWithTypes <- fillWithTypes(cliFunc.args).toResult + func = cliFunc.copy(args = argsWithTypes) + preparer = new RunPreparer( func, arrow, transformConfig ) - preparer.prepare().map { ci => - (ci.definitions, ci.air) - } - } - } - } - }.leftMap(_.map(_.show).distinct) + ci <- preparer.prepare().toResult + } yield ci.definitions -> ci.air } - } match { - case Valid(pr) => pr - case Invalid(errs) => IO.pure(Invalid(NonEmptyChain.fromNonEmptyList(errs))) } } @@ -94,7 +110,7 @@ object APICompilation { imports: List[String], aquaConfig: AquaAPIConfig, backend: Backend - ): IO[ValidatedNec[String, Chain[AquaCompiled[FileModuleId]]]] = { + ): IO[APIResult[Chain[AquaCompiled[FileModuleId]]]] = { given AquaIO[IO] = new AquaFilesIO[IO] val path = Path(pathStr) @@ -112,7 +128,7 @@ object APICompilation { imports: List[String], aquaConfig: AquaAPIConfig, backend: Backend - ): IO[ValidatedNec[String, Chain[AquaCompiled[FileModuleId]]]] = { + ): IO[APIResult[Chain[AquaCompiled[FileModuleId]]]] = { given AquaIO[IO] = new AquaFilesIO[IO] val path = Path("") @@ -135,14 +151,13 @@ object APICompilation { aquaConfig: AquaAPIConfig, sources: AquaSources[IO, AquaFileError, FileModuleId], backend: Backend - ): IO[ValidatedNec[String, Chain[AquaCompiled[FileModuleId]]]] = { - + ): IO[APIResult[Chain[AquaCompiled[FileModuleId]]]] = ( LogLevels.levelFromString(aquaConfig.logLevel), Constants.parse(aquaConfig.constants) - ).traverseN { (level, constants) => - - LogFormatter.initLogger(Some(level)) + ).tupled.toResult.flatTraverse { + case (level, constants) => + LogFormatter.initLogger(Some(level)) val transformConfig = aquaConfig.getTransformConfig val config = AquaCompilerConf(constants ++ transformConfig.constantsList) @@ -163,8 +178,6 @@ object APICompilation { override def generate(aqua: AquaRes): Seq[Generated] = backend.generate(aqua) }, config - ) - .map(_.leftMap(_.map(_.show).distinct).value.value.toValidated) - }.map(_.leftMap(NonEmptyChain.fromNonEmptyList).andThen(identity)) - } + ).map(_.toResult) + } } diff --git a/api/api/src/main/scala/aqua/api/package.scala b/api/api/src/main/scala/aqua/api/package.scala new file mode 100644 index 000000000..ed61948ae --- /dev/null +++ b/api/api/src/main/scala/aqua/api/package.scala @@ -0,0 +1,17 @@ +package aqua + +import cats.data.{Chain, EitherT, NonEmptyChain, Writer} + +package object api { + + type APIWarnings = [A] =>> Writer[ + Chain[String], + A + ] + + type APIResult = [A] =>> EitherT[ + APIWarnings, + NonEmptyChain[String], + A + ] +} diff --git a/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala b/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala index de2730885..134e6e46d 100644 --- a/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala +++ b/aqua-run/src/main/scala/aqua/run/FuncCompiler.scala @@ -1,6 +1,6 @@ package aqua.run -import aqua.ErrorRendering.given +import aqua.Rendering.given import aqua.compiler.{AquaCompiler, AquaCompilerConf, CompileResult, CompilerAPI} import aqua.files.{AquaFileSources, FileModuleId} import aqua.{AquaIO, SpanParser} @@ -23,6 +23,7 @@ import cats.syntax.traverse.* import cats.syntax.option.* import cats.syntax.either.* import cats.syntax.validated.* +import cats.syntax.apply.* import fs2.io.file.{Files, Path} import scribe.Logging @@ -34,46 +35,37 @@ class FuncCompiler[F[_]: Files: AquaIO: Async]( transformConfig: TransformConfig ) extends Logging { + type Result = [A] =>> CompileResult[FileModuleId, AquaFileError, FileSpan.F][A] + private def compileToContext( path: Path, imports: List[Path], config: AquaCompilerConf = AquaCompilerConf(transformConfig.constantsList) - ) = { + ): F[Result[Chain[AquaContext]]] = { val sources = new AquaFileSources[F](path, imports) - CompilerAPI - .compileToContext[F, AquaFileError, FileModuleId, FileSpan.F]( - sources, - SpanParser.parser, - config - ) - .map( - _.value.value - .leftMap( - _.map(_.show) - ) - .toValidated - ) + CompilerAPI.compileToContext[F, AquaFileError, FileModuleId, FileSpan.F]( + sources, + SpanParser.parser, + config + ) } - private def compileBuiltins() = { + private def compileBuiltins(): F[Result[Chain[AquaContext]]] = for { path <- PackagePath.builtin.getPath() context <- compileToContext(path, Nil) - } yield { - context - } - } + } yield context // Compile and get only one function def compile( preludeImports: List[Path] = Nil, withBuiltins: Boolean = false - ): F[ValidatedNec[String, Chain[AquaContext]]] = { + ): F[Result[Chain[AquaContext]]] = { for { // compile builtins and add it to context builtinsV <- if (withBuiltins) compileBuiltins() - else Chain.empty.validNec.pure + else Chain.empty.pure[Result].pure[F] compileResult <- input.traverse { ap => // compile only context to wrap and call function later Clock[F].timed( @@ -84,7 +76,7 @@ class FuncCompiler[F[_]: Files: AquaIO: Async]( } yield { logger.debug(s"Compile time: ${compileTime.toMillis}ms") // add builtins to the end of context - contextV.andThen(c => builtinsV.map(bc => c ++ bc)) + (contextV, builtinsV).mapN(_ ++ _) } } } diff --git a/io/src/main/scala/aqua/ErrorRendering.scala b/io/src/main/scala/aqua/Rendering.scala similarity index 87% rename from io/src/main/scala/aqua/ErrorRendering.scala rename to io/src/main/scala/aqua/Rendering.scala index 59ac2fab4..9f7d3ba30 100644 --- a/io/src/main/scala/aqua/ErrorRendering.scala +++ b/io/src/main/scala/aqua/Rendering.scala @@ -6,21 +6,21 @@ import aqua.files.FileModuleId import aqua.io.AquaFileError import aqua.parser.lift.{FileSpan, Span} import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} -import aqua.semantics.{HeaderError, RulesViolated, WrongAST} +import aqua.semantics.{HeaderError, RulesViolated, SemanticWarning, WrongAST} import cats.parse.LocationMap import cats.parse.Parser.Expectation import cats.parse.Parser.Expectation.* import cats.{Eval, Show} -object ErrorRendering { +object Rendering { - def showForConsole(errorType: String, span: FileSpan, messages: List[String]): String = + def showForConsole(typ: String, span: FileSpan, messages: List[String]): String = span .focus(3) .map( _.toConsoleStr( - errorType, + typ, messages, Console.RED ) @@ -31,6 +31,16 @@ object ErrorRendering { ) ) + Console.RESET + "\n" + given Show[AquaWarning[FileSpan.F]] = Show.show { case AquaWarning.CompileWarning(warning) => + warning match { + case SemanticWarning(token, hints) => + token.unit._1 + .focus(0) + .map(_.toConsoleStr("Warning", hints, Console.YELLOW)) + .getOrElse("(Dup warning, but offset is beyond the script)") + } + } + given Show[AquaError[FileModuleId, AquaFileError, FileSpan.F]] = Show.show { case AquaParserError(err) => err match { diff --git a/parser/src/main/scala/aqua/parser/lift/FileSpan.scala b/parser/src/main/scala/aqua/parser/lift/FileSpan.scala index 78cba615f..b1d876e12 100644 --- a/parser/src/main/scala/aqua/parser/lift/FileSpan.scala +++ b/parser/src/main/scala/aqua/parser/lift/FileSpan.scala @@ -18,12 +18,12 @@ object FileSpan { case class Focus(name: String, locationMap: Eval[LocationMap], ctx: Int, spanFocus: Span.Focus) { def toConsoleStr( - errorType: String, + typ: String, msgs: List[String], onLeft: String, onRight: String = Console.RESET ): String = - onLeft + "---- " + errorType + ": " + s"$name:${spanFocus.line._1 + 1}:${spanFocus.column + 1}" + onRight + + onLeft + "---- " + typ + ": " + s"$name:${spanFocus.line._1 + 1}:${spanFocus.column + 1}" + onRight + spanFocus.toConsoleStr( msgs, onLeft, From aad5b6675ea433212077f16e2bf24cf75992b548 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 10:52:56 +0000 Subject: [PATCH 17/31] Add simple warning --- .../scala/aqua/semantics/rules/names/NamesInterpreter.scala | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala index bd718aea1..a2811dd79 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala @@ -103,7 +103,11 @@ class NamesInterpreter[S[_], X](using } case None => mapStackHeadM(report.error(name, "Cannot define a variable in the root scope").as(false))( - fr => (fr.addName(name, `type`) -> true).pure + fr => + report + .warning(name, "Variable name is too short.") + .whenA(name.value.length < 3) + .as(fr.addName(name, `type`) -> true) ) <* locations.addToken(name.value, name) } From 5c0de8dfef1dcd9f4d105120779cfd4444f43e2f Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 11:52:48 +0000 Subject: [PATCH 18/31] Refactor to ValidatedNec --- api/api/.js/src/main/scala/api/AquaAPI.scala | 82 ++++++------ .../main/scala/aqua/api/APICompilation.scala | 122 +++++++++--------- .../src/main/scala/aqua/run/CliFunc.scala | 26 ++-- .../main/scala/aqua.constants/Constants.scala | 32 ++--- .../main/scala/aqua/logging/LogLevels.scala | 23 ++-- 5 files changed, 144 insertions(+), 141 deletions(-) diff --git a/api/api/.js/src/main/scala/api/AquaAPI.scala b/api/api/.js/src/main/scala/api/AquaAPI.scala index ea25baf58..459544e77 100644 --- a/api/api/.js/src/main/scala/api/AquaAPI.scala +++ b/api/api/.js/src/main/scala/api/AquaAPI.scala @@ -13,7 +13,7 @@ import aqua.logging.{LogFormatter, LogLevels} import aqua.constants.Constants import aqua.io.* import aqua.raw.ops.Call -import aqua.run.{CallInfo, CallPreparer, CliFunc, FuncCompiler, RunPreparer} +import aqua.run.{CliFunc, FuncCompiler} import aqua.parser.lexer.{LiteralToken, Token} import aqua.parser.lift.FileSpan.F import aqua.parser.lift.{FileSpan, Span} @@ -69,11 +69,9 @@ object AquaAPI extends App with Logging { aquaConfigJS: js.UndefOr[AquaConfig] ): Promise[CompilationResult] = { aquaConfigJS.toOption - .map(cjs => AquaConfig.fromJS(cjs)) - .getOrElse( - validNec(AquaAPIConfig()) - ) - .map { config => + .map(AquaConfig.fromJS) + .getOrElse(validNec(AquaAPIConfig())) + .traverse { config => val importsList = imports.toList input match { @@ -83,10 +81,10 @@ object AquaAPI extends App with Logging { compileCall(c, importsList, config) } - } match { - case Valid(v) => v.unsafeToFuture().toJSPromise - case Invalid(errs) => js.Promise.resolve(CompilationResult.errs(errs.toChain.toList)) - } + } + .map(_.leftMap(errs => CompilationResult.errs(errs.toChain.toList)).merge) + .unsafeToFuture() + .toJSPromise } // Compile all non-call inputs @@ -137,7 +135,12 @@ object AquaAPI extends App with Logging { } - private def compileCall(call: types.Call, imports: List[String], config: AquaAPIConfig) = { + // Compile a function call + private def compileCall( + call: types.Call, + imports: List[String], + config: AquaAPIConfig + ) = { val path = call.input match { case i: types.Input => i.input case p: types.Path => p.path @@ -169,44 +172,41 @@ object AquaAPI extends App with Logging { private def errorsToResult( errors: NonEmptyChain[String], warnings: Chain[String] - ): CompilationResult = - CompilationResult.errs( - errors.toChain.toList, - warnings.toList - ) + ): CompilationResult = CompilationResult.errs( + errors.toChain.toList, + warnings.toList + ) private def compiledToTsSourceResult( compiled: Chain[AquaCompiled[FileModuleId]], warnings: Chain[String] - ): CompilationResult = - CompilationResult.result( - sources = compiled.toList - .flatMap(c => - c.compiled - .find(_.suffix == TypeScriptBackend.ext) - .map(_.content) - .map(GeneratedSource.tsSource(c.sourceId.toString, _)) - ), - warnings = warnings.toList - ) + ): CompilationResult = CompilationResult.result( + sources = compiled.toList + .flatMap(c => + c.compiled + .find(_.suffix == TypeScriptBackend.ext) + .map(_.content) + .map(GeneratedSource.tsSource(c.sourceId.toString, _)) + ), + warnings = warnings.toList + ) private def compiledToJsSourceResult( compiled: Chain[AquaCompiled[FileModuleId]], warnings: Chain[String] - ): CompilationResult = - CompilationResult.result( - sources = compiled.toList.flatMap { c => - for { - dtsContent <- c.compiled - .find(_.suffix == JavaScriptBackend.dtsExt) - .map(_.content) - jsContent <- c.compiled - .find(_.suffix == JavaScriptBackend.ext) - .map(_.content) - } yield GeneratedSource.jsSource(c.sourceId.toString, jsContent, dtsContent) - }, - warnings = warnings.toList - ) + ): CompilationResult = CompilationResult.result( + sources = compiled.toList.flatMap { c => + for { + dtsContent <- c.compiled + .find(_.suffix == JavaScriptBackend.dtsExt) + .map(_.content) + jsContent <- c.compiled + .find(_.suffix == JavaScriptBackend.ext) + .map(_.content) + } yield GeneratedSource.jsSource(c.sourceId.toString, jsContent, dtsContent) + }, + warnings = warnings.toList + ) private def generatedToAirResult( compiled: Chain[AquaCompiled[FileModuleId]], diff --git a/api/api/src/main/scala/aqua/api/APICompilation.scala b/api/api/src/main/scala/aqua/api/APICompilation.scala index a2e4b2971..322dccc3c 100644 --- a/api/api/src/main/scala/aqua/api/APICompilation.scala +++ b/api/api/src/main/scala/aqua/api/APICompilation.scala @@ -11,7 +11,7 @@ import aqua.logging.{LogFormatter, LogLevels} import aqua.constants.Constants import aqua.io.* import aqua.raw.ops.Call -import aqua.run.{CallInfo, CallPreparer, CliFunc, FuncCompiler, RunPreparer} +import aqua.run.{CliFunc, FuncCompiler, RunPreparer} import aqua.parser.lexer.{LiteralToken, Token} import aqua.parser.lift.FileSpan.F import aqua.parser.lift.{FileSpan, Span} @@ -25,7 +25,16 @@ import aqua.res.AquaRes import cats.Applicative import cats.~> -import cats.data.{Chain, EitherT, NonEmptyChain, Validated, ValidatedNec, Writer, ValidatedNel, NonEmptyList} +import cats.data.{ + Chain, + EitherT, + NonEmptyChain, + NonEmptyList, + Validated, + ValidatedNec, + ValidatedNel, + Writer +} import cats.data.Validated.{invalid, invalidNec, validNec, Invalid, Valid} import cats.syntax.applicative.* import cats.syntax.apply.* @@ -37,29 +46,10 @@ import cats.syntax.show.* import cats.syntax.traverse.* import cats.syntax.either.* import fs2.io.file.{Files, Path} -import scribe.{Logging, Level} +import scribe.{Level, Logging} object APICompilation { - extension [A, C <: NonEmptyChain[String] | NonEmptyList[String]](v: Validated[C, A]) { - def toResult: APIResult[A] = - v.toEither.leftMap{ - case v: NonEmptyChain[String] => v - case v: NonEmptyList[String] => - NonEmptyChain.fromNonEmptyList(v) - }.toEitherT - } - - extension [A](v: CompileResult[FileModuleId, AquaFileError, FileSpan.F][A]) { - def toResult: APIResult[A] = - v.leftMap(_.map(_.show)).mapK( - new (CompileWarnings[FileSpan.F] ~> APIWarnings) { - override def apply[A](w: CompileWarnings[FileSpan.F][A]): APIWarnings[A] = - w.mapWritten(_.map(_.show)) - } - ) - } - def compileCall( functionStr: String, pathStr: String, @@ -72,9 +62,7 @@ object APICompilation { ( LogLevels.levelFromString(aquaConfig.logLevel), Constants.parse(aquaConfig.constants) - ).tupled.toResult.flatTraverse { - case (level, constants) => - + ).tupled.toResult.flatTraverse { case (level, constants) => val transformConfig = aquaConfig.getTransformConfig.copy(constants = constants) LogFormatter.initLogger(Some(level)) @@ -95,10 +83,10 @@ object APICompilation { argsWithTypes <- fillWithTypes(cliFunc.args).toResult func = cliFunc.copy(args = argsWithTypes) preparer = new RunPreparer( - func, - arrow, - transformConfig - ) + func, + arrow, + transformConfig + ) ci <- preparer.prepare().toResult } yield ci.definitions -> ci.air } @@ -151,33 +139,51 @@ object APICompilation { aquaConfig: AquaAPIConfig, sources: AquaSources[IO, AquaFileError, FileModuleId], backend: Backend - ): IO[APIResult[Chain[AquaCompiled[FileModuleId]]]] = - ( - LogLevels.levelFromString(aquaConfig.logLevel), - Constants.parse(aquaConfig.constants) - ).tupled.toResult.flatTraverse { - case (level, constants) => - LogFormatter.initLogger(Some(level)) - - val transformConfig = aquaConfig.getTransformConfig - val config = AquaCompilerConf(constants ++ transformConfig.constantsList) - - CompilerAPI - .compile[IO, AquaFileError, FileModuleId, FileSpan.F]( - sources, - SpanParser.parser, - new AirValidator[IO] { - override def init(): IO[Unit] = Applicative[IO].pure(()) - override def validate(airs: List[AirFunction]): IO[ValidatedNec[String, Unit]] = - Applicative[IO].pure(validNec(())) - }, - new Backend.Transform { - override def transform(ex: AquaContext): AquaRes = - Transform.contextRes(ex, transformConfig) - - override def generate(aqua: AquaRes): Seq[Generated] = backend.generate(aqua) - }, - config - ).map(_.toResult) - } + ): IO[APIResult[Chain[AquaCompiled[FileModuleId]]]] = ( + LogLevels.levelFromString(aquaConfig.logLevel), + Constants.parse(aquaConfig.constants) + ).tupled.toResult.flatTraverse { case (level, constants) => + LogFormatter.initLogger(Some(level)) + + val transformConfig = aquaConfig.getTransformConfig + val config = AquaCompilerConf(constants ++ transformConfig.constantsList) + + CompilerAPI + .compile[IO, AquaFileError, FileModuleId, FileSpan.F]( + sources, + SpanParser.parser, + new AirValidator[IO] { + override def init(): IO[Unit] = Applicative[IO].pure(()) + override def validate(airs: List[AirFunction]): IO[ValidatedNec[String, Unit]] = + Applicative[IO].pure(validNec(())) + }, + new Backend.Transform { + override def transform(ex: AquaContext): AquaRes = + Transform.contextRes(ex, transformConfig) + + override def generate(aqua: AquaRes): Seq[Generated] = backend.generate(aqua) + }, + config + ) + .map(_.toResult) + } + + extension [A](v: ValidatedNec[String, A]) { + + def toResult: APIResult[A] = + v.toEither.toEitherT + } + + extension [A](v: CompileResult[FileModuleId, AquaFileError, FileSpan.F][A]) { + + def toResult: APIResult[A] = + v.leftMap(_.map(_.show)) + .mapK( + new (CompileWarnings[FileSpan.F] ~> APIWarnings) { + + override def apply[A](w: CompileWarnings[FileSpan.F][A]): APIWarnings[A] = + w.mapWritten(_.map(_.show)) + } + ) + } } diff --git a/aqua-run/src/main/scala/aqua/run/CliFunc.scala b/aqua-run/src/main/scala/aqua/run/CliFunc.scala index ae8f554f9..5e0ba2d37 100644 --- a/aqua-run/src/main/scala/aqua/run/CliFunc.scala +++ b/aqua-run/src/main/scala/aqua/run/CliFunc.scala @@ -5,8 +5,8 @@ import aqua.parser.lift.Span import aqua.raw.value.{CollectionRaw, LiteralRaw, ValueRaw, VarRaw} import aqua.types.{ArrayType, BottomType} -import cats.data.{NonEmptyList, Validated, ValidatedNel} -import cats.data.Validated.{invalid, invalidNel, validNel} +import cats.data.{NonEmptyChain, NonEmptyList, Validated, ValidatedNec} +import cats.data.Validated.{invalid, invalidNec, validNec} import cats.{~>, Id} import cats.syntax.traverse.* import cats.syntax.validated.* @@ -18,25 +18,27 @@ case class CliFunc(name: String, args: List[ValueRaw] = Nil) object CliFunc { - def spanToId: Span.S ~> Id = new (Span.S ~> Id) { + private val spanToId: Span.S ~> Id = new (Span.S ~> Id) { override def apply[A](span: Span.S[A]): Id[A] = span.extract } - def fromString(func: String): ValidatedNel[String, CliFunc] = { + def fromString(func: String): ValidatedNec[String, CliFunc] = { CallArrowToken.callArrow .parseAll(func.trim) - .toValidated - .leftMap( - _.expected.map(_.context.mkString("\n")) + .leftMap(error => + NonEmptyChain + .fromNonEmptyList(error.expected) + .map(_.context.mkString("\n")) ) + .toValidated .map(_.mapK(spanToId)) .andThen(expr => expr.args.traverse { case LiteralToken(value, ts) => - LiteralRaw(value, ts).valid + LiteralRaw(value, ts).validNec case VarToken(name) => - VarRaw(name.value, BottomType).valid + VarRaw(name.value, BottomType).validNec case CollectionToken(_, values) => values.traverse { case LiteralToken(value, ts) => @@ -53,11 +55,11 @@ object CliFunc { .map(l => CollectionRaw(l, ArrayType(l.head.baseType))) .getOrElse(ValueRaw.Nil) ) - .toValidatedNel + .toValidatedNec case CallArrowToken(_, _, _) => - "Function calls as arguments are not supported.".invalidNel + "Function calls as arguments are not supported.".invalidNec case _ => - "Unsupported argument.".invalidNel + "Unsupported argument.".invalidNec }.map(args => CliFunc(expr.funcName.value, args)) ) } diff --git a/utils/constants/src/main/scala/aqua.constants/Constants.scala b/utils/constants/src/main/scala/aqua.constants/Constants.scala index 0782f788b..b5e8f29bf 100644 --- a/utils/constants/src/main/scala/aqua.constants/Constants.scala +++ b/utils/constants/src/main/scala/aqua.constants/Constants.scala @@ -4,26 +4,20 @@ import aqua.parser.expr.ConstantExpr import aqua.raw.ConstantRaw import aqua.raw.value.LiteralRaw -import cats.data.{NonEmptyList, Validated, ValidatedNel} +import cats.data.{NonEmptyList, Validated, ValidatedNec} +import cats.syntax.traverse.* +import cats.syntax.either.* object Constants { - def parse(strs: List[String]): ValidatedNel[String, List[ConstantRaw]] = { - val parsed = strs.map(s => ConstantExpr.onlyLiteral.parseAll(s)) - - val errors = parsed.zip(strs).collect { case (Left(_), str) => - str - } - - NonEmptyList - .fromList(errors) - .fold( - Validated.validNel[String, List[ConstantRaw]](parsed.collect { case Right(v) => - ConstantRaw(v._1.value, LiteralRaw(v._2.value, v._2.ts), false) - }) - ) { errors => - val errorMsgs = errors.map(str => s"Invalid constant definition '$str'.") - Validated.invalid(errorMsgs) - } - } + def parse(strs: List[String]): ValidatedNec[String, List[ConstantRaw]] = + strs.traverse(s => + ConstantExpr.onlyLiteral + .parseAll(s) + .leftMap(_ => s"Invalid constant definition '$s'.") + .toValidatedNec + .map { case (name, literal) => + ConstantRaw(name.value, LiteralRaw(literal.value, literal.ts), false) + } + ) } diff --git a/utils/logging/src/main/scala/aqua/logging/LogLevels.scala b/utils/logging/src/main/scala/aqua/logging/LogLevels.scala index 01edf71fc..c34fd26d3 100644 --- a/utils/logging/src/main/scala/aqua/logging/LogLevels.scala +++ b/utils/logging/src/main/scala/aqua/logging/LogLevels.scala @@ -3,8 +3,9 @@ package aqua.logging import cats.syntax.option.* import cats.syntax.either.* import cats.syntax.foldable.* -import cats.data.Validated.{invalidNel, validNel} -import cats.data.{NonEmptyList, Validated, ValidatedNel} +import cats.syntax.validated.* +import cats.data.Validated.* +import cats.data.{NonEmptyList, Validated, ValidatedNec} import scribe.Level case class LogLevels( @@ -20,10 +21,10 @@ object LogLevels { def apply(level: Level): LogLevels = LogLevels(level, level, level) - def levelFromString(s: String): ValidatedNel[String, Level] = + def levelFromString(s: String): ValidatedNec[String, Level] = LogLevel.stringToLogLevel .get(s.toLowerCase.trim()) - .toValidNel(s"Invalid log-level '$s'. $logHelpMessage") + .toValidNec(s"Invalid log-level '$s'. $logHelpMessage") lazy val error = s"Invalid log-level format. $logHelpMessage" @@ -32,17 +33,17 @@ object LogLevels { name: String, level: String, logLevels: LogLevels - ): ValidatedNel[String, LogLevels] = { + ): ValidatedNec[String, LogLevels] = { levelFromString(level).andThen { level => name.trim().toLowerCase() match { case "compiler" => - validNel(logLevels.copy(compiler = level)) + logLevels.copy(compiler = level).validNec case "fluencejs" => - validNel(logLevels.copy(fluencejs = level)) + logLevels.copy(fluencejs = level).validNec case "aquavm" => - validNel(logLevels.copy(aquavm = level)) + logLevels.copy(aquavm = level).validNec case s => - invalidNel( + invalidNec( s"Unknown component '$s' in log-level. Please use one of these: 'aquavm', 'compiler' and 'fluencejs'" ) } @@ -51,14 +52,14 @@ object LogLevels { // Format: '' or 'compiler=,fluencejs=,aquavm=', // where is one of these strings: 'all', 'trace', 'debug', 'info', 'warn', 'error', 'off' - def fromString(s: String): ValidatedNel[String, LogLevels] = + def fromString(s: String): ValidatedNec[String, LogLevels] = s.split(",") .toList .foldLeftM(LogLevels()) { case (levels, level) => level.split("=").toList match { case n :: l :: Nil => fromStrings(n, l, levels).toEither case l :: Nil => levelFromString(l).map(apply).toEither - case _ => invalidNel(error).toEither + case _ => error.invalidNec.toEither } } .toValidated From d4e6b3d2349a2daf28b9608f0d1ffff3691d9568 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 16:27:52 +0000 Subject: [PATCH 19/31] Refactor --- api/api/.js/src/main/scala/api/AquaAPI.scala | 2 +- .../main/scala/api/types/OutputTypes.scala | 6 +- .../scala/aqua/compiler/AquaCompiler.scala | 71 +++++++++---------- .../scala/aqua/compiler/CompilerAPI.scala | 14 ++-- .../.js/src/main/scala/aqua/lsp/AquaLSP.scala | 4 +- .../.jvm/src/main/scala/aqua/lsp/Test.scala | 1 + .../src/main/scala/aqua/lsp/LSPCompiler.scala | 43 +++++------ .../src/main/scala/aqua/linker/Linker.scala | 5 +- 8 files changed, 79 insertions(+), 67 deletions(-) diff --git a/api/api/.js/src/main/scala/api/AquaAPI.scala b/api/api/.js/src/main/scala/api/AquaAPI.scala index 459544e77..d673e052b 100644 --- a/api/api/.js/src/main/scala/api/AquaAPI.scala +++ b/api/api/.js/src/main/scala/api/AquaAPI.scala @@ -140,7 +140,7 @@ object AquaAPI extends App with Logging { call: types.Call, imports: List[String], config: AquaAPIConfig - ) = { + ): IO[CompilationResult] = { val path = call.input match { case i: types.Input => i.input case p: types.Path => p.path diff --git a/api/api/.js/src/main/scala/api/types/OutputTypes.scala b/api/api/.js/src/main/scala/api/types/OutputTypes.scala index d0d8fbc21..6e1a19539 100644 --- a/api/api/.js/src/main/scala/api/types/OutputTypes.scala +++ b/api/api/.js/src/main/scala/api/types/OutputTypes.scala @@ -2,6 +2,7 @@ package api.types import aqua.js.{FunctionDefJs, ServiceDefJs} import aqua.model.transform.TransformConfig + import cats.data.Validated.{invalidNec, validNec, Invalid, Valid} import cats.data.{Chain, NonEmptyChain, Validated, ValidatedNec} @@ -9,7 +10,6 @@ import scala.scalajs.js import scala.scalajs.js.JSConverters.* import scala.scalajs.js.annotation.{JSExport, JSExportTopLevel} import scala.scalajs.js.| -import aqua.parser.lexer.Token.func @JSExportTopLevel("AquaFunction") case class AquaFunction( @@ -32,7 +32,9 @@ case class GeneratedSource( ) object GeneratedSource { - def tsSource(name: String, tsSource: String) = new GeneratedSource(name, tsSource, null, null) + + def tsSource(name: String, tsSource: String) = + new GeneratedSource(name, tsSource, null, null) def jsSource(name: String, jsSource: String, tsTypes: String) = new GeneratedSource(name, null, jsSource, tsTypes) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index 4faca1c75..2de17d51a 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -38,31 +38,6 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( type CompileWarns = [A] =>> CompileWarnings[S][A] type CompileRes = [A] =>> CompileResult[I, E, S][A] - private val warningsK: semantics.ProcessWarnings ~> CompileWarns = - new FunctionK[semantics.ProcessWarnings, CompileWarns] { - - override def apply[A]( - fa: semantics.ProcessWarnings[A] - ): CompileWarns[A] = - fa.mapWritten(_.map(AquaWarning.CompileWarning.apply)) - } - - extension (res: semantics.ProcessResult) { - - def toCompileRes: CompileRes[C] = - res - .leftMap(_.map(CompileError.apply)) - .mapK(warningsK) - } - - extension [A](res: ValidatedNec[SemanticError[S], A]) { - - def toCompileRes: CompileRes[A] = - res.toEither - .leftMap(_.map(CompileError.apply)) - .toEitherT[CompileWarns] - } - type CompiledCtx = CompileRes[Ctx] type CompiledCtxT = CompiledCtx => CompiledCtx @@ -72,14 +47,12 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( ): CompileRes[Map[I, C]] = { logger.trace("linking modules...") + // By default, provide an empty context for this module's id + val empty: I => CompiledCtx = i => NonEmptyMap.one(i, Monoid[C].empty).pure[CompileRes] + for { linked <- Linker - .link( - modules, - cycleError, - // By default, provide an empty context for this module's id - i => NonEmptyMap.one(i, Monoid.empty[C]).asRight.toEitherT - ) + .link(modules, cycleError, empty) .toEither .toEitherT[CompileWarns] res <- EitherT( @@ -124,14 +97,40 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( } yield NonEmptyMap.one(mod.id, rc) ) .value - .map( - _.toEitherT[CompileWarns].flatMap(modules => - linkModules( + .map(resolved => + for { + modules <- resolved.toEitherT[CompileWarns] + linked <- linkModules( modules, - cycle => CycleError[I, E, S](cycle.map(_.id)) + cycle => CycleError(cycle.map(_.id)) ) - ) + } yield linked ) } + private val warningsK: semantics.ProcessWarnings ~> CompileWarns = + new FunctionK[semantics.ProcessWarnings, CompileWarns] { + + override def apply[A]( + fa: semantics.ProcessWarnings[A] + ): CompileWarns[A] = + fa.mapWritten(_.map(AquaWarning.CompileWarning.apply)) + } + + extension (res: semantics.ProcessResult) { + + def toCompileRes: CompileRes[C] = + res + .leftMap(_.map(CompileError.apply)) + .mapK(warningsK) + } + + extension [A](res: ValidatedNec[SemanticError[S], A]) { + + def toCompileRes: CompileRes[A] = + res.toEither + .leftMap(_.map(CompileError.apply)) + .toEitherT[CompileWarns] + } + } diff --git a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala index b472d99d0..722c59a23 100644 --- a/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala +++ b/compiler/src/main/scala/aqua/compiler/CompilerAPI.scala @@ -55,16 +55,22 @@ object CompilerAPI extends Logging { private def getAquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad]( config: AquaCompilerConf ): AquaCompiler[F, E, I, S, RawContext] = { - implicit val rc: Monoid[RawContext] = RawContext + given Monoid[RawContext] = RawContext .implicits( - RawContext.blank - .copy(parts = Chain.fromSeq(config.constantsList).map(const => RawContext.blank -> const)) + RawContext.blank.copy( + parts = Chain + .fromSeq(config.constantsList) + .map(const => RawContext.blank -> const) + ) ) .rawContextMonoid val semantics = new RawSemantics[S]() - new AquaCompiler[F, E, I, S, RawContext](new HeaderHandler[S, RawContext](), semantics) + new AquaCompiler[F, E, I, S, RawContext]( + new HeaderHandler(), + semantics + ) } // Get result generated by backend diff --git a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala index 9e0f92dc3..72c2e5c91 100644 --- a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala +++ b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala @@ -167,9 +167,7 @@ object AquaLSP extends App with Logging { val link = for { from <- fromOp to <- toOp - } yield { - TokenLink(from, to) - } + } yield TokenLink(from, to) if (link.isEmpty) logger.warn(s"Incorrect coordinates for token '${from.unit._1.name}'") diff --git a/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala b/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala index ce9806aa2..a10465401 100644 --- a/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala +++ b/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala @@ -7,6 +7,7 @@ import aqua.lsp.LSPCompiler import aqua.parser.lift.FileSpan import aqua.raw.ConstantRaw import aqua.{AquaIO, SpanParser} + import cats.data.Validated import cats.effect.{IO, IOApp, Sync} import fs2.io.file.Path diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala index f4135bcdd..2a500704b 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala @@ -21,36 +21,39 @@ object LSPCompiler { private def getLspAquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad]( config: AquaCompilerConf ): AquaCompiler[F, E, I, S, LspContext[S]] = { - implicit val rc: Monoid[LspContext[S]] = LspContext + given Monoid[LspContext[S]] = LspContext .implicits( - LspContext - .blank[S] - .copy(raw = - RawContext.blank.copy(parts = - Chain.fromSeq(config.constantsList).map(const => RawContext.blank -> const) - ) + LspContext.blank.copy(raw = + RawContext.blank.copy( + parts = Chain + .fromSeq(config.constantsList) + .map(const => RawContext.blank -> const) ) + ) ) .lspContextMonoid - implicit val headerSemMonoid: Monoid[HeaderSem[S, LspContext[S]]] = - new Monoid[HeaderSem[S, LspContext[S]]] { - override def empty: HeaderSem[S, LspContext[S]] = HeaderSem(rc.empty, (c, _) => validNec(c)) + given Monoid[HeaderSem[S, LspContext[S]]] with { + override def empty: HeaderSem[S, LspContext[S]] = + HeaderSem(Monoid[LspContext[S]].empty, (c, _) => validNec(c)) - override def combine( - a: HeaderSem[S, LspContext[S]], - b: HeaderSem[S, LspContext[S]] - ): HeaderSem[S, LspContext[S]] = { - HeaderSem( - a.initCtx |+| b.initCtx, - (c, i) => a.finInitCtx(c, i).andThen(b.finInitCtx(_, i)) - ) - } + override def combine( + a: HeaderSem[S, LspContext[S]], + b: HeaderSem[S, LspContext[S]] + ): HeaderSem[S, LspContext[S]] = { + HeaderSem( + a.initCtx |+| b.initCtx, + (c, i) => a.finInitCtx(c, i).andThen(b.finInitCtx(_, i)) + ) } + } val semantics = new LspSemantics[S]() - new AquaCompiler[F, E, I, S, LspContext[S]](new HeaderHandler[S, LspContext[S]](), semantics) + new AquaCompiler[F, E, I, S, LspContext[S]]( + new HeaderHandler(), + semantics + ) } def compileToLsp[F[_]: Monad, E, I: Order, S[_]: Comonad]( diff --git a/linker/src/main/scala/aqua/linker/Linker.scala b/linker/src/main/scala/aqua/linker/Linker.scala index 2e6cc9611..8669e201b 100644 --- a/linker/src/main/scala/aqua/linker/Linker.scala +++ b/linker/src/main/scala/aqua/linker/Linker.scala @@ -132,7 +132,10 @@ object Linker extends Logging { else { val result = iter(modules.loaded.values.toList, Map.empty, cycleError) - result.map(_.collect { case (i, f) if modules.exports(i) => i -> f(empty(i)) }) + result.map(_.collect { + case (i, f) if modules.exports(i) => + i -> f(empty(i)) + }) } } From 5ed154a6bed9def9f3dd2095d6057945df0cbdb1 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 16:30:45 +0000 Subject: [PATCH 20/31] Add comment --- compiler/src/main/scala/aqua/compiler/AquaCompiler.scala | 5 +++++ 1 file changed, 5 insertions(+) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index 2de17d51a..ca3673574 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -94,6 +94,11 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( rc <- headerSem .finCtx(processed) .toCompileRes + /** + * Here we build a map of contexts while processing modules. + * Should not linker provide this info inside this process? + * Building this map complicates things a lot. + */ } yield NonEmptyMap.one(mod.id, rc) ) .value From 7a790077ed00bad01938ca12d1363a5671198206 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Fri, 22 Sep 2023 16:59:12 +0000 Subject: [PATCH 21/31] Propagate warnings to LspContext --- .../scala/aqua/compiler/AquaCompiler.scala | 6 ++-- .../.jvm/src/main/scala/aqua/lsp/Test.scala | 4 +-- .../src/main/scala/aqua/lsp/LSPCompiler.scala | 2 ++ .../src/main/scala/aqua/lsp/LspContext.scala | 19 +++++++------ .../main/scala/aqua/lsp/LspSemantics.scala | 28 ++++++++++--------- .../src/main/scala/aqua/raw/RawContext.scala | 8 +++--- .../main/scala/aqua/semantics/Semantics.scala | 10 ++++--- 7 files changed, 43 insertions(+), 34 deletions(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index ca3673574..eef23b012 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -113,11 +113,11 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( ) } - private val warningsK: semantics.ProcessWarnings ~> CompileWarns = - new FunctionK[semantics.ProcessWarnings, CompileWarns] { + private val warningsK: semantics.Warnings ~> CompileWarns = + new FunctionK[semantics.Warnings, CompileWarns] { override def apply[A]( - fa: semantics.ProcessWarnings[A] + fa: semantics.Warnings[A] ): CompileWarns[A] = fa.mapWritten(_.map(AquaWarning.CompileWarning.apply)) } diff --git a/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala b/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala index a10465401..4a39188d4 100644 --- a/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala +++ b/language-server/language-server-api/.jvm/src/main/scala/aqua/lsp/Test.scala @@ -32,9 +32,9 @@ object Test extends IOApp.Simple { ) .map { case Validated.Invalid(errs) => - errs.map(System.err.println): Unit + errs.toChain.toList.foreach(System.err.println) case Validated.Valid(res) => - res.map(println): Unit + res.foreach(println) } _ <- IO.println("Compilation ends in: " + (System.currentTimeMillis() - start) + " ms") } yield () diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala index 2a500704b..f0eb0dc83 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LSPCompiler.scala @@ -65,6 +65,8 @@ object LSPCompiler { val compiler = getLspAquaCompiler[F, E, I, S](config) compiler .compileRaw(sources, parser) + // NOTE: Ignore warnings here as + // they are collected inside context .map(_.value.value.toValidated) } } diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LspContext.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LspContext.scala index f3afd58ef..aa2ddb555 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LspContext.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LspContext.scala @@ -1,11 +1,11 @@ package aqua.lsp import aqua.parser.lexer.{LiteralToken, NamedTypeToken, Token} -import aqua.raw.RawContext.semiRC import aqua.raw.{RawContext, RawPart} -import aqua.semantics.SemanticError +import aqua.semantics.{SemanticError, SemanticWarning} import aqua.semantics.header.Picker import aqua.types.{ArrowType, Type} + import cats.syntax.monoid.* import cats.{Monoid, Semigroup} @@ -18,14 +18,15 @@ case class LspContext[S[_]]( tokens: Map[String, Token[S]] = Map.empty[String, Token[S]], locations: List[(Token[S], Token[S])] = Nil, importTokens: List[LiteralToken[S]] = Nil, - errors: List[SemanticError[S]] = Nil + errors: List[SemanticError[S]] = Nil, + warnings: List[SemanticWarning[S]] = Nil ) object LspContext { def blank[S[_]]: LspContext[S] = LspContext[S](raw = RawContext()) - implicit def semiLsp[S[_]]: Semigroup[LspContext[S]] = + given [S[_]]: Semigroup[LspContext[S]] = (x: LspContext[S], y: LspContext[S]) => LspContext[S]( raw = x.raw |+| y.raw, @@ -33,20 +34,22 @@ object LspContext { rootArrows = x.rootArrows ++ y.rootArrows, constants = x.constants ++ y.constants, locations = x.locations ++ y.locations, - tokens = x.tokens ++ y.tokens + tokens = x.tokens ++ y.tokens, + errors = x.errors ++ y.errors, + warnings = x.warnings ++ y.warnings ) trait Implicits[S[_]] { - implicit val lspContextMonoid: Monoid[LspContext[S]] + val lspContextMonoid: Monoid[LspContext[S]] } def implicits[S[_]](init: LspContext[S]): Implicits[S] = new Implicits[S] { - override implicit val lspContextMonoid: Monoid[LspContext[S]] = new Monoid[LspContext[S]] { + override val lspContextMonoid: Monoid[LspContext[S]] = new Monoid[LspContext[S]] { override def empty: LspContext[S] = init override def combine(x: LspContext[S], y: LspContext[S]): LspContext[S] = { - semiLsp[S].combine(x, y) + Semigroup[LspContext[S]].combine(x, y) } } diff --git a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala index 0544378a4..b412e7989 100644 --- a/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala +++ b/language-server/language-server-api/src/main/scala/aqua/lsp/LspSemantics.scala @@ -20,18 +20,19 @@ import monocle.macros.GenLens class LspSemantics[S[_]] extends Semantics[S, LspContext[S]] { - def getImportTokens(ast: Ast[S]): List[LiteralToken[S]] = { - ast.head.foldLeft[List[LiteralToken[S]]](Nil) { case (l, header) => - header match { - case ImportExpr(fn) => l :+ fn - case ImportFromExpr(_, fn) => l :+ fn - case UseExpr(fn, _) => l :+ fn - case UseFromExpr(_, fn, _) => l :+ fn - case _ => l - } - } - } + private def getImportTokens(ast: Ast[S]): List[LiteralToken[S]] = + ast.collectHead { + case ImportExpr(fn) => fn + case ImportFromExpr(_, fn) => fn + case UseExpr(fn, _) => fn + case UseFromExpr(_, fn, _) => fn + }.value.toList + /** + * Process the AST and return the semantics result. + * NOTE: LspSemantics never return errors or warnings, + * they are collected in LspContext. + */ def process( ast: Ast[S], init: LspContext[S] @@ -71,8 +72,9 @@ class LspSemantics[S[_]] extends Semantics[S, LspContext[S]] { locations = state.locations.allLocations, importTokens = importTokens, tokens = state.locations.tokens, - errors = state.errors.toList - ).asRight.toEitherT[ProcessWarnings] + errors = state.errors.toList, + warnings = state.warnings.toList + ).pure[Result] } // TODO: return as Eval .value diff --git a/model/raw/src/main/scala/aqua/raw/RawContext.scala b/model/raw/src/main/scala/aqua/raw/RawContext.scala index 1a870d2e5..9f26d0286 100644 --- a/model/raw/src/main/scala/aqua/raw/RawContext.scala +++ b/model/raw/src/main/scala/aqua/raw/RawContext.scala @@ -114,7 +114,7 @@ case class RawContext( object RawContext { val blank: RawContext = RawContext() - implicit val semiRC: Semigroup[RawContext] = + given Semigroup[RawContext] = (x: RawContext, y: RawContext) => RawContext( x.init.flatMap(xi => y.init.map(xi |+| _)) orElse x.init orElse y.init, @@ -126,16 +126,16 @@ object RawContext { ) trait Implicits { - implicit val rawContextMonoid: Monoid[RawContext] + val rawContextMonoid: Monoid[RawContext] } def implicits(init: RawContext): Implicits = new Implicits { - override implicit val rawContextMonoid: Monoid[RawContext] = new Monoid[RawContext] { + override val rawContextMonoid: Monoid[RawContext] = new Monoid[RawContext] { override def empty: RawContext = init override def combine(x: RawContext, y: RawContext): RawContext = - semiRC.combine(x, y) + Semigroup[RawContext].combine(x, y) } } diff --git a/semantics/src/main/scala/aqua/semantics/Semantics.scala b/semantics/src/main/scala/aqua/semantics/Semantics.scala index 969505749..bf76ea63f 100644 --- a/semantics/src/main/scala/aqua/semantics/Semantics.scala +++ b/semantics/src/main/scala/aqua/semantics/Semantics.scala @@ -7,17 +7,19 @@ import cats.data.{Chain, EitherNec, EitherT, NonEmptyChain, ValidatedNec, Writer trait Semantics[S[_], C] { - final type ProcessWarnings = [A] =>> Writer[ + final type Warnings = [A] =>> Writer[ Chain[SemanticWarning[S]], A ] - final type ProcessResult = EitherT[ - ProcessWarnings, + final type Result = [A] =>> EitherT[ + Warnings, NonEmptyChain[SemanticError[S]], - C + A ] + final type ProcessResult = Result[C] + def process( ast: Ast[S], init: C From 4b3cde0b8fe09537e81f4600988d49004ea8cec3 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 07:59:39 +0000 Subject: [PATCH 22/31] Propagate warnings to LSP --- .../.js/src/main/scala/aqua/lsp/AquaLSP.scala | 135 +++++++++++------- 1 file changed, 80 insertions(+), 55 deletions(-) diff --git a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala index 72c2e5c91..84cea97eb 100644 --- a/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala +++ b/language-server/language-server-api/.js/src/main/scala/aqua/lsp/AquaLSP.scala @@ -2,6 +2,7 @@ package aqua.lsp import aqua.compiler.* import aqua.compiler.AquaError.{ParserError as AquaParserError, *} +import aqua.compiler.AquaWarning.* import aqua.files.{AquaFileSources, AquaFilesIO, FileModuleId} import aqua.io.* import aqua.parser.lexer.{LiteralToken, Token} @@ -9,7 +10,7 @@ import aqua.parser.lift.FileSpan.F import aqua.parser.lift.{FileSpan, Span} import aqua.parser.{ArrowReturnError, BlockIndentError, LexerError, ParserError} import aqua.raw.ConstantRaw -import aqua.semantics.{HeaderError, RulesViolated, WrongAST} +import aqua.semantics.{HeaderError, RulesViolated, SemanticWarning, WrongAST} import aqua.{AquaIO, SpanParser} import cats.data.Validated.{invalidNec, validNec, Invalid, Valid} @@ -29,8 +30,9 @@ import scala.scalajs.js.{undefined, UndefOr} @JSExportAll case class CompilationResult( errors: js.Array[ErrorInfo], - locations: js.Array[TokenLink], - importLocations: js.Array[TokenImport] + warnings: js.Array[WarningInfo] = js.Array(), + locations: js.Array[TokenLink] = js.Array(), + importLocations: js.Array[TokenImport] = js.Array() ) @JSExportAll @@ -74,58 +76,79 @@ object ErrorInfo { } } +@JSExportAll +case class WarningInfo(start: Int, end: Int, message: String, location: UndefOr[String]) + +object WarningInfo { + + def apply(fileSpan: FileSpan, message: String): WarningInfo = { + val start = fileSpan.span.startIndex + val end = fileSpan.span.endIndex + WarningInfo(start, end, message, fileSpan.name) + } +} + @JSExportTopLevel("AquaLSP") object AquaLSP extends App with Logging { - def errorToInfo(error: AquaError[FileModuleId, AquaFileError, FileSpan.F]): List[ErrorInfo] = { - error match { - case AquaParserError(err) => - err match { - case BlockIndentError(indent, message) => - ErrorInfo(indent._1, message) :: Nil - case ArrowReturnError(point, message) => - ErrorInfo(point._1, message) :: Nil - case LexerError((span, e)) => - e.expected.toList - .groupBy(_.offset) - .map { case (offset, exps) => - val localSpan = Span(offset, offset + 1) - val fSpan = FileSpan(span.name, span.locationMap, localSpan) - val errorMessages = exps.flatMap(exp => ParserError.expectationToString(exp)) - val msg = s"${errorMessages.head}" :: errorMessages.tail.map(t => "OR " + t) - (offset, ErrorInfo(fSpan, msg.mkString("\n"))) - } - .toList - .sortBy(_._1) - .map(_._2) - .reverse - } - case SourcesError(err) => - ErrorInfo.applyOp(0, 0, err.showForConsole, None) :: Nil - case ResolveImportsError(_, token, err) => - ErrorInfo(token.unit._1, err.showForConsole) :: Nil - case ImportError(token) => - ErrorInfo(token.unit._1, "Cannot resolve import") :: Nil - case CycleError(modules) => - ErrorInfo.applyOp( - 0, - 0, - s"Cycle loops detected in imports: ${modules.map(_.file.fileName)}", - None - ) :: Nil - case CompileError(err) => - err match { - case RulesViolated(token, messages) => - ErrorInfo(token.unit._1, messages.mkString("\n")) :: Nil - case HeaderError(token, message) => - ErrorInfo(token.unit._1, message) :: Nil - case WrongAST(ast) => - ErrorInfo.applyOp(0, 0, "Semantic error: wrong AST", None) :: Nil - - } - case OutputError(_, err) => - ErrorInfo.applyOp(0, 0, err.showForConsole, None) :: Nil - } + private def errorToInfo( + error: AquaError[FileModuleId, AquaFileError, FileSpan.F] + ): List[ErrorInfo] = error match { + case AquaParserError(err) => + err match { + case BlockIndentError(indent, message) => + ErrorInfo(indent._1, message) :: Nil + case ArrowReturnError(point, message) => + ErrorInfo(point._1, message) :: Nil + case LexerError((span, e)) => + e.expected.toList + .groupBy(_.offset) + .map { case (offset, exps) => + val localSpan = Span(offset, offset + 1) + val fSpan = FileSpan(span.name, span.locationMap, localSpan) + val errorMessages = exps.flatMap(exp => ParserError.expectationToString(exp)) + val msg = s"${errorMessages.head}" :: errorMessages.tail.map(t => "OR " + t) + (offset, ErrorInfo(fSpan, msg.mkString("\n"))) + } + .toList + .sortBy(_._1) + .map(_._2) + .reverse + } + case SourcesError(err) => + ErrorInfo.applyOp(0, 0, err.showForConsole, None) :: Nil + case ResolveImportsError(_, token, err) => + ErrorInfo(token.unit._1, err.showForConsole) :: Nil + case ImportError(token) => + ErrorInfo(token.unit._1, "Cannot resolve import") :: Nil + case CycleError(modules) => + ErrorInfo.applyOp( + 0, + 0, + s"Cycle loops detected in imports: ${modules.map(_.file.fileName)}", + None + ) :: Nil + case CompileError(err) => + err match { + case RulesViolated(token, messages) => + ErrorInfo(token.unit._1, messages.mkString("\n")) :: Nil + case HeaderError(token, message) => + ErrorInfo(token.unit._1, message) :: Nil + case WrongAST(ast) => + ErrorInfo.applyOp(0, 0, "Semantic error: wrong AST", None) :: Nil + + } + case OutputError(_, err) => + ErrorInfo.applyOp(0, 0, err.showForConsole, None) :: Nil + case AirValidationError(errors) => + errors.toChain.toList.map(ErrorInfo.applyOp(0, 0, _, None)) + } + + private def warningToInfo( + warning: AquaWarning[FileSpan.F] + ): List[WarningInfo] = warning match { + case CompileWarning(SemanticWarning(token, messages)) => + WarningInfo(token.unit._1, messages.mkString("\n")) :: Nil } @JSExport @@ -186,6 +209,7 @@ object AquaLSP extends App with Logging { val result = fileRes match { case Valid(lsp) => val errors = lsp.errors.map(CompileError.apply).flatMap(errorToInfo) + val warnings = lsp.warnings.map(CompileWarning.apply).flatMap(warningToInfo) errors match case Nil => logger.debug("No errors on compilation.") @@ -194,13 +218,14 @@ object AquaLSP extends App with Logging { CompilationResult( errors.toJSArray, + warnings.toJSArray, locationsToJs(lsp.locations), importsToTokenImport(lsp.importTokens) ) - case Invalid(e: NonEmptyChain[AquaError[FileModuleId, AquaFileError, FileSpan.F]]) => - val errors = e.toNonEmptyList.toList.flatMap(errorToInfo) + case Invalid(e) => + val errors = e.toChain.toList.flatMap(errorToInfo) logger.debug("Errors: " + errors.mkString("\n")) - CompilationResult(errors.toJSArray, List.empty.toJSArray, List.empty.toJSArray) + CompilationResult(errors.toJSArray) } result } From 6406cc79ea725131db7526f8e4f223c938141291 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 08:20:20 +0000 Subject: [PATCH 23/31] Add warnings to js api --- api/api-npm/index.d.ts | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/api/api-npm/index.d.ts b/api/api-npm/index.d.ts index 5612af809..0f67cb064 100644 --- a/api/api-npm/index.d.ts +++ b/api/api-npm/index.d.ts @@ -12,11 +12,12 @@ export class GeneratedSource { tsTypes?: string; } -class CompilationResult { +export class CompilationResult { services: Record; functions: Record; functionCall?: AquaFunction; errors: string[]; + warnings: string[]; generatedSources: GeneratedSource[]; } From 0a89df65e8288855472dcac4dcd8f2b69b8d8553 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 08:26:24 +0000 Subject: [PATCH 24/31] Update LSP js api --- language-server/language-server-npm/aqua-lsp-api.d.ts | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/language-server/language-server-npm/aqua-lsp-api.d.ts b/language-server/language-server-npm/aqua-lsp-api.d.ts index 4fd673d6e..0f9c0e7c8 100644 --- a/language-server/language-server-npm/aqua-lsp-api.d.ts +++ b/language-server/language-server-npm/aqua-lsp-api.d.ts @@ -23,8 +23,16 @@ export interface ErrorInfo { location: string | null } +export interface WarningInfo { + start: number, + end: number, + message: string, + location: string | null +} + export interface CompilationResult { errors: ErrorInfo[], + warnings: WarningInfo[], locations: TokenLink[], importLocations: TokenImport[] } From 16d14f0e1ceae408bb5bd5f0fc462f1f7be301e9 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 08:31:51 +0000 Subject: [PATCH 25/31] Use export declare --- api/api-npm/index.d.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/api/api-npm/index.d.ts b/api/api-npm/index.d.ts index 0f67cb064..6c82b719e 100644 --- a/api/api-npm/index.d.ts +++ b/api/api-npm/index.d.ts @@ -1,18 +1,18 @@ import { ServiceDef, FunctionCallDef } from "@fluencelabs/interfaces"; -export class AquaFunction { +export declare class AquaFunction { funcDef: FunctionCallDef; script: string; } -export class GeneratedSource { +export declare class GeneratedSource { name: string; tsSource?: string; jsSource?: string; tsTypes?: string; } -export class CompilationResult { +export declare class CompilationResult { services: Record; functions: Record; functionCall?: AquaFunction; From cf5e6437f4f5834708d60e2be5875cd30d096ebe Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 08:56:44 +0000 Subject: [PATCH 26/31] Add comment --- .../src/main/scala/aqua/compiler/AquaCompiler.scala | 10 +++++++++- 1 file changed, 9 insertions(+), 1 deletion(-) diff --git a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala index eef23b012..5913e3d32 100644 --- a/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala +++ b/compiler/src/main/scala/aqua/compiler/AquaCompiler.scala @@ -57,7 +57,15 @@ class AquaCompiler[F[_]: Monad, E, I: Order, S[_]: Comonad, C: Monoid: Picker]( .toEitherT[CompileWarns] res <- EitherT( linked.toList.traverse { case (id, ctx) => - ctx.map(_.apply(id).map(id -> _).get).toValidated + ctx + .map( + /** + * NOTE: This should be safe + * as result for id should contain itself + */ + _.apply(id).map(id -> _).get + ) + .toValidated }.map(_.sequence.toEither) ) } yield res.toMap From cb081c8ecc055d34d05c1c431fa1be3294a22115 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 11:11:03 +0000 Subject: [PATCH 27/31] Refactor span rendering --- io/src/main/scala/aqua/Rendering.scala | 4 +- .../scala/aqua/parser/lift/FileSpan.scala | 12 +- .../main/scala/aqua/parser/lift/Span.scala | 134 ++++++++++++------ 3 files changed, 98 insertions(+), 52 deletions(-) diff --git a/io/src/main/scala/aqua/Rendering.scala b/io/src/main/scala/aqua/Rendering.scala index 9f7d3ba30..9138746f1 100644 --- a/io/src/main/scala/aqua/Rendering.scala +++ b/io/src/main/scala/aqua/Rendering.scala @@ -15,12 +15,12 @@ import cats.{Eval, Show} object Rendering { - def showForConsole(typ: String, span: FileSpan, messages: List[String]): String = + def showForConsole(messageType: String, span: FileSpan, messages: List[String]): String = span .focus(3) .map( _.toConsoleStr( - typ, + messageType, messages, Console.RED ) diff --git a/parser/src/main/scala/aqua/parser/lift/FileSpan.scala b/parser/src/main/scala/aqua/parser/lift/FileSpan.scala index b1d876e12..e1a2b0c83 100644 --- a/parser/src/main/scala/aqua/parser/lift/FileSpan.scala +++ b/parser/src/main/scala/aqua/parser/lift/FileSpan.scala @@ -9,8 +9,14 @@ import scala.language.implicitConversions // TODO: move FileSpan to another package? case class FileSpan(name: String, locationMap: Eval[LocationMap], span: Span) { + /** + * Focus on the line pointed by the span + * + * @param ctx How many lines to capture before and after the line + * @return FileSpan.Focus + */ def focus(ctx: Int): Option[FileSpan.Focus] = - span.focus(locationMap, ctx).map(FileSpan.Focus(name, locationMap, ctx, _)) + span.focus(locationMap.value, ctx).map(FileSpan.Focus(name, locationMap, ctx, _)) } object FileSpan { @@ -18,12 +24,12 @@ object FileSpan { case class Focus(name: String, locationMap: Eval[LocationMap], ctx: Int, spanFocus: Span.Focus) { def toConsoleStr( - typ: String, + messageType: String, msgs: List[String], onLeft: String, onRight: String = Console.RESET ): String = - onLeft + "---- " + typ + ": " + s"$name:${spanFocus.line._1 + 1}:${spanFocus.column + 1}" + onRight + + onLeft + "---- " + messageType + ": " + s"$name:${spanFocus.focus.number + 1}:${spanFocus.column + 1}" + onRight + spanFocus.toConsoleStr( msgs, onLeft, diff --git a/parser/src/main/scala/aqua/parser/lift/Span.scala b/parser/src/main/scala/aqua/parser/lift/Span.scala index 661a1a0a8..feb0c1885 100644 --- a/parser/src/main/scala/aqua/parser/lift/Span.scala +++ b/parser/src/main/scala/aqua/parser/lift/Span.scala @@ -1,86 +1,127 @@ package aqua.parser.lift import cats.data.NonEmptyList -import cats.parse.{LocationMap, Parser0, Parser as P} +import cats.parse.{LocationMap, Parser as P, Parser0} import cats.{Comonad, Eval} import scala.language.implicitConversions case class Span(startIndex: Int, endIndex: Int) { - def focus(locationMap: Eval[LocationMap], ctx: Int): Option[Span.Focus] = { - val map = locationMap.value - map.toLineCol(startIndex).flatMap { case (line, column) => - map - .getLine(line) - .map { l => - val pre = - (Math.max(0, line - ctx) until line).map(i => map.getLine(i).map(i -> _)).toList.flatten - val linePos = { - val (l1, l2) = l.splitAt(column) - val (lc, l3) = l2.splitAt(endIndex - startIndex) - (line, l1, lc, l3) - } - val post = - ((line + 1) to (line + ctx)).map(i => map.getLine(i).map(i -> _)).toList.flatten - Span.Focus( - pre, - linePos, - post, - column - ) - } - } - } + /** + * Focus on the line pointed by the span + * + * @param locationMap Locations Map + * @param ctx how many lines to capture before and after the line + * @return Span.Focus + */ + def focus(locationMap: LocationMap, ctx: Int): Option[Span.Focus] = + for { + lineCol <- locationMap.toLineCol(startIndex) + (lineNum, columnNum) = lineCol + line <- locationMap.getLine(lineNum) + focused = Span.focus(line, columnNum, endIndex - startIndex) + pre = Span.getLines(locationMap, lineNum - ctx, lineNum) + post = Span.getLines(locationMap, lineNum + 1, lineNum + ctx + 1) + } yield Span.Focus( + pre, + focused.numbered(lineNum), + post, + columnNum + ) } object Span { + private def getLines( + locationMap: LocationMap, + from: Int, + to: Int + ): List[NumberedLine[String]] = + (from until to) + .map(i => + locationMap + .getLine(i) + .map(NumberedLine(i, _)) + ) + .toList + .flatten + + private def focus( + str: String, + idx: Int, + len: Int + ): FocusedLine = FocusedLine( + str.substring(0, idx), + str.substring(idx, idx + len), + str.substring(idx + len) + ) + + final case class NumberedLine[T]( + number: Int, + line: T + ) + + final case class FocusedLine( + pre: String, + focus: String, + post: String + ) { + + def numbered(n: Int): NumberedLine[FocusedLine] = + NumberedLine(n, this) + } + case class Focus( - pre: List[(Int, String)], - line: (Int, String, String, String), - post: List[(Int, String)], + pre: List[NumberedLine[String]], + focus: NumberedLine[FocusedLine], + post: List[NumberedLine[String]], column: Int ) { - private lazy val lastN = post.lastOption.map(_._1).getOrElse(line._1) + 1 + private lazy val lastN = post.lastOption.map(_.number).getOrElse(focus.number) + 1 private lazy val lastNSize = lastN.toString.length - private def formatLine(l: (Int, String), onLeft: String, onRight: String) = - formatLN(l._1, onLeft, onRight) + l._2 + private def formatLine(l: NumberedLine[String], onLeft: String, onRight: String) = + formatLN(l.number, onLeft, onRight) + l.line private def formatLN(ln: Int, onLeft: String, onRight: String) = { val s = (ln + 1).toString onLeft + s + (" " * (lastNSize - s.length)) + onRight + " " } + /** + * Format the focus for console output + * + * @param msgs Messages to display + * @param onLeft Control sequence to put on the left + * @param onRight Control sequence to put on the right + */ def toConsoleStr( msgs: List[String], onLeft: String, onRight: String = Console.RESET ): String = { - val line3Length = line._3.length - val line3Mult = if (line3Length == 0) 1 else line3Length - val message = msgs.map(m => (" " * (line._2.length + lastNSize + 1)) + m).mkString("\n") + val focusLength = focus.line.focus.length + val focusMult = if (focusLength == 0) 1 else focusLength + val message = msgs + .map(m => (" " * (focus.line.focus.length + lastNSize + 1)) + m) + .mkString("\n") pre.map(formatLine(_, onLeft, onRight)).mkString("\n") + "\n" + - formatLN(line._1, onLeft, onRight) + - line._2 + - onLeft + - line._3 + - onRight + - line._4 + + formatLN(focus.number, onLeft, onRight) + + focus.line.pre + + onLeft + focus.line.focus + onRight + + focus.line.post + "\n" + - (" " * (line._2.length + lastNSize + 1)) + + (" " * (focus.line.pre.length + lastNSize + 1)) + onLeft + - ("^" * line3Mult) + - ("=" * line._4.length) + + ("^" * focusMult) + + ("=" * focus.line.post.length) + onRight + "\n" + - onLeft + - message + - onRight + + onLeft + message + onRight + "\n" + post.map(formatLine(_, onLeft, onRight)).mkString("\n") } @@ -104,7 +145,6 @@ object Span { def lift0: Parser0[Span.S[T]] = Span.spanLiftParser.lift0(p) } - implicit object spanLiftParser extends LiftParser[S] { override def lift[T](p: P[T]): P[S[T]] = From a2e3f592c0626384f59a227e31e6f231f9241820 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 11:34:09 +0000 Subject: [PATCH 28/31] Remove variable name warning --- .../scala/aqua/semantics/rules/names/NamesInterpreter.scala | 6 +----- 1 file changed, 1 insertion(+), 5 deletions(-) diff --git a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala index a2811dd79..bd718aea1 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/names/NamesInterpreter.scala @@ -103,11 +103,7 @@ class NamesInterpreter[S[_], X](using } case None => mapStackHeadM(report.error(name, "Cannot define a variable in the root scope").as(false))( - fr => - report - .warning(name, "Variable name is too short.") - .whenA(name.value.length < 3) - .as(fr.addName(name, `type`) -> true) + fr => (fr.addName(name, `type`) -> true).pure ) <* locations.addToken(name.value, name) } From eded8e8acd5c4bfa42be12d4d6c590fb17cc1396 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 11:52:00 +0000 Subject: [PATCH 29/31] Add warning on unused call results --- .../semantics/expr/func/CallArrowSem.scala | 17 +++++------- .../semantics/rules/types/TypesAlgebra.scala | 6 ++++- .../rules/types/TypesInterpreter.scala | 26 +++++++++++++++---- 3 files changed, 33 insertions(+), 16 deletions(-) diff --git a/semantics/src/main/scala/aqua/semantics/expr/func/CallArrowSem.scala b/semantics/src/main/scala/aqua/semantics/expr/func/CallArrowSem.scala index 14b24a3f3..970fb5f87 100644 --- a/semantics/src/main/scala/aqua/semantics/expr/func/CallArrowSem.scala +++ b/semantics/src/main/scala/aqua/semantics/expr/func/CallArrowSem.scala @@ -10,19 +10,21 @@ import aqua.semantics.rules.ValuesAlgebra import aqua.semantics.rules.names.NamesAlgebra import aqua.semantics.rules.types.TypesAlgebra import aqua.types.{StreamType, Type} + import cats.Monad import cats.syntax.flatMap.* import cats.syntax.functor.* import cats.syntax.traverse.* import cats.syntax.option.* import cats.syntax.applicative.* +import cats.syntax.apply.* import cats.syntax.comonad.* class CallArrowSem[S[_]](val expr: CallArrowExpr[S]) extends AnyVal { import expr.* - private def getExports[Alg[_]: Monad](callArrow: CallArrowRaw)(implicit + private def getExports[Alg[_]: Monad](callArrow: CallArrowRaw)(using N: NamesAlgebra[S, Alg], T: TypesAlgebra[S, Alg] ): Alg[List[Call.Export]] = @@ -42,16 +44,11 @@ class CallArrowSem[S[_]](val expr: CallArrowExpr[S]) extends AnyVal { ): Alg[Option[FuncOp]] = for { // TODO: Accept other expressions callArrowRaw <- V.valueToCallArrowRaw(expr.callArrow) - maybeOp <- callArrowRaw.traverse(car => - variables - .drop(car.baseType.codomain.length) - .headOption - .fold(getExports(car))( - T.expectNoExport(_).as(Nil) - ) - .map(maybeExports => CallArrowRawTag(maybeExports, car).funcOpLeaf) + tag <- callArrowRaw.traverse(car => + getExports(car).map(CallArrowRawTag(_, car)) <* + T.checkArrowCallResults(callArrow, car.baseType, variables) ) - } yield maybeOp + } yield tag.map(_.funcOpLeaf) def program[Alg[_]: Monad](implicit N: NamesAlgebra[S, Alg], diff --git a/semantics/src/main/scala/aqua/semantics/rules/types/TypesAlgebra.scala b/semantics/src/main/scala/aqua/semantics/rules/types/TypesAlgebra.scala index 725db4aff..3fc8476b8 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/types/TypesAlgebra.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/types/TypesAlgebra.scala @@ -61,7 +61,11 @@ trait TypesAlgebra[S[_], Alg[_]] { givenType: Type ): Alg[Option[Type]] - def expectNoExport(token: Token[S]): Alg[Unit] + def checkArrowCallResults( + token: Token[S], + arrowType: ArrowType, + results: List[Name[S]] + ): Alg[Unit] def checkArgumentsNumber(token: Token[S], expected: Int, givenNum: Int): Alg[Boolean] diff --git a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala index f5c78b764..08a3a0a86 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala @@ -405,13 +405,29 @@ class TypesInterpreter[S[_], X](using .as(none) )(_.some.pure) - override def expectNoExport(token: Token[S]): State[X, Unit] = - report - .error( + override def checkArrowCallResults( + token: Token[S], + arrowType: ArrowType, + results: List[Name[S]] + ): State[X, Unit] = for { + _ <- results + .drop(arrowType.codomain.length) + .traverse_(result => + report + .error( + result, + "Types mismatch. Cannot assign to a variable " + + "the result of a call that returns nothing" + ) + ) + _ <- report + .warning( token, - "Types mismatch. Cannot assign to a variable the result of a call that returns nothing" + s"Arrow returns ${arrowType.codomain.length} values, " + + s"but only ${results.length} are used" ) - .as(()) + .whenA(arrowType.codomain.length > results.length) + } yield () override def checkArgumentsNumber( token: Token[S], From 32772746f3f9de33b8c948de9979612210cb95d4 Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 12:49:41 +0000 Subject: [PATCH 30/31] Add unit tests --- .../rules/types/TypesInterpreter.scala | 11 ++++- .../scala/aqua/semantics/SemanticsSpec.scala | 44 +++++++++++++++---- 2 files changed, 45 insertions(+), 10 deletions(-) diff --git a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala index 08a3a0a86..c6263495f 100644 --- a/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala +++ b/semantics/src/main/scala/aqua/semantics/rules/types/TypesInterpreter.scala @@ -423,8 +423,15 @@ class TypesInterpreter[S[_], X](using _ <- report .warning( token, - s"Arrow returns ${arrowType.codomain.length} values, " + - s"but only ${results.length} are used" + s"Arrow returns ${arrowType.codomain.length match { + case 0 => "no values" + case 1 => "a value" + case i => s"$i values" + }} values, but ${results.length match { + case 0 => "none are" + case 1 => "only one is" + case i => s"only $i are" + }} used" ) .whenA(arrowType.codomain.length > results.length) } yield () diff --git a/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala b/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala index 35e25f731..70531d5a8 100644 --- a/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala +++ b/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala @@ -13,8 +13,7 @@ import org.scalatest.flatspec.AnyFlatSpec import org.scalatest.matchers.should.Matchers import org.scalatest.Inside import cats.~> -import cats.data.Chain -import cats.data.NonEmptyChain +import cats.data.{Chain, EitherNec, NonEmptyChain} import cats.syntax.show.* import cats.syntax.traverse.* import cats.syntax.foldable.* @@ -32,13 +31,23 @@ class SemanticsSpec extends AnyFlatSpec with Matchers with Inside { val semantics = new RawSemantics[Span.S]() + def insideResult(script: String)( + test: PartialFunction[ + ( + Chain[SemanticWarning[Span.S]], + EitherNec[SemanticError[Span.S], RawContext] + ), + Any + ] + ): Unit = inside(parser(script)) { case Validated.Valid(ast) => + val init = RawContext.blank + inside(semantics.process(ast, init).value.run)(test) + } + def insideBody(script: String)(test: RawTag.Tree => Any): Unit = - inside(parser(script)) { case Validated.Valid(ast) => - val init = RawContext.blank - inside(semantics.process(ast, init).value.value) { case Right(ctx) => - inside(ctx.funcs.headOption) { case Some((_, func)) => - test(func.arrow.body) - } + insideResult(script) { case (_, Right(ctx)) => + inside(ctx.funcs.headOption) { case Some((_, func)) => + test(func.arrow.body) } } @@ -648,4 +657,23 @@ class SemanticsSpec extends AnyFlatSpec with Matchers with Inside { atLeast(1, errors.toChain.toList) shouldBe a[RulesViolated[Span.S]] } } + + it should "produce warning on unused call results" in { + val script = """|func test() -> string, string: + | stream: *string + | stream <<- "a" + | stream <<- "b" + | <- stream[0], stream[1] + | + |func main() -> string: + | a <- test() + | <- a + |""".stripMargin + + println(script) + + insideResult(script) { case (warnings, Right(_)) => + warnings.exists(_.hints.exists(_.contains("used"))) should be(true) + } + } } From 98e8258bb518d41162ade083648fca158a4d990c Mon Sep 17 00:00:00 2001 From: InversionSpaces Date: Mon, 25 Sep 2023 12:50:46 +0000 Subject: [PATCH 31/31] Remove println --- semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala | 2 -- 1 file changed, 2 deletions(-) diff --git a/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala b/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala index 70531d5a8..15bb93c5c 100644 --- a/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala +++ b/semantics/src/test/scala/aqua/semantics/SemanticsSpec.scala @@ -670,8 +670,6 @@ class SemanticsSpec extends AnyFlatSpec with Matchers with Inside { | <- a |""".stripMargin - println(script) - insideResult(script) { case (warnings, Right(_)) => warnings.exists(_.hints.exists(_.contains("used"))) should be(true) }