diff --git a/Dockerfile b/Dockerfile
new file mode 100644
index 0000000..093b320
--- /dev/null
+++ b/Dockerfile
@@ -0,0 +1,4 @@
+FROM openjdk:17-jdk-slim
+COPY target/*.jar app.jar
+EXPOSE 8080
+ENTRYPOINT ["java", "-jar", "/app.jar"]
\ No newline at end of file
diff --git a/pom.xml b/pom.xml
index 2b4df90..861bd33 100644
--- a/pom.xml
+++ b/pom.xml
@@ -88,6 +88,35 @@
vaadin-testbench-junit5
test
+
+ net.openhft
+ compiler
+ 2.26ea0
+
+
+ com.hilerio
+ ace-widget
+ 2.0.0
+
+
+ io.hypersistence
+ hypersistence-utils-hibernate-63
+ 3.7.3
+
+
+ in.virit
+ viritin
+ 2.8.22
+
+
+ com.flowingcode.addons
+ simple-timer
+ 2.2.0
+
+
+ org.projectlombok
+ lombok
+
diff --git a/src/main/bundles/README.md b/src/main/bundles/README.md
new file mode 100644
index 0000000..c9737d1
--- /dev/null
+++ b/src/main/bundles/README.md
@@ -0,0 +1,32 @@
+This directory is automatically generated by Vaadin and contains the pre-compiled
+frontend files/resources for your project (frontend development bundle).
+
+It should be added to Version Control System and committed, so that other developers
+do not have to compile it again.
+
+Frontend development bundle is automatically updated when needed:
+- an npm/pnpm package is added with @NpmPackage or directly into package.json
+- CSS, JavaScript or TypeScript files are added with @CssImport, @JsModule or @JavaScript
+- Vaadin add-on with front-end customizations is added
+- Custom theme imports/assets added into 'theme.json' file
+- Exported web component is added.
+
+If your project development needs a hot deployment of the frontend changes,
+you can switch Flow to use Vite development server (default in Vaadin 23.3 and earlier versions):
+- set `vaadin.frontend.hotdeploy=true` in `application.properties`
+- configure `vaadin-maven-plugin`:
+```
+
+ true
+
+```
+- configure `jetty-maven-plugin`:
+```
+
+
+ true
+
+
+```
+
+Read more [about Vaadin development mode](https://vaadin.com/docs/next/flow/configuration/development-mode#precompiled-bundle).
\ No newline at end of file
diff --git a/src/main/bundles/dev.bundle b/src/main/bundles/dev.bundle
new file mode 100644
index 0000000..14f10fc
Binary files /dev/null and b/src/main/bundles/dev.bundle differ
diff --git a/src/main/frontend/index.html b/src/main/frontend/index.html
new file mode 100644
index 0000000..d36e593
--- /dev/null
+++ b/src/main/frontend/index.html
@@ -0,0 +1,23 @@
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
diff --git a/src/main/java/com/primefactorsolutions/model/AppUser.java b/src/main/java/com/primefactorsolutions/model/AppUser.java
new file mode 100644
index 0000000..73dc753
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/AppUser.java
@@ -0,0 +1,23 @@
+package com.primefactorsolutions.model;
+
+import jakarta.persistence.Entity;
+import jakarta.persistence.FetchType;
+import jakarta.persistence.OneToMany;
+import lombok.AllArgsConstructor;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import lombok.NoArgsConstructor;
+
+import java.util.List;
+
+@Entity
+@Data
+@AllArgsConstructor
+@NoArgsConstructor
+@EqualsAndHashCode(callSuper = true)
+public class AppUser extends BaseEntity {
+ private String email;
+
+ @OneToMany(fetch = FetchType.EAGER, mappedBy = "appUser")
+ private List assessments;
+}
diff --git a/src/main/java/com/primefactorsolutions/model/Assessment.java b/src/main/java/com/primefactorsolutions/model/Assessment.java
new file mode 100644
index 0000000..c682bbf
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/Assessment.java
@@ -0,0 +1,64 @@
+package com.primefactorsolutions.model;
+
+import jakarta.persistence.*;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import lombok.NoArgsConstructor;
+
+import java.time.Instant;
+import java.util.List;
+import java.util.Optional;
+
+@Entity
+@Data
+@NoArgsConstructor
+@EqualsAndHashCode(callSuper = true)
+public class Assessment extends BaseEntity {
+ @OneToMany(fetch = FetchType.EAGER)
+ private List questions;
+
+ @OneToMany(fetch = FetchType.EAGER, mappedBy = "assessment", cascade = {CascadeType.ALL})
+ private List submissions;
+
+ @ManyToOne
+ private AppUser appUser;
+
+ @OneToMany(fetch = FetchType.EAGER, mappedBy = "assessment", cascade = {CascadeType.ALL})
+ private List assessmentEvents = List.of();
+
+ public Submission getCurrentSubmission() {
+ return submissions.getLast();
+ }
+
+ public Long getRemainingTimeSeconds() {
+ final Optional started = assessmentEvents.stream().filter(e -> e.getStatus() == AssessmentStatus.STARTED)
+ .map(AssessmentEvent::getTimestamp)
+ .findFirst();
+
+ return started.map(instant -> 300L - (Instant.now().getEpochSecond() - instant.getEpochSecond()))
+ .orElse(300L);
+ }
+
+ public Instant getStartingTime() {
+ final Optional started = assessmentEvents.stream().filter(e -> e.getStatus() == AssessmentStatus.STARTED)
+ .map(AssessmentEvent::getTimestamp)
+ .findFirst();
+
+ return started.orElse(null);
+ }
+
+ public boolean isCompleted() {
+ return assessmentEvents.stream().filter(e -> e.getStatus() == AssessmentStatus.COMPLETED)
+ .map(AssessmentEvent::getTimestamp)
+ .findFirst()
+ .isPresent();
+ }
+
+ public boolean isFirst(Submission currSubmission) {
+ return getQuestions().indexOf(currSubmission.getQuestion()) == 0;
+ }
+
+ public boolean isLast(Submission currSubmission) {
+ return getQuestions().indexOf(currSubmission.getQuestion()) == getQuestions().size() - 1;
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/model/AssessmentEvent.java b/src/main/java/com/primefactorsolutions/model/AssessmentEvent.java
new file mode 100644
index 0000000..8da7084
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/AssessmentEvent.java
@@ -0,0 +1,23 @@
+package com.primefactorsolutions.model;
+
+import jakarta.persistence.Entity;
+import jakarta.persistence.ManyToOne;
+import lombok.AllArgsConstructor;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import lombok.NoArgsConstructor;
+
+import java.time.Instant;
+
+@Entity
+@Data
+@EqualsAndHashCode(callSuper = true)
+@AllArgsConstructor
+@NoArgsConstructor
+public class AssessmentEvent extends BaseEntity {
+ private Instant timestamp;
+ private AssessmentStatus status;
+
+ @ManyToOne
+ private Assessment assessment;
+}
diff --git a/src/main/java/com/primefactorsolutions/model/AssessmentStatus.java b/src/main/java/com/primefactorsolutions/model/AssessmentStatus.java
new file mode 100644
index 0000000..430e37f
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/AssessmentStatus.java
@@ -0,0 +1,7 @@
+package com.primefactorsolutions.model;
+
+public enum AssessmentStatus {
+ CREATED,
+ STARTED,
+ COMPLETED
+}
diff --git a/src/main/java/com/primefactorsolutions/model/BaseEntity.java b/src/main/java/com/primefactorsolutions/model/BaseEntity.java
new file mode 100644
index 0000000..ddc143e
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/BaseEntity.java
@@ -0,0 +1,51 @@
+package com.primefactorsolutions.model;
+
+import jakarta.persistence.GeneratedValue;
+import jakarta.persistence.GenerationType;
+import jakarta.persistence.Id;
+import jakarta.persistence.MappedSuperclass;
+import jakarta.persistence.Version;
+
+import java.util.UUID;
+
+@MappedSuperclass
+public abstract class BaseEntity {
+
+ @Id
+ @GeneratedValue(strategy = GenerationType.UUID)
+ private UUID id;
+
+ @Version
+ private int version;
+
+ public UUID getId() {
+ return id;
+ }
+
+ public void setId(final UUID id) {
+ this.id = id;
+ }
+
+ public int getVersion() {
+ return version;
+ }
+
+ @Override
+ public int hashCode() {
+ if (getId() != null) {
+ return getId().hashCode();
+ }
+ return super.hashCode();
+ }
+
+ @Override
+ public boolean equals(final Object obj) {
+ if (!(obj instanceof BaseEntity that)) {
+ return false; // null or not an AbstractEntity class
+ }
+ if (getId() != null) {
+ return getId().equals(that.getId());
+ }
+ return super.equals(that);
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/model/Question.java b/src/main/java/com/primefactorsolutions/model/Question.java
new file mode 100644
index 0000000..475713d
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/Question.java
@@ -0,0 +1,16 @@
+package com.primefactorsolutions.model;
+
+import jakarta.persistence.Entity;
+import lombok.AllArgsConstructor;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import lombok.NoArgsConstructor;
+
+@Entity
+@Data
+@AllArgsConstructor
+@NoArgsConstructor
+@EqualsAndHashCode(callSuper = true)
+public class Question extends BaseEntity{
+ private String content;
+}
diff --git a/src/main/java/com/primefactorsolutions/model/Submission.java b/src/main/java/com/primefactorsolutions/model/Submission.java
new file mode 100644
index 0000000..749c5c6
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/Submission.java
@@ -0,0 +1,34 @@
+package com.primefactorsolutions.model;
+
+import io.hypersistence.utils.hibernate.type.json.JsonType;
+import jakarta.persistence.Column;
+import jakarta.persistence.Entity;
+import jakarta.persistence.ManyToOne;
+import lombok.AllArgsConstructor;
+import lombok.Data;
+import lombok.EqualsAndHashCode;
+import lombok.NoArgsConstructor;
+import org.hibernate.annotations.Type;
+
+import java.util.Map;
+
+@Entity
+@Data
+@AllArgsConstructor
+@NoArgsConstructor
+@EqualsAndHashCode(callSuper = true)
+public class Submission extends BaseEntity {
+ @ManyToOne
+ private Question question;
+
+ private String response;
+
+ @Type(JsonType.class)
+ @Column(columnDefinition = "json")
+ private Map results;
+
+ private SubmissionStatus submissionStatus;
+
+ @ManyToOne
+ private Assessment assessment;
+}
diff --git a/src/main/java/com/primefactorsolutions/model/SubmissionStatus.java b/src/main/java/com/primefactorsolutions/model/SubmissionStatus.java
new file mode 100644
index 0000000..c6424ac
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/model/SubmissionStatus.java
@@ -0,0 +1,6 @@
+package com.primefactorsolutions.model;
+
+public enum SubmissionStatus {
+ OK,
+ FAIL
+}
diff --git a/src/main/java/com/primefactorsolutions/repositories/AssessmentRepository.java b/src/main/java/com/primefactorsolutions/repositories/AssessmentRepository.java
new file mode 100644
index 0000000..81813d8
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/repositories/AssessmentRepository.java
@@ -0,0 +1,9 @@
+package com.primefactorsolutions.repositories;
+
+import com.primefactorsolutions.model.Assessment;
+import org.springframework.data.jpa.repository.JpaRepository;
+
+import java.util.UUID;
+
+public interface AssessmentRepository extends JpaRepository {
+}
diff --git a/src/main/java/com/primefactorsolutions/repositories/UserRepository.java b/src/main/java/com/primefactorsolutions/repositories/UserRepository.java
new file mode 100644
index 0000000..70f46d0
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/repositories/UserRepository.java
@@ -0,0 +1,9 @@
+package com.primefactorsolutions.repositories;
+
+import com.primefactorsolutions.model.AppUser;
+import org.springframework.data.jpa.repository.JpaRepository;
+
+import java.util.UUID;
+
+public interface UserRepository extends JpaRepository {
+}
diff --git a/src/main/java/com/primefactorsolutions/service/AssessmentService.java b/src/main/java/com/primefactorsolutions/service/AssessmentService.java
new file mode 100644
index 0000000..f92bd9c
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/AssessmentService.java
@@ -0,0 +1,129 @@
+package com.primefactorsolutions.service;
+
+import com.primefactorsolutions.model.*;
+import com.primefactorsolutions.repositories.AssessmentRepository;
+import lombok.Data;
+import org.checkerframework.checker.units.qual.A;
+import org.springframework.stereotype.Service;
+
+import javax.swing.text.html.Option;
+import java.time.Instant;
+import java.util.List;
+import java.util.Map;
+import java.util.Optional;
+import java.util.UUID;
+
+@Service
+@Data
+public class AssessmentService {
+
+ private final AssessmentRepository assessmentRepository;
+
+ public Assessment createOrUpdate(final Assessment assessment) {
+ final Assessment saved = assessmentRepository.save(assessment);
+
+ return saved;
+ }
+
+ public List getAssessments() {
+ return assessmentRepository.findAll();
+ }
+
+ public Assessment getAssessment(final UUID id) {
+ return assessmentRepository.findById(id).orElse(null);
+ }
+
+ public Assessment startAssessment(final UUID id) {
+ final Assessment assessment = assessmentRepository.findById(id).get();
+ final Optional started = assessment.getAssessmentEvents().stream()
+ .filter(e -> e.getStatus() == AssessmentStatus.STARTED)
+ .findFirst();
+
+ if (started.isPresent()) {
+ return assessment;
+ }
+
+ assessment.getAssessmentEvents().add(new AssessmentEvent(Instant.now(), AssessmentStatus.STARTED, assessment));
+ assessmentRepository.save(assessment);
+
+ return assessment;
+ }
+
+ public Submission getNextSubmission(final Assessment assessment, final Submission currSubmission) {
+ if (currSubmission == null) {
+ if (assessment.isCompleted()) {
+ return null;
+ }
+
+ final Question firstQuestion = assessment.getQuestions().stream().findFirst().get();
+ final Submission submissionToReturn = new Submission(firstQuestion, firstQuestion.getContent(), Map.of(), SubmissionStatus.FAIL, assessment);
+ assessment.getSubmissions().add(submissionToReturn);
+ assessmentRepository.save(assessment);
+
+ return submissionToReturn;
+ }
+
+ final Question currQuestion = currSubmission.getQuestion();
+ int idx = assessment.getQuestions().indexOf(currQuestion);
+
+ if (idx == assessment.getQuestions().size() - 1) {
+ return null;
+ }
+
+ final Question nextQuestion = assessment.getQuestions().get(idx + 1);
+
+ final Optional submissionToReturn = assessment.getSubmissions().stream()
+ .filter(s -> s.getQuestion().equals(nextQuestion))
+ .findFirst();
+
+ final Submission result;
+ if (submissionToReturn.isEmpty()) {
+ result = new Submission(nextQuestion, nextQuestion.getContent(), Map.of(), SubmissionStatus.FAIL, assessment);
+ assessment.getSubmissions().add(result);
+ } else {
+ result = submissionToReturn.get();
+ }
+
+ assessmentRepository.save(assessment);
+
+ return result;
+ }
+
+ public Submission getPrevSubmission(final Assessment assessment, final Submission currSubmission) {
+ if (currSubmission == null) {
+ return null;
+ }
+
+ final Question currQuestion = currSubmission.getQuestion();
+ int idx = assessment.getQuestions().indexOf(currQuestion);
+
+ if (idx == 0) {
+ return null;
+ }
+
+ final Question prevQuestion = assessment.getQuestions().get(idx - 1);
+
+ assessmentRepository.save(assessment);
+
+
+ return assessment.getSubmissions().stream()
+ .filter(s -> s.getQuestion().equals(prevQuestion))
+ .findFirst().orElseThrow(() -> new IllegalStateException("submission invalid"));
+ }
+
+ public Assessment completeAssessment(final UUID id, final Submission submission) {
+ Assessment assessment = assessmentRepository.findById(id).get();
+ Optional completed = assessment.getAssessmentEvents().stream()
+ .filter(e -> e.getStatus() == AssessmentStatus.COMPLETED)
+ .findFirst();
+
+ if (completed.isPresent()) {
+ return assessment;
+ }
+
+ assessment.getAssessmentEvents().add(new AssessmentEvent(Instant.now(), AssessmentStatus.COMPLETED, assessment));
+ assessmentRepository.save(assessment);
+
+ return assessment;
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/service/CompilerService.java b/src/main/java/com/primefactorsolutions/service/CompilerService.java
new file mode 100644
index 0000000..f1440cd
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/CompilerService.java
@@ -0,0 +1,55 @@
+package com.primefactorsolutions.service;
+
+import com.primefactorsolutions.service.compiler.InMemoryClass;
+import com.primefactorsolutions.service.compiler.InMemoryFileManager;
+import com.primefactorsolutions.service.compiler.JavaSourceFromString;
+import lombok.SneakyThrows;
+import lombok.extern.slf4j.Slf4j;
+import org.springframework.stereotype.Service;
+
+import javax.tools.DiagnosticCollector;
+import javax.tools.JavaCompiler;
+import javax.tools.JavaFileObject;
+import javax.tools.ToolProvider;
+import java.util.Collections;
+import java.util.List;
+import java.util.Map;
+import java.util.Optional;
+import java.util.stream.Collectors;
+
+@Service
+@Slf4j
+public class CompilerService {
+ private static final Map RESULT_MESSAGE = Map.of(Boolean.TRUE, "OK", Boolean.FALSE, "ERROR");
+
+ @SneakyThrows
+ public Optional doCompile(final String javaCode) {
+ final String qualifiedClassName = "com.primefactorsolutions.TestClass";
+ final JavaCompiler compiler = ToolProvider.getSystemJavaCompiler();
+ final DiagnosticCollector diagnostics = new DiagnosticCollector<>();
+ final InMemoryFileManager manager = new InMemoryFileManager(compiler.getStandardFileManager(null, null, null));
+
+ final List sourceFiles = Collections.singletonList(new JavaSourceFromString(qualifiedClassName, javaCode));
+ final JavaCompiler.CompilationTask task = compiler.getTask(null, manager, diagnostics,
+ List.of("-proc:full", "-Xlint:-options"), null, sourceFiles);
+
+ boolean result = task.call();
+
+ if (!result) {
+ final String errors = diagnostics.getDiagnostics().stream()
+ .map(String::valueOf)
+ .collect(Collectors.joining("\n"));
+
+ return Optional.of(errors);
+ } else {
+ final ClassLoader classLoader = manager.getClassLoader(null);
+ final Class> clazz = classLoader.loadClass(qualifiedClassName);
+ final InMemoryClass instanceOfClass = (InMemoryClass) clazz.newInstance();
+ final Map results = instanceOfClass.runCode();
+
+ return Optional.of(results.entrySet().stream()
+ .map(e -> String.format("%-50s ... %4s", e.getKey(), RESULT_MESSAGE.get(e.getValue())))
+ .collect(Collectors.joining("\n")));
+ }
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/service/UserService.java b/src/main/java/com/primefactorsolutions/service/UserService.java
new file mode 100644
index 0000000..dfb1797
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/UserService.java
@@ -0,0 +1,29 @@
+package com.primefactorsolutions.service;
+
+import com.primefactorsolutions.model.AppUser;
+import com.primefactorsolutions.repositories.UserRepository;
+import lombok.Data;
+import org.springframework.stereotype.Service;
+
+import java.util.List;
+import java.util.UUID;
+
+@Service
+@Data
+public class UserService {
+ private final UserRepository userRepository;
+
+ public AppUser createOrUpdate(final AppUser assessment) {
+ final AppUser saved = userRepository.save(assessment);
+
+ return saved;
+ }
+
+ public List getUsers() {
+ return userRepository.findAll();
+ }
+
+ public AppUser getUser(final UUID id) {
+ return userRepository.findById(id).orElse(null);
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/service/compiler/InMemoryClass.java b/src/main/java/com/primefactorsolutions/service/compiler/InMemoryClass.java
new file mode 100644
index 0000000..801bc40
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/compiler/InMemoryClass.java
@@ -0,0 +1,8 @@
+package com.primefactorsolutions.service.compiler;
+
+import java.util.Map;
+
+public interface InMemoryClass {
+
+ Map runCode();
+}
diff --git a/src/main/java/com/primefactorsolutions/service/compiler/InMemoryClassLoader.java b/src/main/java/com/primefactorsolutions/service/compiler/InMemoryClassLoader.java
new file mode 100644
index 0000000..926bb97
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/compiler/InMemoryClassLoader.java
@@ -0,0 +1,30 @@
+package com.primefactorsolutions.service.compiler;
+
+import static java.util.Objects.requireNonNull;
+
+import java.util.Map;
+
+public class InMemoryClassLoader extends ClassLoader {
+
+ private final InMemoryFileManager manager;
+
+ public InMemoryClassLoader(ClassLoader parent, InMemoryFileManager manager) {
+ super(parent);
+ this.manager = requireNonNull(manager, "manager must not be null");
+ }
+
+ @Override
+ protected Class> findClass(String name) throws ClassNotFoundException {
+
+ Map compiledClasses = manager
+ .getBytesMap();
+
+ if (compiledClasses.containsKey(name)) {
+ byte[] bytes = compiledClasses.get(name)
+ .getBytes();
+ return defineClass(name, bytes, 0, bytes.length);
+ } else {
+ throw new ClassNotFoundException();
+ }
+ }
+}
\ No newline at end of file
diff --git a/src/main/java/com/primefactorsolutions/service/compiler/InMemoryFileManager.java b/src/main/java/com/primefactorsolutions/service/compiler/InMemoryFileManager.java
new file mode 100644
index 0000000..8602a35
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/compiler/InMemoryFileManager.java
@@ -0,0 +1,52 @@
+package com.primefactorsolutions.service.compiler;
+
+import java.util.Hashtable;
+import java.util.Map;
+import javax.tools.FileObject;
+import javax.tools.ForwardingJavaFileManager;
+import javax.tools.JavaFileManager;
+import javax.tools.JavaFileObject;
+import javax.tools.JavaFileObject.Kind;
+import javax.tools.StandardJavaFileManager;
+
+public class InMemoryFileManager extends ForwardingJavaFileManager {
+
+ private final Map compiledClasses;
+ private final ClassLoader loader;
+
+ public InMemoryFileManager(StandardJavaFileManager standardManager) {
+ super(standardManager);
+ this.compiledClasses = new Hashtable<>();
+ this.loader = new InMemoryClassLoader(this.getClass()
+ .getClassLoader(),
+ this
+ );
+ }
+
+ /**
+ * Used to get the class loader for our compiled class. It creates an anonymous class extending
+ * the SecureClassLoader which uses the byte code created by the compiler and stored in the
+ * JavaClassObject, and returns the Class for it
+ *
+ * @param location where to place or search for file objects.
+ */
+ @Override
+ public ClassLoader getClassLoader(Location location) {
+ return loader;
+ }
+
+ @Override
+ public JavaFileObject getJavaFileForOutput(Location location, String className, Kind kind,
+ FileObject sibling) {
+
+ JavaClassAsBytes classAsBytes = new JavaClassAsBytes(
+ className, kind);
+ compiledClasses.put(className, classAsBytes);
+
+ return classAsBytes;
+ }
+
+ public Map getBytesMap() {
+ return compiledClasses;
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/service/compiler/JavaClassAsBytes.java b/src/main/java/com/primefactorsolutions/service/compiler/JavaClassAsBytes.java
new file mode 100644
index 0000000..6a2d2dd
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/compiler/JavaClassAsBytes.java
@@ -0,0 +1,26 @@
+package com.primefactorsolutions.service.compiler;
+
+import javax.tools.SimpleJavaFileObject;
+import java.io.ByteArrayOutputStream;
+import java.io.OutputStream;
+import java.net.URI;
+
+public class JavaClassAsBytes extends SimpleJavaFileObject {
+
+ protected ByteArrayOutputStream bos =
+ new ByteArrayOutputStream();
+
+ public JavaClassAsBytes(String name, Kind kind) {
+ super(URI.create("string:///" + name.replace('.', '/')
+ + kind.extension), kind);
+ }
+
+ public byte[] getBytes() {
+ return bos.toByteArray();
+ }
+
+ @Override
+ public OutputStream openOutputStream() {
+ return bos;
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/service/compiler/JavaSourceFromString.java b/src/main/java/com/primefactorsolutions/service/compiler/JavaSourceFromString.java
new file mode 100644
index 0000000..6fa41ac
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/service/compiler/JavaSourceFromString.java
@@ -0,0 +1,22 @@
+package com.primefactorsolutions.service.compiler;
+
+import javax.tools.SimpleJavaFileObject;
+import java.net.URI;
+
+import static java.util.Objects.requireNonNull;
+
+public class JavaSourceFromString extends SimpleJavaFileObject {
+
+ private String sourceCode;
+
+ public JavaSourceFromString(String name, String sourceCode) {
+ super(URI.create("string:///" + name.replace('.', '/') + Kind.SOURCE.extension),
+ Kind.SOURCE);
+ this.sourceCode = requireNonNull(sourceCode, "sourceCode must not be null");
+ }
+
+ @Override
+ public CharSequence getCharContent(boolean ignoreEncodingErrors) {
+ return sourceCode;
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/AssessmentView.java b/src/main/java/com/primefactorsolutions/views/AssessmentView.java
new file mode 100644
index 0000000..3f927e1
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/AssessmentView.java
@@ -0,0 +1,51 @@
+package com.primefactorsolutions.views;
+
+import com.primefactorsolutions.model.Assessment;
+import com.primefactorsolutions.service.AssessmentService;
+import com.vaadin.flow.component.Component;
+import com.vaadin.flow.component.textfield.TextField;
+import com.vaadin.flow.router.BeforeEvent;
+import com.vaadin.flow.router.HasUrlParameter;
+import com.vaadin.flow.router.PageTitle;
+import com.vaadin.flow.router.Route;
+import com.vaadin.flow.spring.annotation.SpringComponent;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.context.annotation.Scope;
+import org.vaadin.firitin.form.BeanValidationForm;
+
+import java.util.List;
+import java.util.UUID;
+
+@SpringComponent
+@Scope("prototype")
+@PageTitle("Assessments")
+@Route(value = "/assessments", layout = MainLayout.class)
+public class AssessmentView extends BeanValidationForm implements HasUrlParameter {
+ private final AssessmentService assessmentService;
+
+ private final TextField appUserName = new TextField();
+
+ public AssessmentView(AssessmentService assessmentService) {
+ super(Assessment.class);
+ setEntityWithEnabledSave(new Assessment());
+ this.assessmentService = assessmentService;
+
+ setSavedHandler((SavedHandler) assessment -> {
+ System.out.println(assessment);
+ });
+ }
+
+
+ @Override
+ public void setParameter(final BeforeEvent beforeEvent, final String s) {
+ if (StringUtils.isNotBlank(s) && !"new".equals(s)) {
+ var assessment = assessmentService.getAssessment(UUID.fromString(s));
+
+ }
+ }
+
+ @Override
+ protected List getFormComponents() {
+ return List.of(appUserName);
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/AssessmentsListView.java b/src/main/java/com/primefactorsolutions/views/AssessmentsListView.java
new file mode 100644
index 0000000..5856050
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/AssessmentsListView.java
@@ -0,0 +1,61 @@
+package com.primefactorsolutions.views;
+
+import com.primefactorsolutions.model.Assessment;
+import com.primefactorsolutions.service.AssessmentService;
+import com.vaadin.flow.component.html.Main;
+import com.vaadin.flow.data.provider.DataProvider;
+import com.vaadin.flow.data.provider.DataProviderListener;
+import com.vaadin.flow.data.provider.Query;
+import com.vaadin.flow.router.PageTitle;
+import com.vaadin.flow.router.Route;
+import com.vaadin.flow.shared.Registration;
+import com.vaadin.flow.spring.annotation.SpringComponent;
+import org.springframework.context.annotation.Scope;
+import org.vaadin.firitin.components.grid.VGrid;
+
+import java.util.stream.Stream;
+
+@SpringComponent
+@Scope("prototype")
+@PageTitle("Assessments")
+@Route(value = "/assessments", layout = MainLayout.class)
+public class AssessmentsListView extends Main {
+
+ public AssessmentsListView(final AssessmentService assessmentService) {
+ final VGrid grid = new VGrid<>(Assessment.class);
+ grid.setDataProvider(new DataProvider() {
+ @Override
+ public boolean isInMemory() {
+ return false;
+ }
+
+ @Override
+ public int size(Query query) {
+ return 0;
+ }
+
+ @Override
+ public Stream fetch(Query query) {
+ return null;
+ }
+
+ @Override
+ public void refreshItem(Assessment assessment) {
+
+ }
+
+ @Override
+ public void refreshAll() {
+
+ }
+
+ @Override
+ public Registration addDataProviderListener(DataProviderListener dataProviderListener) {
+ return null;
+ }
+ });
+
+ add(grid);
+ }
+
+}
diff --git a/src/main/java/com/primefactorsolutions/views/EvaluationCompletedView.java b/src/main/java/com/primefactorsolutions/views/EvaluationCompletedView.java
new file mode 100644
index 0000000..5e4ee89
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/EvaluationCompletedView.java
@@ -0,0 +1,14 @@
+package com.primefactorsolutions.views;
+
+import com.vaadin.flow.component.Text;
+import com.vaadin.flow.component.html.Main;
+import com.vaadin.flow.router.PageTitle;
+import com.vaadin.flow.router.Route;
+
+@PageTitle("Assessment")
+@Route(value = "/evaluation/completed", layout = MainLayout.class)
+public class EvaluationCompletedView extends Main {
+ public EvaluationCompletedView() {
+ add(new Text("La evaluacion ha sido completada. Nos contactaremos con usted."));
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/EvaluationView.java b/src/main/java/com/primefactorsolutions/views/EvaluationView.java
new file mode 100644
index 0000000..b66ba4f
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/EvaluationView.java
@@ -0,0 +1,414 @@
+package com.primefactorsolutions.views;
+
+import com.flowingcode.vaadin.addons.simpletimer.SimpleTimer;
+import com.hilerio.ace.AceEditor;
+import com.hilerio.ace.AceMode;
+import com.hilerio.ace.AceTheme;
+import com.primefactorsolutions.model.*;
+import com.primefactorsolutions.service.AssessmentService;
+import com.primefactorsolutions.service.CompilerService;
+import com.vaadin.flow.component.*;
+import com.vaadin.flow.component.button.Button;
+import com.vaadin.flow.component.button.ButtonVariant;
+import com.vaadin.flow.component.contextmenu.MenuItem;
+import com.vaadin.flow.component.dialog.Dialog;
+import com.vaadin.flow.component.html.*;
+import com.vaadin.flow.component.html.DescriptionList.Description;
+import com.vaadin.flow.component.html.DescriptionList.Term;
+import com.vaadin.flow.component.icon.Icon;
+import com.vaadin.flow.component.menubar.MenuBar;
+import com.vaadin.flow.component.menubar.MenuBarVariant;
+import com.vaadin.flow.component.notification.Notification;
+import com.vaadin.flow.component.notification.NotificationVariant;
+import com.vaadin.flow.component.orderedlayout.FlexComponent;
+import com.vaadin.flow.component.orderedlayout.HorizontalLayout;
+import com.vaadin.flow.component.orderedlayout.VerticalLayout;
+import com.vaadin.flow.component.textfield.TextField;
+import com.vaadin.flow.router.*;
+import com.vaadin.flow.spring.annotation.SpringComponent;
+import com.vaadin.flow.theme.lumo.LumoUtility.Background;
+import com.vaadin.flow.theme.lumo.LumoUtility.BoxSizing;
+import com.vaadin.flow.theme.lumo.LumoUtility.Display;
+import com.vaadin.flow.theme.lumo.LumoUtility.Flex;
+import com.vaadin.flow.theme.lumo.LumoUtility.FlexDirection;
+import com.vaadin.flow.theme.lumo.LumoUtility.FontSize;
+import com.vaadin.flow.theme.lumo.LumoUtility.FontWeight;
+import com.vaadin.flow.theme.lumo.LumoUtility.Gap;
+import com.vaadin.flow.theme.lumo.LumoUtility.Height;
+import com.vaadin.flow.theme.lumo.LumoUtility.Margin;
+import com.vaadin.flow.theme.lumo.LumoUtility.Overflow;
+import com.vaadin.flow.theme.lumo.LumoUtility.Padding;
+import com.vaadin.flow.theme.lumo.LumoUtility.TextColor;
+import org.jetbrains.annotations.NotNull;
+import org.springframework.context.annotation.Scope;
+
+import java.time.ZoneId;
+import java.time.ZonedDateTime;
+import java.time.format.DateTimeFormatter;
+import java.util.Optional;
+import java.util.UUID;
+import java.util.stream.Collectors;
+
+@PageTitle("Evaluacion")
+@SpringComponent
+@Scope("prototype")
+@Route(value = "/evaluation", layout = MainLayout.class)
+public class EvaluationView extends Main implements HasUrlParameter {
+
+ final CompilerService compilerService;
+ final AssessmentService assessmentService;
+
+ AceEditor questionEditor = null;
+ Dialog dialog = null;
+ Dialog completeDialog = null;
+ AceEditor result = null;
+
+ Assessment assessment = null;
+ Submission currSubmission = null;
+ Boolean isCompleted = false;
+
+ Button start = null;
+ MenuItem prev = null;
+ MenuItem next = null;
+ Section sidebar = null;
+ SimpleTimer timer = null;
+
+ DescriptionList dl = null;
+
+ Section editorSection = null;
+ Section startSection = null;
+
+ Section completedSection = null;
+
+ public EvaluationView(final CompilerService compilerService, final AssessmentService assessmentService) {
+ this.compilerService = compilerService;
+ this.assessmentService = assessmentService;
+
+ addClassNames(Display.FLEX, Flex.GROW, Height.FULL);
+
+ initStartSection();
+ initCompletedSection();
+ initEditorSection();
+ initResultDialog();
+ initCompleteDialog();
+ initSidebar();
+
+ add(completedSection, startSection, editorSection, sidebar, dialog);
+
+ updateUI();
+ }
+
+ private void initResultDialog() {
+ dialog = new Dialog();
+ dialog.setHeaderTitle("Resultados");
+
+ result = new AceEditor();
+ result.setTheme(AceTheme.xcode);
+ result.setMode(AceMode.text);
+ result.setFontSize(12);
+ result.setHeight("100%");
+ result.setWidth("100%");
+ result.setReadOnly(true);
+ result.setMinlines(20);
+ result.setMaxlines(30);
+ result.setDisplayIndentGuides(false);
+ result.setShowGutter(false);
+
+ final VerticalLayout dialogLayout = new VerticalLayout(result);
+ dialogLayout.setPadding(false);
+ dialogLayout.setSpacing(false);
+ dialogLayout.setAlignItems(FlexComponent.Alignment.STRETCH);
+ dialogLayout.getStyle()
+ .set("width", "800px")
+ .set("max-width", "100%");
+
+ dialog.add(dialogLayout);
+
+ final Button saveButton = new Button("Guardar y Siguiente", e -> {
+ this.currSubmission.setResponse(this.questionEditor.getValue());
+ goToNext();
+ dialog.close();
+ });
+ saveButton.addThemeVariants(ButtonVariant.LUMO_PRIMARY);
+ final Button closeButton = new Button("Cerrar", e -> dialog.close());
+
+ dialog.getFooter().add(closeButton, saveButton);
+ }
+
+ private void initCompleteDialog() {
+ completeDialog = new Dialog();
+ completeDialog.setHeaderTitle("Terminar Evaluacion");
+
+ final VerticalLayout dialogLayout = new VerticalLayout(new Text("Desea terminar la evaluacion? No podra editar las respuestas despues de terminar."));
+ dialogLayout.setPadding(false);
+ dialogLayout.setSpacing(false);
+ dialogLayout.setAlignItems(FlexComponent.Alignment.STRETCH);
+ dialogLayout.getStyle()
+ .set("width", "800px")
+ .set("max-width", "100%");
+
+ completeDialog.add(dialogLayout);
+
+ final Button completeButton = new Button("Terminar", e -> {
+ completeDialog.close();
+ this.assessment = assessmentService.completeAssessment(assessment.getId(), currSubmission);
+ goToCompleted();
+ updateUI();
+ });
+ completeButton.addThemeVariants(ButtonVariant.LUMO_PRIMARY, ButtonVariant.LUMO_ERROR);
+ final Button closeButton = new Button("Cerrar", e -> completeDialog.close());
+
+ completeDialog.getFooter().add(closeButton, completeButton);
+ }
+
+ private void initEditorSection() {
+ editorSection = new Section();
+ editorSection.addClassNames(Display.FLEX, FlexDirection.COLUMN, Flex.GROW, Height.FULL);
+ questionEditor = getCodeEditor();
+
+ final MenuBar runMenuBar = new MenuBar();
+ runMenuBar.addThemeVariants(MenuBarVariant.LUMO_PRIMARY);
+ runMenuBar.addItem("Verificar/Ejecutar", (ComponentEventListener>) menuItemClickEvent -> {
+ final String javaCode = questionEditor.getValue();
+ final Optional result = this.compilerService.doCompile(javaCode);
+
+ if (result.isPresent()) {
+ this.result.setValue(result.get());
+ this.dialog.open();
+ }
+ });
+
+ final MenuBar navMenuBar = new MenuBar();
+ prev = navMenuBar.addItem("Anterior pregunta", (ComponentEventListener>) menuItemClickEvent -> {
+ System.out.println(">>> prev");
+ this.currSubmission.setResponse(this.questionEditor.getValue());
+ this.currSubmission = this.assessmentService.getPrevSubmission(assessment, this.currSubmission);
+ updateUI();
+ });
+ next = navMenuBar.addItem("Siguiente pregunta", (ComponentEventListener>) menuItemClickEvent -> {
+ this.currSubmission.setResponse(this.questionEditor.getValue());
+ goToNext();
+ });
+
+ final Div menuBar = new Div();
+ menuBar.add(runMenuBar, navMenuBar);
+
+ setInlineBlock(runMenuBar);
+ setInlineBlock(navMenuBar);
+
+ editorSection.add(menuBar, questionEditor);
+ }
+
+ private void initCompletedSection() {
+ completedSection = new Section();
+ completedSection.addClassNames(Display.FLEX, FlexDirection.COLUMN, Flex.GROW, Height.FULL);
+ completedSection.setVisible(false);
+ Div completedHelp = new Div();
+ completedHelp.add(new Text("Evaluacion ha sido completada. Gracias!"));
+
+ completedSection.add(completedHelp);
+ }
+
+ private void initStartSection() {
+ startSection = new Section();
+ startSection.addClassNames(Display.FLEX, FlexDirection.COLUMN, Flex.GROW, Height.FULL);
+
+ Div startHelp = new Div();
+ startHelp.add(new Text("Bienvenido(a) al examen de evaluacion de PFS. Apriete el boton 'Empezar' para empezar la evaluacion. Tiene 30 minutos para completar."));
+ startHelp.add(new HtmlComponent("br"));
+ startHelp.add(new Text("Las preguntas son de implementacion de codigo en JAVA. Las preguntas deben poder compilar y pasar los tests incluidos en el codigo. Puede pasar un pregunta o volver a una pregunta anterior."));
+
+ TextField tf = new TextField();
+ tf.setLabel("Ingrese su email (donde recibio el link de la evaluacion):");
+
+ start = new Button("Empezar");
+ start.addThemeVariants(ButtonVariant.LUMO_PRIMARY, ButtonVariant.LUMO_SUCCESS);
+ start.addClickListener((ComponentEventListener>) buttonClickEvent -> {
+ System.out.println(">>> start");
+ this.assessment = this.assessmentService.startAssessment(this.assessment.getId());
+
+ if (tf.getValue().trim().equalsIgnoreCase(this.assessment.getAppUser().getEmail())) {
+ this.currSubmission = this.assessmentService.getNextSubmission(assessment, this.currSubmission);
+ updateUI();
+ } else {
+ Notification notification = new Notification();
+ notification.addThemeVariants(NotificationVariant.LUMO_ERROR);
+ notification.setPosition(Notification.Position.TOP_CENTER);
+
+ Div text = new Div(new Text("Email invalido. Verifique que el email corresponde al email donde recibio el enlace a la evaluacion."));
+
+ Button closeButton = new Button(new Icon("lumo", "cross"));
+ closeButton.addThemeVariants(ButtonVariant.LUMO_TERTIARY_INLINE);
+ closeButton.setAriaLabel("Close");
+ closeButton.addClickListener(event -> {
+ notification.close();
+ });
+
+ HorizontalLayout layout = new HorizontalLayout(text, closeButton);
+ layout.setAlignItems(FlexComponent.Alignment.CENTER);
+
+ notification.add(layout);
+ notification.open();
+ }
+ });
+
+ startSection.add(startHelp, tf, start);
+ }
+
+ @NotNull
+ private AceEditor getCodeEditor() {
+ final AceEditor questionEditor;
+ questionEditor = new AceEditor();
+ questionEditor.setTheme(AceTheme.xcode);
+ questionEditor.setMode(AceMode.java);
+ questionEditor.setFontSize(15);
+ questionEditor.setHeight("100%");
+ questionEditor.setWidth("100%");
+
+ questionEditor.setReadOnly(false);
+ questionEditor.setShowInvisibles(true);
+ questionEditor.setShowGutter(false);
+ questionEditor.setShowPrintMargin(false);
+ questionEditor.setDisplayIndentGuides(false);
+ questionEditor.setUseWorker(false);
+
+ questionEditor.setSofttabs(true);
+ questionEditor.setTabSize(4);
+ questionEditor.setWrap(true);
+ questionEditor.setMinlines(20);
+ questionEditor.setMaxlines(30);
+ // aceEditor.setPlaceholder("... Edit ...");
+ questionEditor.setAutoComplete(true);
+ questionEditor.setCustomAutoCompletion(new String[] { "TEST", "TEST2", "TEST3" });
+ questionEditor.setHighlightActiveLine(true);
+ questionEditor.setHighlightSelectedWord(false);
+ return questionEditor;
+ }
+
+ private void setInlineBlock(MenuBar menuBar) {
+ menuBar.getStyle().set("display", "inline-block");
+ }
+
+ private void goToNext() {
+ System.out.println(">>> next");
+ this.currSubmission = this.assessmentService.getNextSubmission(assessment, this.currSubmission);
+ updateUI();
+ }
+
+ private void initSidebar() {
+ sidebar = new Section();
+ sidebar.addClassNames(Background.CONTRAST_5, BoxSizing.BORDER, Display.FLEX, FlexDirection.COLUMN,
+ Flex.SHRINK_NONE, Overflow.AUTO, Padding.LARGE);
+ sidebar.setWidth("256px");
+
+ dl = new DescriptionList();
+ dl.addClassNames(Display.FLEX, FlexDirection.COLUMN, Gap.LARGE, Margin.Bottom.SMALL, Margin.Top.NONE,
+ FontSize.SMALL);
+
+ final Text text = new Text("Tiempo restante:");
+
+ timer = new SimpleTimer(0);
+ timer.setMinutes(true);
+ timer.addTimerEndEvent((ComponentEventListener) timerEndedEvent -> {
+ Notification.show("Tiempo completado.", 5, Notification.Position.TOP_CENTER);
+ this.assessment = assessmentService.completeAssessment(assessment.getId(), currSubmission);
+ goToCompleted();
+ updateUI();
+ });
+ timer.setFractions(false);
+
+ final Button completeButton = new Button("Terminar evaluacion");
+ completeButton.addThemeVariants(ButtonVariant.LUMO_PRIMARY, ButtonVariant.LUMO_ERROR);
+ completeButton.addClickListener((ComponentEventListener>) buttonClickEvent -> {
+ this.completeDialog.open();
+ });
+
+ // Add it all together
+ sidebar.add(dl, text, timer, completeButton);
+ }
+
+ private void updateUI() {
+ if (currSubmission == null) {
+ editorSection.setVisible(false);
+ startSection.setVisible(true);
+ sidebar.setVisible(false);
+ } else {
+ questionEditor.setValue(this.currSubmission.getResponse());
+ editorSection.setVisible(true);
+ startSection.setVisible(false);
+ sidebar.setVisible(true);
+ }
+
+ if (this.assessment != null) {
+ prev.setEnabled(currSubmission != null && !assessment.isFirst(currSubmission));
+ next.setEnabled(currSubmission == null || !assessment.isLast(currSubmission));
+
+ if (this.assessment.isCompleted()) {
+ goToCompleted();
+ this.editorSection.setVisible(false);
+ this.sidebar.setVisible(false);
+ this.startSection.setVisible(false);
+ this.completedSection.setVisible(true);
+ }
+
+ if (dl.getChildren().collect(Collectors.toList()).isEmpty()) {
+ dl.add(
+ createItem("Candidato:", assessment.getAppUser().getEmail()),
+ createItem("Hora comienzo:", Optional.ofNullable(assessment.getStartingTime())
+ .map(t -> ZonedDateTime.ofInstant(t, ZoneId.of("GMT-4")).format(DateTimeFormatter.ISO_LOCAL_DATE_TIME))
+ .orElse("N/A"))
+ );
+ }
+
+ timer.pause();
+ timer.setStartTime(this.assessment.getRemainingTimeSeconds());
+ timer.setMinutes(true);
+ timer.reset();
+ timer.start();
+ }
+ }
+
+ private Div createItem(String label, String value) {
+ return new Div(createTerm(label), createDescription(value));
+ }
+
+ // private Div createBadgeItem(String label, String value) {
+ // return new Div(createTerm(label), createDescription(value, "badge"));
+ // }
+
+ private Term createTerm(String label) {
+ Term term = new Term(label);
+ term.addClassNames(FontWeight.MEDIUM, TextColor.SECONDARY);
+ return term;
+ }
+
+ private Description createDescription(String value, String... themeNames) {
+ Description desc = new Description(value);
+ desc.addClassName(Margin.Left.NONE);
+
+ for (String themeName : themeNames) {
+ desc.getElement().getThemeList().add(themeName);
+ }
+
+ return desc;
+ }
+
+ @Override
+ public void setParameter(BeforeEvent beforeEvent, String s) {
+ this.assessment = this.assessmentService.getAssessment(UUID.fromString(s));
+
+ if (this.assessment.isCompleted()) {
+ goToCompleted();
+ }
+
+ this.currSubmission = this.assessment.getSubmissions().isEmpty()
+ ? null
+ : this.assessmentService.getNextSubmission(assessment, null);
+
+ updateUI();
+ }
+
+ private void goToCompleted() {
+ this.isCompleted = true;
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/MainLayout.java b/src/main/java/com/primefactorsolutions/views/MainLayout.java
index ae3b8a7..abdcfee 100644
--- a/src/main/java/com/primefactorsolutions/views/MainLayout.java
+++ b/src/main/java/com/primefactorsolutions/views/MainLayout.java
@@ -1,6 +1,5 @@
package com.primefactorsolutions.views;
-import com.primefactorsolutions.views.codeeditor.CodeEditorView;
import com.vaadin.flow.component.applayout.AppLayout;
import com.vaadin.flow.component.applayout.DrawerToggle;
import com.vaadin.flow.component.html.Footer;
@@ -9,10 +8,8 @@ import com.vaadin.flow.component.html.Header;
import com.vaadin.flow.component.html.Span;
import com.vaadin.flow.component.orderedlayout.Scroller;
import com.vaadin.flow.component.sidenav.SideNav;
-import com.vaadin.flow.component.sidenav.SideNavItem;
import com.vaadin.flow.router.PageTitle;
import com.vaadin.flow.theme.lumo.LumoUtility;
-import org.vaadin.lineawesome.LineAwesomeIcon;
/**
* The main view is a top-level placeholder for other views.
@@ -50,7 +47,7 @@ public class MainLayout extends AppLayout {
private SideNav createNavigation() {
SideNav nav = new SideNav();
- nav.addItem(new SideNavItem("CodeEditor", CodeEditorView.class, LineAwesomeIcon.EDIT.create()));
+ //nav.addItem(new SideNavItem("CodeEditor", CodeEditorView.class, LineAwesomeIcon.EDIT.create()));
return nav;
}
diff --git a/src/main/java/com/primefactorsolutions/views/MainView.java b/src/main/java/com/primefactorsolutions/views/MainView.java
new file mode 100644
index 0000000..0ca481c
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/MainView.java
@@ -0,0 +1,14 @@
+package com.primefactorsolutions.views;
+
+import com.vaadin.flow.component.Text;
+import com.vaadin.flow.component.html.Main;
+import com.vaadin.flow.router.PageTitle;
+import com.vaadin.flow.router.Route;
+
+@PageTitle("Assessment")
+@Route(value = "", layout = MainLayout.class)
+public class MainView extends Main {
+ public MainView() {
+ add(new Text("welcome"));
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/UserView.java b/src/main/java/com/primefactorsolutions/views/UserView.java
new file mode 100644
index 0000000..e9e49ed
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/UserView.java
@@ -0,0 +1,51 @@
+package com.primefactorsolutions.views;
+
+import com.primefactorsolutions.model.AppUser;
+import com.primefactorsolutions.service.UserService;
+import com.vaadin.flow.component.Component;
+import com.vaadin.flow.component.textfield.TextField;
+import com.vaadin.flow.router.BeforeEvent;
+import com.vaadin.flow.router.HasUrlParameter;
+import com.vaadin.flow.router.PageTitle;
+import com.vaadin.flow.router.Route;
+import com.vaadin.flow.spring.annotation.SpringComponent;
+import org.apache.commons.lang3.StringUtils;
+import org.springframework.context.annotation.Scope;
+import org.vaadin.firitin.form.BeanValidationForm;
+
+import java.util.List;
+import java.util.UUID;
+
+@SpringComponent
+@Scope("prototype")
+@PageTitle("Assessments")
+@Route(value = "/users", layout = MainLayout.class)
+public class UserView extends BeanValidationForm implements HasUrlParameter {
+ private final UserService userService;
+
+ private final TextField appUserName = new TextField();
+
+ public UserView(UserService userService) {
+ super(AppUser.class);
+ setEntityWithEnabledSave(new AppUser("foo@bar.com", List.of()));
+ this.userService = userService;
+
+ setSavedHandler((SavedHandler) appUser -> {
+ System.out.println(appUser);
+ });
+ }
+
+
+ @Override
+ public void setParameter(final BeforeEvent beforeEvent, final String s) {
+ if (StringUtils.isNotBlank(s) && !"new".equals(s)) {
+ var assessment = userService.getUser(UUID.fromString(s));
+
+ }
+ }
+
+ @Override
+ protected List getFormComponents() {
+ return List.of(appUserName);
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/UsersListView.java b/src/main/java/com/primefactorsolutions/views/UsersListView.java
new file mode 100644
index 0000000..1c86f0b
--- /dev/null
+++ b/src/main/java/com/primefactorsolutions/views/UsersListView.java
@@ -0,0 +1,64 @@
+package com.primefactorsolutions.views;
+
+import com.primefactorsolutions.model.AppUser;
+import com.vaadin.flow.component.html.Main;
+import com.vaadin.flow.data.provider.DataProvider;
+import com.vaadin.flow.data.provider.DataProviderListener;
+import com.vaadin.flow.data.provider.Query;
+import com.vaadin.flow.router.PageTitle;
+import com.vaadin.flow.router.Route;
+import com.vaadin.flow.shared.Registration;
+import com.vaadin.flow.spring.annotation.SpringComponent;
+import org.springframework.context.annotation.Scope;
+import org.vaadin.firitin.components.grid.VGrid;
+
+import java.util.List;
+import java.util.stream.Stream;
+
+@SpringComponent
+@Scope("prototype")
+@PageTitle("Users")
+@Route(value = "/users", layout = MainLayout.class)
+public class UsersListView extends Main {
+
+ public UsersListView() {
+ final VGrid grid = new VGrid<>(AppUser.class);
+
+ grid.setDataProvider(new DataProvider() {
+ @Override
+ public boolean isInMemory() {
+ return false;
+ }
+
+ @Override
+ public int size(Query query) {
+ return 1;
+ }
+
+ @Override
+ public Stream fetch(Query query) {
+ int limit = query.getLimit();
+ int pagerSize = query.getPageSize();
+ int page = query.getPage();
+ return Stream.of(new AppUser("foo@bar.com", List.of()));
+ }
+
+ @Override
+ public void refreshItem(AppUser appUser) {
+
+ }
+
+ @Override
+ public void refreshAll() {
+
+ }
+
+ @Override
+ public Registration addDataProviderListener(DataProviderListener dataProviderListener) {
+ return null;
+ }
+ });
+
+ add(grid);
+ }
+}
diff --git a/src/main/java/com/primefactorsolutions/views/codeeditor/CodeEditorView.java b/src/main/java/com/primefactorsolutions/views/codeeditor/CodeEditorView.java
deleted file mode 100644
index 73eeac1..0000000
--- a/src/main/java/com/primefactorsolutions/views/codeeditor/CodeEditorView.java
+++ /dev/null
@@ -1,97 +0,0 @@
-package com.primefactorsolutions.views.codeeditor;
-
-import com.primefactorsolutions.views.MainLayout;
-import com.vaadin.flow.component.html.*;
-import com.vaadin.flow.component.html.DescriptionList.Description;
-import com.vaadin.flow.component.html.DescriptionList.Term;
-import com.vaadin.flow.component.richtexteditor.RichTextEditor;
-import com.vaadin.flow.component.richtexteditor.RichTextEditorVariant;
-import com.vaadin.flow.component.select.Select;
-import com.vaadin.flow.router.PageTitle;
-import com.vaadin.flow.router.Route;
-import com.vaadin.flow.router.RouteAlias;
-import com.vaadin.flow.theme.lumo.LumoUtility.Accessibility;
-import com.vaadin.flow.theme.lumo.LumoUtility.Background;
-import com.vaadin.flow.theme.lumo.LumoUtility.Border;
-import com.vaadin.flow.theme.lumo.LumoUtility.BorderColor;
-import com.vaadin.flow.theme.lumo.LumoUtility.BoxSizing;
-import com.vaadin.flow.theme.lumo.LumoUtility.Display;
-import com.vaadin.flow.theme.lumo.LumoUtility.Flex;
-import com.vaadin.flow.theme.lumo.LumoUtility.FlexDirection;
-import com.vaadin.flow.theme.lumo.LumoUtility.FontSize;
-import com.vaadin.flow.theme.lumo.LumoUtility.FontWeight;
-import com.vaadin.flow.theme.lumo.LumoUtility.Gap;
-import com.vaadin.flow.theme.lumo.LumoUtility.Height;
-import com.vaadin.flow.theme.lumo.LumoUtility.Margin;
-import com.vaadin.flow.theme.lumo.LumoUtility.Overflow;
-import com.vaadin.flow.theme.lumo.LumoUtility.Padding;
-import com.vaadin.flow.theme.lumo.LumoUtility.TextColor;
-
-@PageTitle("CodeEditor")
-@Route(value = "", layout = MainLayout.class)
-@RouteAlias(value = "", layout = MainLayout.class)
-public class CodeEditorView extends Main {
-
- public CodeEditorView() {
- addClassNames(Display.FLEX, Flex.GROW, Height.FULL);
-
- // Editor
- RichTextEditor editor = new RichTextEditor();
- editor.addClassNames(Border.RIGHT, BorderColor.CONTRAST_10, Flex.GROW);
- editor.addThemeVariants(RichTextEditorVariant.LUMO_NO_BORDER);
- editor.asDelta().setValue(
- "[{\"insert\":\"High quality rich text editor for the web\"},{\"attributes\":{\"header\":2},\"insert\":\"\\n\"},{\"insert\":\" is a Web Component providing rich text editor functionality, part of the \"},{\"attributes\":{\"link\":\"https://vaadin.com/components\"},\"insert\":\"Vaadin components\"},{\"insert\":\".\\nIt handles the following formatting:\\n\"},{\"attributes\":{\"bold\":true},\"insert\":\"Bold\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"attributes\":{\"italic\":true},\"insert\":\"Italic\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"attributes\":{\"underline\":true},\"insert\":\"Underline\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"attributes\":{\"strike\":true},\"insert\":\"Strike-through\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"insert\":\"Headings (H1, H2, H3)\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"insert\":\"Lists (ordered and unordered)\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"insert\":\"Text align (left, center, right)\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"attributes\":{\"script\":\"sub\"},\"insert\":\"Sub\"},{\"insert\":\"script and \"},{\"attributes\":{\"script\":\"super\"},\"insert\":\"super\"},{\"insert\":\"script\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"attributes\":{\"link\":\"https://vaadin.com\"},\"insert\":\"Hyperlink\"},{\"attributes\":{\"list\":\"bullet\"},\"insert\":\"\\n\"},{\"insert\":\"In addition to text formatting, additional content blocks can be added.\\nImages\"},{\"attributes\":{\"header\":3},\"insert\":\"\\n\"},{\"insert\":{\"image\":\"\"}},{\"insert\":\"\\nBlockquotes\"},{\"attributes\":{\"header\":3},\"insert\":\"\\n\"},{\"insert\":\"Lorem ipsum dolor sit amet, consectetur adipisicing elit, sed do eiusmod tempor incididunt ut labore et dolore magna aliqua.\"},{\"attributes\":{\"blockquote\":true},\"insert\":\"\\n\"},{\"insert\":\"Code blocks\"},{\"attributes\":{\"header\":3},\"insert\":\"\\n\"},{\"insert\":\"\"},{\"attributes\":{\"code-block\":true},\"insert\":\"\\n\"},{\"insert\":\" \"},{\"attributes\":{\"code-block\":true},\"insert\":\"\\n\"},{\"insert\":\"\"},{\"attributes\":{\"code-block\":true},\"insert\":\"\\n\"},{\"insert\":\"\\n\"}]");
-
- // Sidebar
- add(editor, createSidebar());
- }
-
- private Section createSidebar() {
- Section sidebar = new Section();
- sidebar.addClassNames(Background.CONTRAST_5, BoxSizing.BORDER, Display.FLEX, FlexDirection.COLUMN,
- Flex.SHRINK_NONE, Overflow.AUTO, Padding.LARGE);
- sidebar.setWidth("256px");
-
- H2 title = new H2("Project details");
- title.addClassName(Accessibility.SCREEN_READER_ONLY);
-
- DescriptionList dl = new DescriptionList();
- dl.addClassNames(Display.FLEX, FlexDirection.COLUMN, Gap.LARGE, Margin.Bottom.SMALL, Margin.Top.NONE,
- FontSize.SMALL);
-
- dl.add(createItem("Owner", "My Name"), createItem("Created", "2021-08-14 14:48"),
- createItem("Last modified", "2021-08-14 14:50"), createBadgeItem("Status", "Draft"));
-
- Select select = new Select();
- select.setLabel("Project");
- select.setItems("My Project", "Your Project", "Their Project");
- select.setValue("My Project");
-
- // Add it all together
- sidebar.add(title, dl, select);
- return sidebar;
- }
-
- private Div createItem(String label, String value) {
- return new Div(createTerm(label), createDescription(value));
- }
-
- private Div createBadgeItem(String label, String value) {
- return new Div(createTerm(label), createDescription(value, "badge"));
- }
-
- private Term createTerm(String label) {
- Term term = new Term(label);
- term.addClassNames(FontWeight.MEDIUM, TextColor.SECONDARY);
- return term;
- }
-
- private Description createDescription(String value, String... themeNames) {
- Description desc = new Description(value);
- desc.addClassName(Margin.Left.NONE);
- for (String themeName : themeNames) {
- desc.getElement().getThemeList().add(themeName);
- }
- return desc;
- }
-}
diff --git a/src/main/resources/application.properties b/src/main/resources/application.properties
index ed37e71..1f266a2 100644
--- a/src/main/resources/application.properties
+++ b/src/main/resources/application.properties
@@ -6,5 +6,13 @@ spring.mustache.check-template-location = false
vaadin.launch-browser=true
# To improve the performance during development.
# For more information https://vaadin.com/docs/latest/integrations/spring/configuration#special-configuration-parameters
-vaadin.allowed-packages = com.vaadin,org.vaadin,com.primefactorsolutions
+vaadin.allowed-packages = com.vaadin,org.vaadin,com.primefactorsolutions,com.hilerio.ace,com.flowingcode.vaadin
+
+
+spring.datasource.url=jdbc:h2:mem:testdb
+spring.datasource.driverClassName=org.h2.Driver
+spring.datasource.username=sa
+spring.datasource.password=sa
+spring.jpa.database-platform=org.hibernate.dialect.H2Dialect
spring.jpa.defer-datasource-initialization = true
+spring.h2.console.enabled=true
diff --git a/src/main/resources/data.sql b/src/main/resources/data.sql
new file mode 100644
index 0000000..41ddc80
--- /dev/null
+++ b/src/main/resources/data.sql
@@ -0,0 +1,10 @@
+insert into app_user (id, version, email) values ('23471ab3-f639-4d2b-9541-7227f4ea7ee6', 1, 'foo@bar.com');
+
+insert into question (id, version, content) values ('a7e00ff8-da41-4624-b31c-1b13c3f2e3ae', 1, 'foo bar');
+insert into question (id, version, content) values ('8a4b213c-ca81-4c38-b56d-d7028c2dde88', 1, 'foo buzz');
+
+insert into assessment(id, version, app_user_id) values ('46b153f4-23fd-462f-8430-fbe67b83caab', 1, '23471ab3-f639-4d2b-9541-7227f4ea7ee6');
+
+insert into ASSESSMENT_QUESTIONS (assessment_id, questions_id) values ('46b153f4-23fd-462f-8430-fbe67b83caab', 'a7e00ff8-da41-4624-b31c-1b13c3f2e3ae');
+
+insert into ASSESSMENT_QUESTIONS (assessment_id, questions_id) values ('46b153f4-23fd-462f-8430-fbe67b83caab', '8a4b213c-ca81-4c38-b56d-d7028c2dde88');
\ No newline at end of file
diff --git a/src/main/resources/questions/fibonaci.java.txt b/src/main/resources/questions/fibonaci.java.txt
new file mode 100644
index 0000000..037876c
--- /dev/null
+++ b/src/main/resources/questions/fibonaci.java.txt
@@ -0,0 +1,55 @@
+package com.primefactorsolutions;
+
+import com.primefactorsolutions.service.compiler.InMemoryClass;
+import java.util.*;
+import java.lang.reflect.*;
+import java.util.stream.Collectors;
+
+public class TestClass implements InMemoryClass {
+
+ // ------------ IMPLEMENTAR AQUI --------------------------------
+
+ /**
+ *
+ */
+ public static int getFibonaci(int n) {
+ // TODO: implementar aca
+ throw new RuntimeException("not implemented yet");
+ }
+
+
+ // ------------- NO MODIFICAR DESDE ESTA LINEA --------------------
+
+ public static class Tests {
+ public static Map.Entry testFibonaciSingle() {
+ boolean result = getFibonaci(1) == 0;
+
+ return Map.entry("fibonaci primer", result);
+ }
+
+ public static Map.Entry testFibonaciInvalido() {
+ try {
+ getFibonaci(-1);
+ return Map.entry("palindrome vacio es valido", false);
+ } catch (IllegalArgumentException e) {
+ return Map.entry("palindrome vacio es valido", true);
+ }
+ }
+ }
+
+ @Override
+ public Map runCode() {
+ Map results = Arrays.stream(Tests.class.getDeclaredMethods())
+ .map(m -> {
+ try {
+ return (Map.Entry) m.invoke(null);
+ } catch (IllegalAccessException e) {
+ throw new RuntimeException(e);
+ } catch (InvocationTargetException e) {
+ throw new RuntimeException(e);
+ }
+ }).collect(Collectors.toMap(e -> e.getKey(), e -> e.getValue()));
+
+ return results;
+ }
+}
\ No newline at end of file
diff --git a/src/main/resources/questions/palindroma.java.txt b/src/main/resources/questions/palindroma.java.txt
new file mode 100644
index 0000000..b1462d1
--- /dev/null
+++ b/src/main/resources/questions/palindroma.java.txt
@@ -0,0 +1,60 @@
+package com.primefactorsolutions;
+
+import com.primefactorsolutions.service.compiler.InMemoryClass;
+import java.util.*;
+import java.lang.reflect.*;
+import java.util.stream.Collectors;
+
+public class TestClass implements InMemoryClass {
+
+ // ------------ IMPLEMENTAR AQUI --------------------------------
+
+ public static String getPalindroma(String s) {
+ return "TODO";
+ }
+
+
+ // ------------- NO MODIFICAR DESDE ESTA LINEA --------------------
+
+ public static class Tests {
+ public static Map.Entry testPalindromaValido() {
+ boolean result = getPalindroma("Ab").equals("ABBA");
+
+ return Map.entry("palindrome es valido", result);
+ }
+
+ public static Map.Entry testPalindromaVacio() {
+ boolean result = getPalindroma("").equals("");
+
+ return Map.entry("palindrome vacio es valido", result);
+ }
+
+ public static Map.Entry testPalindromaUnico() {
+ boolean result = getPalindroma("z").equals("ZZ");
+
+ return Map.entry("palindrome una solo caracter es valido", result);
+ }
+
+ public static Map.Entry testPalindromaNull() {
+ boolean result = getPalindroma(null) == null;
+
+ return Map.entry("palindrome null es valido", result);
+ }
+ }
+
+ @Override
+ public Map runCode() {
+ Map results = Arrays.stream(Tests.class.getDeclaredMethods())
+ .map(m -> {
+ try {
+ return (Map.Entry) m.invoke(null);
+ } catch (IllegalAccessException e) {
+ throw new RuntimeException(e);
+ } catch (InvocationTargetException e) {
+ throw new RuntimeException(e);
+ }
+ }).collect(Collectors.toMap(e -> e.getKey(), e -> e.getValue()));
+
+ return results;
+ }
+}
\ No newline at end of file