Commit 0627b685 authored by Bruno Burke's avatar Bruno Burke 🍔
Browse files

code-execution is working serverside and clientside

parent 1765e70e
Pipeline #2397 passed with stages
in 2 minutes and 27 seconds
/*.log
/*-init.clj
/resources/public/js
/resources/public/css
/resources/public/node_modules
out
pom.xml
pom.xml.asc
......
(defproject lernmeister.components "0.1.9"
(defproject lernmeister.components "0.1.11"
:dependencies [[org.clojure/clojure "1.9.0-beta2"]
[org.clojure/clojurescript "1.9.946"]
[reagent "0.7.0"]
......@@ -6,6 +6,7 @@
[hickory "0.7.1"]
[cljsjs/katex "0.7.1-0"]
[clj-http "3.7.0"]
[cljs-ajax "0.7.3"]
[cheshire "5.8.0"] ;;; clj-http json support
[devcards "0.2.4" :exclusions [cljsjs/react]]]
......@@ -14,7 +15,12 @@
:source-paths ["src/cljc" "src/cljs" "src/clj"]
:test-paths ["test/clj" "test/cljc"]
:plugins [[lein-cljsbuild "1.1.7"]
[lein-garden "0.3.0"]]
[lein-garden "0.3.0"]
[lein-npm "0.6.2"]]
:npm {:root "resources/public"
:dependencies [[codemirror "^5.29.0"]
[materialize-css "^0.99.0"]
[quill "^1.3.2"]]}
:clean-targets ^{:protect false} ["resources/public/js"
"target"
......
......@@ -6,6 +6,10 @@
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/materialize/0.99.0/css/materialize.min.css">
<script type="text/javascript" src="https://code.jquery.com/jquery-3.2.1.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/materialize/0.99.0/js/materialize.min.js"></script>
<script src="node_modules/codemirror/lib/codemirror.js"></script>
<script src="node_modules/codemirror/mode/python/python.js"></script>
<link rel="stylesheet" href="node_modules/codemirror/lib/codemirror.css">
<style id="com-rigsomelight-devcards-addons-css"></style>
<meta name="viewport" content="width=device-width, initial-scale=1.0"/>
......
......@@ -2,4 +2,4 @@
(:require [garden.def :refer [defstyles]]))
(defstyles screen
[:div {:color "green"}])
)
(ns lernmeister.components.exercise-types.programming.code-check
(:require [clj-http.client :as client]))
(:require [clj-http.client :as client]
[lernmeister.components.options :refer [get-option]]))
(def code-runner (str "http://fb02tiitvm05:3001/compile"))
(def code-runner-api-key "...PyCodeExecution__3}dx")
(defn get-code-runner-url []
(str (:code-runner-url (get-option :programming-exercise))))
(defn get-code-runner-api-key []
(:code-runner-api-key (get-option :programming-exercise)))
(defn run-code [code language]
(println "Send code: " code)
(:body (client/post code-runner {:form-params {:code code :language language}
:headers {"X-API-Key" code-runner-api-key}
(:body (client/post (get-code-runner-url) {:form-params {:code code :language language}
:headers {"X-API-Key" (get-code-runner-api-key)}
:coerce :always
:as :json
:content-type :json})))
......
(ns lernmeister.components.options)
(defonce options (atom {}))
(defn add-option! [key value]
(println "Add Option " key)
(swap! options assoc key value))
(defn get-option [key]
(get @options key))
......@@ -18,6 +18,11 @@
[:span.light-green-text "✔ "])
)
(defn neutral-tick []
(fn []
[:span.blue-text "◌ "])
)
(defn wrong-tick []
(fn []
[:span.red-text "✘ "])
......@@ -60,7 +65,15 @@
)
)
(defn codebox [code]
[:code {:style {:border "1px solid black"
:background-color "#EEEEFF"
:white-space :pre-line
:padding "10px"
:font-size "0.9em"
:display :inline-block}}
(str code)
])
(ns lernmeister.components.exercise-types.programming.coderunner
(:require [lernmeister.components.options :refer [add-option! get-option]]))
(defn run-code [code handler & {:keys [code-runner-url timeoutMs]}]
(let [code-runner (:code-runner-fn (get-option :programming-exercise))]
(code-runner :params {:code code
:timeoutMs (or timeoutMs 750)}
:handler handler
:code-runner-url code-runner-url)))
......@@ -3,6 +3,7 @@
[lernmeister.components.content-elements.core :as content-manager]
[lernmeister.components.common :as common]
[lernmeister.components.content-elements.exercise :as exercises]
[lernmeister.components.exercise-types.programming.coderunner :refer [run-code]]
[clojure.string]))
(defn is-correct [result]
......@@ -49,72 +50,96 @@
))
))
(defn exercise-renderer [exercise & {:keys [result answer on-change]}]
(defn exercise-renderer [exercise & {:keys [result answer on-change status]}]
(let [eid (:id exercise)
cm-editor-id (str (gensym) "-" eid)
cm-editor (atom nil)
code-result (reagent/atom nil) ;(re-frame/subscribe [:code-result eid])
code-runner-pending (reagent/atom nil); (re-frame/subscribe [:code-runner-pending eid])
code-testrunner-pending (reagent/atom nil); (re-frame/subscribe [:code-testrunner-pending eid])
test-results (reagent/atom nil); (re-frame/subscribe [:code-test-results eid])
]
status (reagent/atom {})
run #(do
(swap! status assoc :code-runner-pending true)
(run-code (.getValue @cm-editor)
(fn [result]
(swap! status dissoc :code-runner-pending)
(swap! status assoc :code-result result))
:code-runner-url (:code-runner-url exercise)
))]
(reagent/create-class
{
:component-name "programming-exercise"
:component-did-mount
(fn []
(reset! cm-editor
(.fromTextArea js/CodeMirror (js/document.getElementById cm-editor-id)
(clj->js
{:mode "python"
:lineNumbers true})))
(.on @cm-editor "change" #(on-change (.getValue @cm-editor)))
(.setValue @cm-editor (or answer (:code-template exercise))))
(when-not result
(reset! cm-editor
(.fromTextArea js/CodeMirror (js/document.getElementById cm-editor-id)
(clj->js
{:mode "python"
:lineNumbers true})))
(.on @cm-editor "change" #(on-change (.getValue @cm-editor)))
(.setValue @cm-editor (or answer (:code-template exercise)))))
:reagent-render
(fn [exercise & {:keys [result answer on-change]}]
[:div.card
[:div.card-content
[:span.card-title [:b (:title exercise)]]
(doall (for [ce (:task-description exercise)]
^{:key (:id ce)}[(content-manager/get-renderer (:type ce)) ce]
))
[:textarea {:id cm-editor-id}]
(when @code-result
[:div
[:h2 "Auswertung"]
(if @code-runner-pending
[:div.progress.blue.lighten-5
[:div.indeterminate.blue]]
[:div.card.card-outline-success.mb-3.text-center
[:div.card-block
[:blockquote.card-blockquote
[:p @code-result]
]]
])
]
)
(when @test-results
[:div
[:h2 "Testresults"]
(if (pos? (count @code-testrunner-pending))
[:div.progress.blue.lighten-5
[:div.indeterminate.blue]]
(for [tr @test-results]
^{:key (str eid (first tr))}
[test-result (second tr) (first tr) (get-in exercise [:unit-tests (first tr)])]))
])]
[:div.card-action
(if @code-runner-pending
[:button.btn.green {:type "button" :disabled "disabled"} "Run"]
[:button.btn.light-green {:on-click identity #_(re-frame/dispatch [:run-code (.getValue @cm-editor) exercise])} "Run"]
)
" "
(if (pos? (count @code-testrunner-pending))
[:button.btn.orange {:type "button" :disabled "disabled"} "Test / Submit"]
[:button.btn.orange {:on-click identity #_(re-frame/dispatch [:test-code (.getValue @cm-editor) exercise])} "Test / Submit"]
)]
]
)})))
(let [test-results (get result :test-results)
coderesult (:code-result @status)
code-runner-pending (:code-runner-pending @status)
code-testrunner-pending (:code-testrunner-pending @status)]
[:div.card {:class
(cond
(and (:points result) (:points-max result)
(= (:points result) (:points-max result))) "green lighten-4"
(and (:points result) (:points-max result)
(< 0 (:points result) (:points-max result))) "yellow lighten-4"
(and (:points result) (<= (:points result) 0)) "red lighten-4"
:else "")}
[:div.card-content
[:span.card-title [:b (:title exercise)]]
(doall (for [ce (:task-description exercise)]
^{:key (:id ce)}[(content-manager/get-renderer (:type ce)) ce]))
(if-not result
[:textarea {:id cm-editor-id}]
[:div [:p "Deine Lösung:"]
[common/codebox answer]])
[:br]
(when coderesult
[:div
(if code-runner-pending
[:div.progress.blue.lighten-5
[:div.indeterminate.blue]]
(if (:isError coderesult)
[common/codebox (:stderr coderesult)]
[common/codebox (:stdout coderesult)]
))])
(when result
[:div.right.secondary-content
(let [passed (:passed result)
failed (:failed result)
tests (+ passed failed)]
(str passed "/" tests " erfolgreiche Unit-Tests "))
(cond
(and (:points result) (:points-max result)
(= (:points result) (:points-max result))) [common/correct-tick]
(and (:points result) (:points-max result)
(< 0 (:points result) (:points-max result))) [common/neutral-tick]
(and (:points result) (<= (:points result) 0)) [common/wrong-tick]
:else "")
(if (pos? (count code-testrunner-pending))
[:div.progress.blue.lighten-5
[:div.indeterminate.blue]]
(for [tr test-results]
^{:key (str eid (first tr))}
[test-result (second tr) (first tr) (get-in exercise [:unit-tests (first tr)])]))])]
(when-not result
[:div.card-action
(if code-runner-pending
[:button.btn.green {:type "button" :disabled "disabled"} "Run"]
[:button.btn.light-green {:on-click #(run)} "Run"]
)
" "
#_(if (pos? (count code-testrunner-pending))
[:button.btn.orange {:type "button" :disabled "disabled"} "Test / Submit"]
[:button.btn.orange {:on-click identity #_(re-frame/dispatch [:test-code (.getValue @cm-editor) exercise])} "Test / Submit"]
)])
])
)})))
......
......@@ -4,6 +4,7 @@
(:require
[lernmeister.components.first-card]
[devcards.lernmeister.components.tex-test-card]
[devcards.lernmeister.components.exercise-programming]
[devcards.lernmeister.components.exercise-editor-card]
[devcards.lernmeister.components.exercise-expression]
[devcards.lernmeister.components.exercise-sc-card]))
......
(ns devcards.lernmeister.components.exercise-programming
(:require-macros
[devcards.core :refer [defcard-doc
defcard-rg
defcard
mkdn-pprint-source]])
(:require
[lernmeister.components.sample-data :as data]
[lernmeister.components.core]
[lernmeister.components.options :refer [add-option!]]
[lernmeister.components.exercise-types.programming.views.show :as ex-show]
[lernmeister.components.content-elements.exercise :as exercise]
[lernmeister.components.exercise-types.core :refer [get-exercise get-types]]
[lernmeister.components.content-elements.core :as content-manager]
[lernmeister.components.material-design :as md]
[ajax.core :refer [GET POST]]
[clojure.pprint :refer [pprint]]
[devcards.core]
[reagent.core :as reagent]))
(add-option!
:programming-exercise {:code-runner-fn
(fn [& {:keys [params handler]}]
(POST "http://localhost:3005/compile"
{:handler handler
:params params
:format :json
:response-format :json
:keywords? true
:headers {:X-API-KEY "...PyCodeExecution__3}dx"}}))})
(defonce exercise-data data/programming-exercise)
(def answer (reagent/atom nil))
(defcard "# Programming Exercise-Type")
(defcard @exercise-data)
(defcard-rg prog-exercise
(fn []
[:div
[ex-show/exercise-renderer @exercise-data
;;{:points 3 :points-max 6 :correct true}
:answer @answer
:on-change #(reset! answer %)]]))
(defcard-rg prog-exercise-answered
(fn []
[:div
[ex-show/exercise-renderer @exercise-data
;;{:points 3 :points-max 6 :correct true}
:answer (:code-template @exercise-data)
:result {:failed 2 :passed 7 :points 2 :points-max 5}
:on-change #(reset! answer %)]]))
......@@ -112,3 +112,50 @@
"Eine Schnecke kriecht mit konstanter Geschwindigkeit einen Weinberg hinauf. Für die 150 m lange Strecke vom Sockel bis zum 90 m hohen Gipfel benötigt die Schnecke 270 s. Eine Steilkante von 75 m Höhe schließt sich direkt an den Gipfel an. Unten geht es noch 300 m abfällig weiter, bis das Höhenniveau des Bergsockels wieder erreicht ist. Die Schnecke beschließt, es ruhiger angehen zu lassen und schneckt die letzten 300 m mit 0,1 m/s weiter.\n1.\tWie lange war die Schnecke insgesamt unterwegs?",
:type "text"}],
:text ""}))
(def programming-exercise
(reagent/atom {:code-template "def manhattan(a, b):\n return 0",
:shuffled false,
:type :programming,
:unit-tests
[{:id "ut2017-06-27T09-38-52-G__919",
:code "",
:title "Reihenfolge der Parameter ist irrelevant",
:points 0,
:testcode
"import random\n\nn = 4\nv1 = random.sample(range(-100, 100), n)\nv2 = random.sample(range(-100, 100), n)\n\nprint(manhattan(v1,v2) == manhattan(v2,v1))",
:correct-points 1,
:incorrect-points 0}
{:id "ut2017-06-27T09-43-01-G__930",
:code "",
:title "Absolutwert wird berechnet",
:points 0,
:testcode
"print(manhattan([-2], [-4]) == manhattan([-4], [-2]) == 2)",
:correct-points 1,
:incorrect-points 0}
{:id "ut2017-06-27T09-45-56-G__933",
:code "",
:title "Korrekter Wert wird berechnet",
:points 0,
:testcode
"import random\n\ndef manhattan_distance(start, end):\n return sum(abs(e - s) for s,e in zip(start, end))\n\ncorrect = True\n \nfor i in range(0,100):\n n = random.randint(1,6)\n v1 = random.sample(range(-100, 100), n)\n v2 = random.sample(range(-100, 100), n)\n correct = correct and (manhattan_distance(v1,v2) == manhattan(v1,v2))\n \nprint(correct)",
:correct-points 1,
:incorrect-points 0}],
:title "Manhattan-Distanz",
:language :python3,
:id "ex2017-06-27T09-28-44-G__897",
:answers [],
:voting false,
:task-description
[{:id "ce2017-06-27T09-36-19-G__898",
:value
"Programmieren Sie eine Funktion mit dem Namen manhattan, welche die Manhattan-Distanz zwischen den Parametern a und b berechnet. \nBei a und b handelt es sich um beliebig-dimensionale Vektoren.",
:type "text"}
{:id "ce2017-06-29T13-43-34-G__42",
:value "",
:type "image",
:url
""}],
:text ""}))
Supports Markdown
0% or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment