From ba4bee9a9a508eb1a686808241559b331f2d33ea Mon Sep 17 00:00:00 2001
From: Martin <martin@leo.fri1.uni-lj.si>
Date: Tue, 22 Sep 2015 17:28:25 +0200
Subject: Added Greatest Negative problem.

---
 .../problems/functions/greatest_negative/common.py | 114 +++++++++++++++++++++
 python/problems/functions/greatest_negative/en.py  |  16 +++
 python/problems/functions/greatest_negative/sl.py  |  97 ++++++++++++++++++
 3 files changed, 227 insertions(+)
 create mode 100644 python/problems/functions/greatest_negative/common.py
 create mode 100644 python/problems/functions/greatest_negative/en.py
 create mode 100644 python/problems/functions/greatest_negative/sl.py

(limited to 'python/problems/functions')

diff --git a/python/problems/functions/greatest_negative/common.py b/python/problems/functions/greatest_negative/common.py
new file mode 100644
index 0000000..01dfce3
--- /dev/null
+++ b/python/problems/functions/greatest_negative/common.py
@@ -0,0 +1,114 @@
+# coding=utf-8
+
+import re
+from python.util import has_token_sequence, string_almost_equal, \
+    string_contains_number, get_tokens, get_numbers, get_exception_desc
+from server.hints import Hint, HintSequence
+
+id = 197
+group = 'functions'
+number = 3
+visible = True
+
+solution = '''\
+def max_neg(xs):
+    mn = None
+    for x in xs:
+        if x < 0 and (mn == None or x > mn):
+            mn = x
+    return mn
+'''
+
+hint_type = {
+    'no_def': Hint('no_def'),
+    'no_return': Hint('no_return'),
+    'return_indent': Hint('return_indent'),
+    'for_loop': Hint('for_loop'),
+    'not_int': Hint('not_int'),
+    'return_first': Hint('return_first'),
+    'return_last': Hint('return_last'),
+    'return_greatest': Hint('return_greatest'),
+    'return_positive': Hint('return_positive'),
+    'return_absolute': Hint('return_absolute'),
+}
+
+
+def test(python, code):
+    test_lists = [[6, 4, 2, 0],
+               [4, 6, 2, -1],
+               [4, -2, -6, 0],
+               [4, 2, 0, 6],
+               [6, -8, 2, 0],
+               [-8, 6, 2, 0],
+               [8, -6, -2, 0],
+               [-5, -6, -2, -1],
+               [-8, 0, -6, -2],
+               [-1, -8, -6, -2],
+               [42],
+               [-42]]
+
+    test_in = [('max_neg(%s)'%str(l), None) for l in test_lists]
+    test_out = [None, -1, -2, None, -8, -8, -2, -1, -2, -1, None, -42]
+
+    answers = python(code=code, inputs=test_in, timeout=1.0)
+    n_correct = 0
+    tin, tout = None, None
+    for i, (ans, to) in enumerate(zip(answers, test_out)):
+        n_correct += ans[0] == to
+        if ans[0] != to:
+            tin = test_lists[i]
+            tout = to
+    hints = [{'id': 'test_results', 'args': {'passed': n_correct, 'total': len(test_in)}}]
+    if tin:
+        hints.append({'id': 'problematic_test_case', 'args': {'testin': str(tin), 'testout': str(tout)}})
+
+    return n_correct == len(test_in), hints
+
+def hint(python, code):
+    tokens = get_tokens(code)
+    # having function max_neg is necessary!
+    if not has_token_sequence(tokens, ['max_neg']):
+        return [{'id' : 'no_func_name', 'args' : {'func_name' : 'max_neg'}}]
+
+    # run one test to see if there are any exceptions
+    answer = python(code=code, inputs=[('max_neg([1, 3, -3, -6, 4, 3, 2])', None)], timeout=1.0)
+    exc = get_exception_desc(answer[0][3])
+    if exc: return exc
+
+    # if has no def, tell him to define the function
+    if not has_token_sequence(tokens, ['def']):
+        return [{'id' : 'no_def'}]
+
+    # if has no return, tell him to return the value
+    if not has_token_sequence(tokens, ['return']):
+        return [{'id' : 'no_return'}]
+
+    # if has no loop, tell him to use it
+    if not has_token_sequence(tokens, ['while']) and \
+       not has_token_sequence(tokens, ['for']):
+        return [{'id' : 'for_loop'}]
+
+    # if has no condition if, tell him to use it
+    if not has_token_sequence(tokens, ['if']):
+        return [{'id' : 'if_clause'}]
+
+    # test for wrong answers
+    res = answer[0][0]
+    if not isinstance(res, int):
+        return [{'id' : 'not_int'}]
+    cases = {
+        1: [{'id' : 'return_first'}],
+        2: [{'id' : 'return_last'}],
+        4: [{'id' : 'return_greatest'}],
+        3: [{'id' : 'return_positive'}],
+        -6: [{'id' : 'return_absolute'}]
+    }
+    if res in cases:
+        return cases[res]
+
+    # if has no return at the beginning of block,
+    # tell not to return from for
+    if not has_token_sequence(tokens, ['\n', 'return']):
+        return [{'id' : 'return_indent'}]
+
+    return None
diff --git a/python/problems/functions/greatest_negative/en.py b/python/problems/functions/greatest_negative/en.py
new file mode 100644
index 0000000..92e5841
--- /dev/null
+++ b/python/problems/functions/greatest_negative/en.py
@@ -0,0 +1,16 @@
+# coding=utf-8
+
+id = 197
+name = '(translation missing)'
+slug = '(translation missing)'
+
+description = '''\
+<p>(translation missing)</p>'''
+
+hint = {
+    'plan': '''\
+<p>(translation missing)</p>''',
+
+    'no_input_call': '''\
+<p>(translation missing)</p>''',
+}
diff --git a/python/problems/functions/greatest_negative/sl.py b/python/problems/functions/greatest_negative/sl.py
new file mode 100644
index 0000000..c0b7515
--- /dev/null
+++ b/python/problems/functions/greatest_negative/sl.py
@@ -0,0 +1,97 @@
+# coding=utf-8
+import server
+mod = server.problems.load_language('python', 'sl')
+
+
+id = 197
+name = 'Največji negativec'
+slug = 'Največji negativec'
+
+
+description = '''\
+<p>Napiši funkcijo <code>max_neg(xs)</code>, ki vrne največje negativno število
+v seznamu <code>xs</code>. Če seznam nima negativnih števil, naj funkcija vrne <code>None</code>.</p>
+<pre>
+>>> max_neg([5, 1, -6, -7, 2])
+-6
+</pre>
+'''
+
+function = ['''\
+<p>Napiši <b>definicijo funkcije</b> <code>max_neg(xs)</code>.</p>''',
+            '''\
+<p>Definicijo funkcije začnemo z <code>def</code>, temu sledi ime, potem oklepaji,
+v katerih naštejemo argumente funkcije, nato zaklepaj in na koncu dvopičje</p>''',
+            '''\
+<pre>
+def max_neg(xs):
+    # poišči največje negativno število v xs
+
+s = [5, 1, -6, -7, 2]
+print (max_neg(s))
+</pre>''']
+
+main_plan = ['''\
+<p><b>Plan</b>: po vrsti pogledamo vse elemente in sproti hranimo največje negativno
+število.</p>''',
+             '''\
+<p>Na začetku ne moremo največjemu nastaviti vrednosti prvega elementa v seznamu,
+saj je lahko prvi element pozitiven.</p>''',
+             '''\
+</p>Nastavite ga na <code>None</code></p>''',
+             ]
+
+return_clause = ['''\
+<p>Namesto, da izpišemo rezultat, ga vračamo s stavkom <code>return</code>.</p>''',
+                 '''\
+<p>Pazi, da vrneš rezultat, šele ko se zanka izteče.</p>''']
+
+plan = [function,
+        main_plan,
+        return_clause]
+
+for_loop = ['''\
+<p>Preglej elemente z zanko</p>.
+''',
+             '''\
+<pre>
+for x in xs:
+</pre>''']
+
+
+if_clause = ['''\
+<p>Preveri, če je trenutni element negativen in večji od trenutno največjega negativca.</p>''',]
+
+
+hint = {
+    'no_def': function,
+
+    'no_return': return_clause,
+
+    'for_loop': for_loop,
+
+    'if_clause': if_clause,
+
+    'not_int': '''\
+<p>Funkcija ne vrača števila</p>''',
+
+    'return_first': '''\
+<p>Funkcija vrača prvi element v seznamu''',
+
+    'return_last': '''\
+<p>Funkcija vrača zadnji element v seznamu.''',
+
+    'return_greatest': '''\
+<p>Funkcija vrača največji pozitivni element v seznamu (ne pa največji negativni).''',
+
+    'return_positive': '''\
+<p>Funkcija napačno vrača pozitivno vrednost največjega negativnega števila.''',
+
+    'return_absolute': '''\
+<p>Funkcija napačno vrača element z največjo absolutno vrednostjo.</p>''',
+
+    'return_indent': '''\
+<p>Ali imaš stavek <code>return</code> znotraj zanke?
+V tem primeru se lahko zgodi, da se zanka ne izteče do konca.</p>'''
+}
+
-- 
cgit v1.2.1