Skip to content

Commit

Permalink
Fix the array visualisation and misc crashes (#1588)
Browse files Browse the repository at this point in the history
  • Loading branch information
iamrecursion committed Mar 19, 2021
1 parent c6ddc76 commit 2932819
Show file tree
Hide file tree
Showing 27 changed files with 197 additions and 41 deletions.
2 changes: 2 additions & 0 deletions distribution/std-lib/Standard/src/Base.enso
Original file line number Diff line number Diff line change
@@ -1,4 +1,5 @@
import Standard.Base.Data.Any.Extensions
import Standard.Base.Data.Array.Extensions
import Standard.Base.Data.Interval
import Standard.Base.Data.Json
import Standard.Base.Data.List
Expand Down Expand Up @@ -34,6 +35,7 @@ export Standard.Base.Meta
export Standard.Base.System.File

from Standard.Base.Data.Any.Extensions export all
from Standard.Base.Data.Array.Extensions export all
from Standard.Base.Data.List export Nil, Cons
from Standard.Base.Data.Number.Extensions export all hiding Math, String, Double
from Standard.Base.Data.Noise export all hiding Noise
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,8 @@
from Standard.Base import all

## Transform the array into text for displaying as part of its default
visualization.
Array.to_default_visualization_data : Text
Array.to_default_visualization_data =
Vector.Vector this . to_default_visualization_data

13 changes: 13 additions & 0 deletions distribution/std-lib/Standard/src/Base/Data/Json.enso
Original file line number Diff line number Diff line change
Expand Up @@ -81,6 +81,10 @@ type Json
Check the `message` field for detailed information on the specific failure.
type Parse_Error message

## Converts the error to a display representation.
Parse_Error.to_display_text : Text
Parse_Error.to_display_text = "Parse error in parsing JSON: " + this.message.to_text + "."

## Gets the value associated with the given key in this object. Returns
`Nothing` if the associated key is not defined.
Object.get : Text -> Json | Nothing
Expand Down Expand Up @@ -110,6 +114,15 @@ type Marshalling_Error
when the JSON does not contain all the fields required by the atom.
type Missing_Field_Error json field format

to_display_text : Text
to_display_text = case this of
Type_Mismatch_Error json format ->
json_text = Meta.type_to_display_text json
format_text = Meta.type_to_display_text format
"Type mismatch error: the json with type `" + json_text + "` did not match the format `" + format_text + "`."
Missing_Field_Error _ field _ ->
"Missing field in Json: the field `" + field.to_text "` was missing in the json."

## Generically converts an atom into a JSON object.

The input atom is converted into a JSON object, with a `"type"` field set to
Expand Down
7 changes: 7 additions & 0 deletions distribution/std-lib/Standard/src/Base/Data/Vector.enso
Original file line number Diff line number Diff line change
Expand Up @@ -523,6 +523,13 @@ type Vector

Vector new_vec_arr

## Transform the vector into text for displaying as part of its default
visualization.
to_default_visualization_data : Text
to_default_visualization_data =
json = this.take_start 100 . to_json
json.to_text

## A builder type for Enso vectors.

A vector builder is a mutable data structure, that allows to gather a
Expand Down
7 changes: 7 additions & 0 deletions distribution/std-lib/Standard/src/Base/Meta.enso
Original file line number Diff line number Diff line change
Expand Up @@ -147,3 +147,10 @@ is_same_object value_1 value_2 = Builtins.Meta.is_same_object value_1 value_2
get_source_location : Integer -> Text
get_source_location skip_frames =
Builtins.Meta.get_source_location skip_frames+1

## Displays the type of the provided value as text.

Arguments:
- value: The value for which to display the type.
type_to_display_text : Any -> Text
type_to_display_text value = Builtins.Meta.type_to_display_text value
2 changes: 1 addition & 1 deletion distribution/std-lib/Standard/src/Base/System/File.enso
Original file line number Diff line number Diff line change
Expand Up @@ -300,7 +300,7 @@ new path = File (Prim_Io.get_file path)

> Example
Read the `data.csv` file in the project directory.
File.open (Enso_Project.data / "data.csv")
File.read (Enso_Project.data / "data.csv")
read : Text -> Text
read path = (here.new path).read

Expand Down
4 changes: 2 additions & 2 deletions distribution/std-lib/Standard/src/Test.enso
Original file line number Diff line number Diff line change
Expand Up @@ -179,8 +179,8 @@ run_spec ~behavior =
case ex of
Failure _ -> ex
Finished_With_Error err stack_trace_text ->
Failure ("An unexpected error was returned: " + err.to_text + '\n' + stack_trace_text)
_ -> Failure ("An unexpected panic was thrown: " + ex.to_text + '\n' + maybeExc.get_stack_trace_text)
Failure ("An unexpected error was returned: " + err.to_display_text + '\n' + stack_trace_text)
_ -> Failure ("An unexpected panic was thrown: " + ex.to_display_text + '\n' + maybeExc.get_stack_trace_text)
result

## Creates a new test group, desribing properties of the object
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,26 +3,27 @@
/** Container for polyglot method names */
public class MethodNames {
public static class TopScope {
public static final String GET_MODULE = "get_module";
public static final String CREATE_MODULE = "create_module";
public static final String GET_MODULE = "get_module";
public static final String LEAK_CONTEXT = "leak_context";
public static final String REGISTER_MODULE = "register_module";
public static final String UNREGISTER_MODULE = "unregister_module";
public static final String LEAK_CONTEXT = "leak_context";
}

public static class Module {
public static final String EVAL_EXPRESSION = "eval_expression";
public static final String GET_ASSOCIATED_CONSTRUCTOR = "get_associated_constructor";
public static final String GET_METHOD = "get_method";
public static final String GET_CONSTRUCTOR = "get_constructor";
public static final String GET_METHOD = "get_method";
public static final String GET_NAME = "get_name";
public static final String REPARSE = "reparse";
public static final String SET_SOURCE = "set_source";
public static final String SET_SOURCE_FILE = "set_source_file";
public static final String EVAL_EXPRESSION = "eval_expression";
}

public static class Function {
public static final String EQUALS = "equals";
public static final String GET_SOURCE_START = "get_source_start";
public static final String GET_SOURCE_LENGTH = "get_source_length";
public static final String GET_SOURCE_START = "get_source_start";
}
}
17 changes: 14 additions & 3 deletions engine/polyglot-api/src/main/scala/org/enso/polyglot/Module.scala
Original file line number Diff line number Diff line change
Expand Up @@ -9,12 +9,17 @@ import org.graalvm.polyglot.Value
class Module(private val value: Value) {
import MethodNames.Module._

/** @return the name of the module
*/
def getName: String = value.invokeMember(GET_NAME).asString()

/** @return the associated type of this module
*/
def getAssociatedConstructor: Value =
value.invokeMember(GET_ASSOCIATED_CONSTRUCTOR)

/** Gets a constructor definition by name
/** Gets a constructor definition by name.
*
* @param name the constructor name
* @return the polyglot representation of the constructor.
*/
Expand All @@ -27,11 +32,17 @@ class Module(private val value: Value) {
* @param name the name of the method
* @return the runtime representation of the method
*/
def getMethod(constructor: Value, name: String): Function =
new Function(value.invokeMember(GET_METHOD, constructor, name))
def getMethod(constructor: Value, name: String): Option[Function] = {
val newVal = value.invokeMember(GET_METHOD, constructor, name);
if (newVal.isNull) { None }
else {
Some(new Function(newVal))
}
}

/** Evaluates an arbitrary expression as if it were placed in a function
* body inside this module.
*
* @param code the expression to evaluate
* @return the return value of the expression
*/
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ class ApiTest extends AnyFlatSpec with Matchers {
|""".stripMargin
val module = executionContext.evalModule(code, "Test")
val associatedConstructor = module.getAssociatedConstructor
val barFunction = module.getMethod(associatedConstructor, "bar")
val barFunction = module.getMethod(associatedConstructor, "bar").get
val result = barFunction.execute(
associatedConstructor.newInstance(),
10L.asInstanceOf[AnyRef]
Expand All @@ -39,7 +39,7 @@ class ApiTest extends AnyFlatSpec with Matchers {
|""".stripMargin
val module = executionContext.evalModule(code, "Test")
val vectorCons = module.getConstructor("Vector")
val squareNorm = module.getMethod(vectorCons, "squareNorm")
val squareNorm = module.getMethod(vectorCons, "squareNorm").get
val testVector = vectorCons.newInstance(
1L.asInstanceOf[AnyRef],
2L.asInstanceOf[AnyRef],
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,7 @@ class ModuleManagementTest extends AnyFlatSpec with Matchers {

val mainModule = ctx.executionContext.getTopScope.getModule("Test.Main")
val assocCons = mainModule.getAssociatedConstructor
val mainFun1 = mainModule.getMethod(assocCons, "main")
val mainFun1 = mainModule.getMethod(assocCons, "main").get

mainFun1.execute(assocCons).asLong() shouldEqual 12345L

Expand All @@ -53,7 +53,7 @@ class ModuleManagementTest extends AnyFlatSpec with Matchers {
|""".stripMargin)

mainModule.reparse()
val mainFun2 = mainModule.getMethod(assocCons, "main")
val mainFun2 = mainModule.getMethod(assocCons, "main").get
mainFun2.execute(assocCons).asLong() shouldEqual 4567L
}

Expand All @@ -66,28 +66,28 @@ class ModuleManagementTest extends AnyFlatSpec with Matchers {

val mainModule = ctx.executionContext.getTopScope.getModule("Test.Main")
val assocCons = mainModule.getAssociatedConstructor
val mainFun1 = mainModule.getMethod(assocCons, "main")
val mainFun1 = mainModule.getMethod(assocCons, "main").get

mainFun1.execute(assocCons).asLong() shouldEqual 123L

mainModule.setSource("""
|main = 456
|""".stripMargin)
val mainFun2 = mainModule.getMethod(assocCons, "main")
val mainFun2 = mainModule.getMethod(assocCons, "main").get
mainFun2.execute(assocCons).asLong() shouldEqual 456L

mainModule.setSource("""
|main = 789
|""".stripMargin)
val mainFun3 = mainModule.getMethod(assocCons, "main")
val mainFun3 = mainModule.getMethod(assocCons, "main").get
mainFun3.execute(assocCons).asLong() shouldEqual 789L

ctx.writeMain("""
|main = 987
|""".stripMargin)

mainModule.setSourceFile(ctx.pkg.mainFile.getAbsolutePath)
val mainFun4 = mainModule.getMethod(assocCons, "main")
val mainFun4 = mainModule.getMethod(assocCons, "main").get
mainFun4.execute(assocCons).asLong() shouldEqual 987L
}

Expand All @@ -111,7 +111,7 @@ class ModuleManagementTest extends AnyFlatSpec with Matchers {

val mainModule = topScope.getModule("Test.Main")
val assocCons = mainModule.getAssociatedConstructor
val mainFun = mainModule.getMethod(assocCons, "main")
val mainFun = mainModule.getMethod(assocCons, "main").get
mainFun.execute(assocCons).asLong shouldEqual 11L
}

Expand Down Expand Up @@ -141,7 +141,7 @@ class ModuleManagementTest extends AnyFlatSpec with Matchers {

val mainModule = topScope.getModule("Test.Main")
val assocCons = mainModule.getAssociatedConstructor
val mainFun = mainModule.getMethod(assocCons, "main")
val mainFun = mainModule.getMethod(assocCons, "main").get
mainFun.execute(assocCons).asLong shouldEqual 21L
}

Expand All @@ -161,7 +161,7 @@ class ModuleManagementTest extends AnyFlatSpec with Matchers {
"X"
)
val mod1AssocCons = mod1.getAssociatedConstructor
val mod1Main = mod1.getMethod(mod1AssocCons, "bar")
val mod1Main = mod1.getMethod(mod1AssocCons, "bar").get
mod1Main.execute(mod1AssocCons).asLong shouldEqual 124

ctx.executionContext.getTopScope.unregisterModule("Test.Main")
Expand Down
19 changes: 13 additions & 6 deletions engine/runner/src/main/scala/org/enso/runner/Main.scala
Original file line number Diff line number Diff line change
@@ -1,8 +1,5 @@
package org.enso.runner

import java.io.File
import java.util.UUID

import akka.http.scaladsl.model.{IllegalUriException, Uri}
import cats.implicits._
import org.apache.commons.cli.{Option => CliOption, _}
Expand All @@ -12,8 +9,11 @@ import org.enso.loggingservice.LogLevel
import org.enso.pkg.{Contact, PackageManager, SemVerEnsoVersion}
import org.enso.polyglot.{LanguageInfo, Module, PolyglotContext}
import org.enso.version.VersionDescription
import org.graalvm.polyglot.{PolyglotException, Value}
import org.graalvm.polyglot.PolyglotException

import java.io.File
import java.util.UUID
import scala.Console.err
import scala.jdk.CollectionConverters._
import scala.util.Try

Expand Down Expand Up @@ -382,11 +382,18 @@ object Main {
mainModule: Module,
rootPkgPath: Option[File],
mainMethodName: String = "main"
): Value = {
): Unit = {
val mainCons = mainModule.getAssociatedConstructor
val mainFun = mainModule.getMethod(mainCons, mainMethodName)
try {
mainFun.execute(mainCons.newInstance())
mainFun match {
case Some(main) => main.execute(mainCons.newInstance())
case None =>
err.println(
s"The module ${mainModule.getName} does not contain a `main` " +
s"function. It could not be run."
)
}
} catch {
case e: PolyglotException =>
printPolyglotException(e, rootPkgPath)
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,16 @@
package org.enso.interpreter.node.expression.builtin.meta;

import com.oracle.truffle.api.CompilerDirectives.CompilationFinal;
import com.oracle.truffle.api.nodes.Node;
import org.enso.interpreter.dsl.BuiltinMethod;
import org.enso.interpreter.node.expression.builtin.text.util.TypeToDisplayTextNode;
import org.enso.interpreter.runtime.data.text.Text;

@BuiltinMethod(type = "Meta", name = "type_to_display_text", description = "Pretty prints a type.")
public class DisplayTypeNode extends Node {
@Child @CompilationFinal TypeToDisplayTextNode displayTypeNode = TypeToDisplayTextNode.build();

Text execute(Object _this, Object value) {
return Text.create(displayTypeNode.execute(value));
}
}
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
package org.enso.interpreter.node.expression.builtin.meta;

import com.oracle.truffle.api.dsl.Cached;
import com.oracle.truffle.api.dsl.CachedContext;
import com.oracle.truffle.api.dsl.Specialization;
import com.oracle.truffle.api.nodes.Node;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,15 @@
public abstract class TypeToDisplayTextNode extends Node {
public abstract String execute(Object o);

/**
* Create a node that can display types as text.
*
* @return a new type display node
*/
public static TypeToDisplayTextNode build() {
return TypeToDisplayTextNodeGen.create();
}

@Specialization
@CompilerDirectives.TruffleBoundary
String doDisplay(
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@
import java.io.File;
import java.io.IOException;

import java.util.Map;
import org.enso.compiler.core.IR;
import org.enso.compiler.phase.StubIrBuilder;
import org.enso.interpreter.Language;
Expand Down Expand Up @@ -404,8 +405,11 @@ static Object doInvoke(
throws UnknownIdentifierException, ArityException, UnsupportedTypeException {
ModuleScope scope = module.compileScope(context);
switch (member) {
case MethodNames.Module.GET_NAME:
return module.getName().toString();
case MethodNames.Module.GET_METHOD:
return getMethod(scope, arguments);
Function result = getMethod(scope, arguments);
return result == null ? context.getBuiltins().nothing().newInstance() : result;
case MethodNames.Module.GET_CONSTRUCTOR:
return getConstructor(scope, arguments);
case MethodNames.Module.REPARSE:
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,7 @@ public Meta(Language language, ModuleScope scope) {
AtomConstructor meta = new AtomConstructor("Meta", scope).initializeFields();
scope.registerConstructor(meta);

scope.registerMethod(meta, "type_to_display_text", DisplayTypeMethodGen.makeFunction(language));
scope.registerMethod(
meta, "is_unresolved_symbol", IsUnresolvedSymbolMethodGen.makeFunction(language));
scope.registerMethod(
Expand Down Expand Up @@ -50,6 +51,7 @@ public Meta(Language language, ModuleScope scope) {
meta, "get_polyglot_language", GetPolyglotLanguageMethodGen.makeFunction(language));

scope.registerMethod(meta, "is_same_object", IsSameObjectMethodGen.makeFunction(language));
scope.registerMethod(meta, "get_source_location", GetSourceLocationMethodGen.makeFunction(language));
scope.registerMethod(
meta, "get_source_location", GetSourceLocationMethodGen.makeFunction(language));
}
}
Loading

0 comments on commit 2932819

Please sign in to comment.