Convert And and Or to kotlin
This commit is contained in:
		
							parent
							
								
									ac821abde2
								
							
						
					
					
						commit
						8ca2eb199c
					
				| @ -1,43 +0,0 @@ | ||||
| package function.builtin.special; | ||||
| 
 | ||||
| import function.ArgumentValidator; | ||||
| import function.FunctionNames; | ||||
| import function.LispSpecialFunction; | ||||
| import recursion.TailCall; | ||||
| import sexpression.Cons; | ||||
| import sexpression.SExpression; | ||||
| import sexpression.Symbol; | ||||
| 
 | ||||
| import static function.builtin.Eval.eval; | ||||
| import static recursion.TailCalls.done; | ||||
| import static recursion.TailCalls.tailCall; | ||||
| 
 | ||||
| @FunctionNames({ "AND" }) | ||||
| public class AND extends LispSpecialFunction { | ||||
| 
 | ||||
|     private ArgumentValidator argumentValidator; | ||||
| 
 | ||||
|     public AND(String name) { | ||||
|         this.argumentValidator = new ArgumentValidator(name); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public SExpression call(Cons argumentList) { | ||||
|         argumentValidator.validate(argumentList); | ||||
| 
 | ||||
|         return callTailRecursive(argumentList, Symbol.Companion.getT()).invoke(); | ||||
|     } | ||||
| 
 | ||||
|     private TailCall<SExpression> callTailRecursive(Cons argumentList, SExpression lastValue) { | ||||
|         SExpression currentValue = eval(argumentList.getFirst()); | ||||
|         Cons remainingValues = (Cons) argumentList.getRest(); | ||||
| 
 | ||||
|         if (argumentList.isNull()) | ||||
|             return done(lastValue); | ||||
| 
 | ||||
|         if (currentValue.isNull()) | ||||
|             return done(currentValue); | ||||
| 
 | ||||
|         return tailCall(() -> callTailRecursive(remainingValues, currentValue)); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										32
									
								
								src/main/kotlin/function/builtin/special/And.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										32
									
								
								src/main/kotlin/function/builtin/special/And.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,32 @@ | ||||
| package function.builtin.special | ||||
| 
 | ||||
| import function.ArgumentValidator | ||||
| import function.FunctionNames | ||||
| import function.LispSpecialFunction | ||||
| import function.builtin.Eval.Companion.eval | ||||
| import sexpression.Cons | ||||
| import sexpression.SExpression | ||||
| import sexpression.Symbol.Companion.T | ||||
| 
 | ||||
| @FunctionNames("AND") | ||||
| class And(name: String) : LispSpecialFunction() { | ||||
| 
 | ||||
|     private val argumentValidator = ArgumentValidator(name) | ||||
| 
 | ||||
|     override fun call(argumentList: Cons): SExpression { | ||||
|         argumentValidator.validate(argumentList) | ||||
| 
 | ||||
|         return callTailRecursive(argumentList, T) | ||||
|     } | ||||
| 
 | ||||
|     private tailrec fun callTailRecursive(argumentList: Cons, lastValue: SExpression): SExpression { | ||||
|         val currentValue = eval(argumentList.first) | ||||
|         val remainingValues = argumentList.rest as Cons | ||||
| 
 | ||||
|         return when { | ||||
|             argumentList.isNull -> lastValue | ||||
|             currentValue.isNull -> currentValue | ||||
|             else                -> callTailRecursive(remainingValues, currentValue) | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -1,39 +0,0 @@ | ||||
| package function.builtin.special; | ||||
| 
 | ||||
| import function.ArgumentValidator; | ||||
| import function.FunctionNames; | ||||
| import function.LispSpecialFunction; | ||||
| import recursion.TailCall; | ||||
| import sexpression.Cons; | ||||
| import sexpression.SExpression; | ||||
| 
 | ||||
| import static function.builtin.Eval.eval; | ||||
| import static recursion.TailCalls.done; | ||||
| import static recursion.TailCalls.tailCall; | ||||
| 
 | ||||
| @FunctionNames({ "OR" }) | ||||
| public class OR extends LispSpecialFunction { | ||||
| 
 | ||||
|     private ArgumentValidator argumentValidator; | ||||
| 
 | ||||
|     public OR(String name) { | ||||
|         this.argumentValidator = new ArgumentValidator(name); | ||||
|     } | ||||
| 
 | ||||
|     @Override | ||||
|     public SExpression call(Cons argumentList) { | ||||
|         argumentValidator.validate(argumentList); | ||||
| 
 | ||||
|         return callTailRecursive(argumentList).invoke(); | ||||
|     } | ||||
| 
 | ||||
|     private TailCall<SExpression> callTailRecursive(Cons argumentList) { | ||||
|         SExpression currentValue = eval(argumentList.getFirst()); | ||||
|         Cons remainingValues = (Cons) argumentList.getRest(); | ||||
| 
 | ||||
|         if (remainingValues.isNull() || !currentValue.isNull()) | ||||
|             return done(currentValue); | ||||
| 
 | ||||
|         return tailCall(() -> callTailRecursive(remainingValues)); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										30
									
								
								src/main/kotlin/function/builtin/special/Or.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										30
									
								
								src/main/kotlin/function/builtin/special/Or.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,30 @@ | ||||
| package function.builtin.special | ||||
| 
 | ||||
| import function.ArgumentValidator | ||||
| import function.FunctionNames | ||||
| import function.LispSpecialFunction | ||||
| import function.builtin.Eval.Companion.eval | ||||
| import sexpression.Cons | ||||
| import sexpression.SExpression | ||||
| 
 | ||||
| @FunctionNames("OR") | ||||
| class Or(name: String) : LispSpecialFunction() { | ||||
| 
 | ||||
|     private val argumentValidator = ArgumentValidator(name) | ||||
| 
 | ||||
|     override fun call(argumentList: Cons): SExpression { | ||||
|         argumentValidator.validate(argumentList) | ||||
| 
 | ||||
|         return callTailRecursive(argumentList) | ||||
|     } | ||||
| 
 | ||||
|     private tailrec fun callTailRecursive(argumentList: Cons): SExpression { | ||||
|         val currentValue = eval(argumentList.first) | ||||
|         val remainingValues = argumentList.rest as Cons | ||||
| 
 | ||||
|         return if (remainingValues.isNull || !currentValue.isNull) | ||||
|             currentValue | ||||
|         else | ||||
|             callTailRecursive(remainingValues) | ||||
|     } | ||||
| } | ||||
| @ -1,64 +0,0 @@ | ||||
| package function.builtin.special; | ||||
| 
 | ||||
| import function.builtin.Eval.UndefinedSymbolException; | ||||
| import org.junit.Test; | ||||
| import sexpression.LispNumber; | ||||
| import testutil.SymbolAndFunctionCleaner; | ||||
| 
 | ||||
| import static testutil.TestUtilities.assertSExpressionsMatch; | ||||
| import static testutil.TestUtilities.evaluateString; | ||||
| import static testutil.TypeAssertions.assertNil; | ||||
| import static testutil.TypeAssertions.assertT; | ||||
| 
 | ||||
| public class ANDTest extends SymbolAndFunctionCleaner { | ||||
| 
 | ||||
|     @Test | ||||
|     public void andByItself() { | ||||
|         String input = "(and)"; | ||||
| 
 | ||||
|         assertT(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void andWithNil() { | ||||
|         String input = "(and nil)"; | ||||
| 
 | ||||
|         assertNil(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void andWithT() { | ||||
|         String input = "(and t)"; | ||||
| 
 | ||||
|         assertT(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void andWithNumber() { | ||||
|         String input = "(and 7)"; | ||||
| 
 | ||||
|         assertSExpressionsMatch(new LispNumber("7"), evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void andWithSeveralValues() { | ||||
|         String input = "(and t t nil t t)"; | ||||
| 
 | ||||
|         assertNil(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void andWithSeveralNumbers() { | ||||
|         String input = "(and 1 2 3)"; | ||||
| 
 | ||||
|         assertSExpressionsMatch(new LispNumber("3"), evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test(expected = UndefinedSymbolException.class) | ||||
|     public void andShortCircuits() { | ||||
|         String input = "(and nil (setq x 22))"; | ||||
| 
 | ||||
|         assertNil(evaluateString(input)); | ||||
|         evaluateString("x"); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										69
									
								
								src/test/kotlin/function/builtin/special/AndTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/test/kotlin/function/builtin/special/AndTest.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | ||||
| package function.builtin.special | ||||
| 
 | ||||
| import function.builtin.Eval.UndefinedSymbolException | ||||
| import org.junit.jupiter.api.Assertions.assertThrows | ||||
| import org.junit.jupiter.api.Test | ||||
| import sexpression.LispNumber | ||||
| import testutil.LispTestInstance | ||||
| import testutil.SymbolAndFunctionCleaner | ||||
| import testutil.TestUtilities.assertSExpressionsMatch | ||||
| import testutil.TestUtilities.evaluateString | ||||
| import testutil.TypeAssertions.assertNil | ||||
| import testutil.TypeAssertions.assertT | ||||
| 
 | ||||
| @LispTestInstance | ||||
| class AndTest : SymbolAndFunctionCleaner() { | ||||
| 
 | ||||
|     @Test | ||||
|     fun andByItself() { | ||||
|         val input = "(and)" | ||||
| 
 | ||||
|         assertT(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun andWithNil() { | ||||
|         val input = "(and nil)" | ||||
| 
 | ||||
|         assertNil(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun andWithT() { | ||||
|         val input = "(and t)" | ||||
| 
 | ||||
|         assertT(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun andWithNumber() { | ||||
|         val input = "(and 7)" | ||||
| 
 | ||||
|         assertSExpressionsMatch(LispNumber("7"), evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun andWithSeveralValues() { | ||||
|         val input = "(and t t nil t t)" | ||||
| 
 | ||||
|         assertNil(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun andWithSeveralNumbers() { | ||||
|         val input = "(and 1 2 3)" | ||||
| 
 | ||||
|         assertSExpressionsMatch(LispNumber("3"), evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun andShortCircuits() { | ||||
|         val input = "(and nil (setq x 22))" | ||||
| 
 | ||||
|         assertNil(evaluateString(input)) | ||||
| 
 | ||||
|         assertThrows(UndefinedSymbolException::class.java) { | ||||
|             evaluateString("x") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -16,11 +16,13 @@ import sexpression.LispNumber.Companion.ONE | ||||
| import sexpression.Nil | ||||
| import sexpression.Symbol | ||||
| import sexpression.Symbol.Companion.T | ||||
| import testutil.LispTestInstance | ||||
| import testutil.SymbolAndFunctionCleaner | ||||
| import testutil.TestUtilities.assertSExpressionsMatch | ||||
| import testutil.TestUtilities.evaluateString | ||||
| import testutil.TestUtilities.parseString | ||||
| 
 | ||||
| @LispTestInstance | ||||
| class LambdaTest : SymbolAndFunctionCleaner() { | ||||
| 
 | ||||
|     @Test | ||||
|  | ||||
| @ -1,64 +0,0 @@ | ||||
| package function.builtin.special; | ||||
| 
 | ||||
| import function.builtin.Eval.UndefinedSymbolException; | ||||
| import org.junit.Test; | ||||
| import sexpression.LispNumber; | ||||
| import testutil.SymbolAndFunctionCleaner; | ||||
| 
 | ||||
| import static testutil.TestUtilities.assertSExpressionsMatch; | ||||
| import static testutil.TestUtilities.evaluateString; | ||||
| import static testutil.TypeAssertions.assertNil; | ||||
| import static testutil.TypeAssertions.assertT; | ||||
| 
 | ||||
| public class ORTest extends SymbolAndFunctionCleaner { | ||||
| 
 | ||||
|     @Test | ||||
|     public void orByItself() { | ||||
|         String input = "(or)"; | ||||
| 
 | ||||
|         assertNil(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void orWithNil() { | ||||
|         String input = "(or nil)"; | ||||
| 
 | ||||
|         assertNil(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void orWithT() { | ||||
|         String input = "(or t)"; | ||||
| 
 | ||||
|         assertT(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void orWithNumber() { | ||||
|         String input = "(or 7)"; | ||||
| 
 | ||||
|         assertSExpressionsMatch(new LispNumber("7"), evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void orWithSeveralValues() { | ||||
|         String input = "(or nil nil nil t nil)"; | ||||
| 
 | ||||
|         assertT(evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void orWithSeveralNumbers() { | ||||
|         String input = "(or 1 2 3)"; | ||||
| 
 | ||||
|         assertSExpressionsMatch(new LispNumber("1"), evaluateString(input)); | ||||
|     } | ||||
| 
 | ||||
|     @Test(expected = UndefinedSymbolException.class) | ||||
|     public void orShortCircuits() { | ||||
|         String input = "(or t (setq x 22))"; | ||||
| 
 | ||||
|         assertT(evaluateString(input)); | ||||
|         evaluateString("x"); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										69
									
								
								src/test/kotlin/function/builtin/special/OrTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										69
									
								
								src/test/kotlin/function/builtin/special/OrTest.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,69 @@ | ||||
| package function.builtin.special | ||||
| 
 | ||||
| import function.builtin.Eval.UndefinedSymbolException | ||||
| import org.junit.jupiter.api.Assertions.assertThrows | ||||
| import org.junit.jupiter.api.Test | ||||
| import sexpression.LispNumber | ||||
| import testutil.LispTestInstance | ||||
| import testutil.SymbolAndFunctionCleaner | ||||
| import testutil.TestUtilities.assertSExpressionsMatch | ||||
| import testutil.TestUtilities.evaluateString | ||||
| import testutil.TypeAssertions.assertNil | ||||
| import testutil.TypeAssertions.assertT | ||||
| 
 | ||||
| @LispTestInstance | ||||
| class OrTest : SymbolAndFunctionCleaner() { | ||||
| 
 | ||||
|     @Test | ||||
|     fun orByItself() { | ||||
|         val input = "(or)" | ||||
| 
 | ||||
|         assertNil(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun orWithNil() { | ||||
|         val input = "(or nil)" | ||||
| 
 | ||||
|         assertNil(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun orWithT() { | ||||
|         val input = "(or t)" | ||||
| 
 | ||||
|         assertT(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun orWithNumber() { | ||||
|         val input = "(or 7)" | ||||
| 
 | ||||
|         assertSExpressionsMatch(LispNumber("7"), evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun orWithSeveralValues() { | ||||
|         val input = "(or nil nil nil t nil)" | ||||
| 
 | ||||
|         assertT(evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun orWithSeveralNumbers() { | ||||
|         val input = "(or 1 2 3)" | ||||
| 
 | ||||
|         assertSExpressionsMatch(LispNumber("1"), evaluateString(input)) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun orShortCircuits() { | ||||
|         val input = "(or t (setq x 22))" | ||||
| 
 | ||||
|         assertT(evaluateString(input)) | ||||
| 
 | ||||
|         assertThrows(UndefinedSymbolException::class.java) { | ||||
|             evaluateString("x") | ||||
|         } | ||||
|     } | ||||
| } | ||||
| @ -1,94 +0,0 @@ | ||||
| package function.builtin.special; | ||||
| 
 | ||||
| import function.ArgumentValidator.BadArgumentTypeException; | ||||
| import function.ArgumentValidator.TooFewArgumentsException; | ||||
| import function.ArgumentValidator.TooManyArgumentsException; | ||||
| import function.builtin.Eval.UndefinedSymbolException; | ||||
| import org.junit.Test; | ||||
| import sexpression.LispNumber; | ||||
| import table.SymbolTable; | ||||
| import testutil.SymbolAndFunctionCleaner; | ||||
| 
 | ||||
| import static org.junit.Assert.assertNull; | ||||
| import static testutil.TestUtilities.assertSExpressionsMatch; | ||||
| import static testutil.TestUtilities.evaluateString; | ||||
| 
 | ||||
| public class SetqTest extends SymbolAndFunctionCleaner { | ||||
| 
 | ||||
|     @Test | ||||
|     public void setq() { | ||||
|         evaluateString("(setq a 23)"); | ||||
|         assertSExpressionsMatch(new LispNumber("23"), evaluateString("a")); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void lookupDefinedSymbol() { | ||||
|         evaluateString("(setq a 23)"); | ||||
|         assertSExpressionsMatch(new LispNumber("23"), getExecutionContext().lookupSymbolValue("A")); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void lookupUndefinedSymbol() { | ||||
|         assertNull(getExecutionContext().lookupSymbolValue("A")); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void setqGlobalVariable() { | ||||
|         evaluateString("(setq a 23)"); | ||||
|         SymbolTable global = getExecutionContext().getScope(); | ||||
|         getExecutionContext().setScope(new SymbolTable(global)); | ||||
| 
 | ||||
|         evaluateString("(setq a 94)"); | ||||
|         getExecutionContext().setScope(global); | ||||
|         assertSExpressionsMatch(new LispNumber("94"), evaluateString("a")); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void setqLocalVariable() { | ||||
|         SymbolTable global = getExecutionContext().getScope(); | ||||
|         SymbolTable local = new SymbolTable(global); | ||||
|         local.set("A", new LispNumber("99")); | ||||
|         getExecutionContext().setScope(local); | ||||
| 
 | ||||
|         evaluateString("(setq a 94)"); | ||||
|         assertSExpressionsMatch(new LispNumber("94"), evaluateString("a")); | ||||
|     } | ||||
| 
 | ||||
|     @Test(expected = UndefinedSymbolException.class) | ||||
|     public void setqLocalVariableDefined_DoesNotSetGlobal() { | ||||
|         SymbolTable global = getExecutionContext().getScope(); | ||||
|         SymbolTable local = new SymbolTable(global); | ||||
|         local.set("A", new LispNumber("99")); | ||||
|         getExecutionContext().setScope(local); | ||||
| 
 | ||||
|         evaluateString("(setq a 94)"); | ||||
|         getExecutionContext().setScope(global); | ||||
|         evaluateString("a"); | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     public void setqLocalVariableUndefined_SetsGlobal() { | ||||
|         SymbolTable global = getExecutionContext().getScope(); | ||||
|         SymbolTable local = new SymbolTable(global); | ||||
|         getExecutionContext().setScope(local); | ||||
| 
 | ||||
|         evaluateString("(setq a 94)"); | ||||
|         getExecutionContext().setScope(global); | ||||
|         assertSExpressionsMatch(new LispNumber("94"), evaluateString("a")); | ||||
|     } | ||||
| 
 | ||||
|     @Test(expected = BadArgumentTypeException.class) | ||||
|     public void setqWithNonSymbol() { | ||||
|         evaluateString("(setq 1 2)"); | ||||
|     } | ||||
| 
 | ||||
|     @Test(expected = TooFewArgumentsException.class) | ||||
|     public void setqWithTooFewArguments() { | ||||
|         evaluateString("(setq x)"); | ||||
|     } | ||||
| 
 | ||||
|     @Test(expected = TooManyArgumentsException.class) | ||||
|     public void setqWithTooManyArguments() { | ||||
|         evaluateString("(setq a b c)"); | ||||
|     } | ||||
| } | ||||
							
								
								
									
										105
									
								
								src/test/kotlin/function/builtin/special/SetqTest.kt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										105
									
								
								src/test/kotlin/function/builtin/special/SetqTest.kt
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,105 @@ | ||||
| package function.builtin.special | ||||
| 
 | ||||
| import function.ArgumentValidator.BadArgumentTypeException | ||||
| import function.ArgumentValidator.TooFewArgumentsException | ||||
| import function.ArgumentValidator.TooManyArgumentsException | ||||
| import function.builtin.Eval.UndefinedSymbolException | ||||
| import org.assertj.core.api.Assertions.assertThat | ||||
| import org.junit.jupiter.api.Assertions.assertThrows | ||||
| import org.junit.jupiter.api.Test | ||||
| import sexpression.LispNumber | ||||
| import table.SymbolTable | ||||
| import testutil.LispTestInstance | ||||
| import testutil.SymbolAndFunctionCleaner | ||||
| import testutil.TestUtilities.assertSExpressionsMatch | ||||
| import testutil.TestUtilities.evaluateString | ||||
| 
 | ||||
| @LispTestInstance | ||||
| class SetqTest : SymbolAndFunctionCleaner() { | ||||
| 
 | ||||
|     @Test | ||||
|     fun setq() { | ||||
|         evaluateString("(setq a 23)") | ||||
|         assertSExpressionsMatch(LispNumber("23"), evaluateString("a")) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun lookupDefinedSymbol() { | ||||
|         evaluateString("(setq a 23)") | ||||
|         assertSExpressionsMatch(LispNumber("23"), executionContext.lookupSymbolValue("A")!!) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun lookupUndefinedSymbol() { | ||||
|         assertThat(executionContext.lookupSymbolValue("A")).isNull() | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqGlobalVariable() { | ||||
|         evaluateString("(setq a 23)") | ||||
|         val global = executionContext.scope | ||||
|         executionContext.scope = SymbolTable(global) | ||||
| 
 | ||||
|         evaluateString("(setq a 94)") | ||||
|         executionContext.scope = global | ||||
|         assertSExpressionsMatch(LispNumber("94"), evaluateString("a")) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqLocalVariable() { | ||||
|         val global = executionContext.scope | ||||
|         val local = SymbolTable(global) | ||||
|         local["A"] = LispNumber("99") | ||||
|         executionContext.scope = local | ||||
| 
 | ||||
|         evaluateString("(setq a 94)") | ||||
|         assertSExpressionsMatch(LispNumber("94"), evaluateString("a")) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqLocalVariableDefined_DoesNotSetGlobal() { | ||||
|         val global = executionContext.scope | ||||
|         val local = SymbolTable(global) | ||||
|         local["A"] = LispNumber("99") | ||||
|         executionContext.scope = local | ||||
| 
 | ||||
|         evaluateString("(setq a 94)") | ||||
|         executionContext.scope = global | ||||
| 
 | ||||
|         assertThrows(UndefinedSymbolException::class.java) { | ||||
|             evaluateString("a") | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqLocalVariableUndefined_SetsGlobal() { | ||||
|         val global = executionContext.scope | ||||
|         val local = SymbolTable(global) | ||||
|         executionContext.scope = local | ||||
| 
 | ||||
|         evaluateString("(setq a 94)") | ||||
|         executionContext.scope = global | ||||
|         assertSExpressionsMatch(LispNumber("94"), evaluateString("a")) | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqWithNonSymbol() { | ||||
|         assertThrows(BadArgumentTypeException::class.java) { | ||||
|             evaluateString("(setq 1 2)") | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqWithTooFewArguments() { | ||||
|         assertThrows(TooFewArgumentsException::class.java) { | ||||
|             evaluateString("(setq x)") | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     @Test | ||||
|     fun setqWithTooManyArguments() { | ||||
|         assertThrows(TooManyArgumentsException::class.java) { | ||||
|             evaluateString("(setq a b c)") | ||||
|         } | ||||
|     } | ||||
| } | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user