import scala.
util.
parsing.
combinator.
_
def expr
= chainl1
(term, parseIffImp
) ; def parseIffImp
= "<=>" ^^^
{Iff
} |
"=>" ^^^
{Imp
} def term
= chainl1
(factor, parseAndOr
) ; def parseAndOr
= "/\\" ^^^
{And
} |
"\\/" ^^^
{Or
} def factor
: Parser
[Expr
] = "~" ~
> factor ^^
{Not
} | primary
def primary
= "#T" ^^^
{Const
(true)} |
"#F" ^^^
{Const
(false)} | ident ^^
{Atom
} |
"(" ~
> expr
<~
")" }
def toStr
(expr
: Expr, p
: Int, r
: Int
): String
= {
def par
(p0
: Int, s
: String
) = if (p
> p0 ||
(p
== p0
&& r
== 1)) s
else ""
case Atom
(name
) => name
case Not
(e
) => "~" + toStr
(e,
3,r
) case And
(e1, e2
) => par
(2,
"(") + toStr
(e1,
2,
0) +
"/\\" + toStr
(e2,
2,
1) + par
(2,
")") case Or
(e1, e2
) => par
(2,
"(") + toStr
(e1,
2,
0) +
"\\/" + toStr
(e2,
2,
1) + par
(2,
")") case Imp
(e1, e2
) => par
(1,
"(") + toStr
(e1,
1,
0) +
"=>" + toStr
(e2,
1,
1) + par
(1,
")") case Iff
(e1, e2
) => par
(1,
"(") + toStr
(e1,
1,
0) +
"<=>" + toStr
(e2,
1,
1) + par
(1,
")") } }
def simplify
(expr
: Expr
): Expr
= {
def simplifyH
(expr
: Expr
) = expr
match {
case Not
(p
) => simplifyH
(Not
(simplify
(p
))) case And
(p, q
) => simplifyH
(And
(simplify
(p
), simplify
(q
))) case Or
(p, q
) => simplifyH
( Or
(simplify
(p
), simplify
(q
))) case Iff
(p, q
) => simplifyH
(Iff
(simplify
(p
), simplify
(q
))) case Imp
(p, q
) => simplifyH
(Imp
(simplify
(p
), simplify
(q
))) case e
=> e
} }
def nnf
(expr
: Expr
): Expr
= expr
match { case And
(p, q
) => And
(nnf
(p
), nnf
(q
)) case Or
(p, q
) => Or
(nnf
(p
), nnf
(q
)) case Imp
(p, q
) => Or
(nnf
(Not
(p
)), nnf
(q
)) case Iff
(p, q
) => Or
(And
(nnf
(p
), nnf
(q
)), And
(nnf
(Not
(p
)), nnf
(Not
(q
)))) case Not
(Not
(p
)) => nnf
(p
) case Not
(And
(p, q
)) => Or
(nnf
(Not
(p
)), nnf
(Not
(q
))) case Not
( Or
(p, q
)) => And
(nnf
(Not
(p
)), nnf
(Not
(q
))) case Not
(Imp
(p, q
)) => And
(nnf
(p
), nnf
(Not
(q
))) case Not
(Iff
(p, q
)) => Or
(And
(nnf
(p
), nnf
(Not
(q
))), And
(nnf
(Not
(p
)), nnf
(q
))) case e
=> e
}
def cnf
(expr
: Expr
): Expr
= {
def distr
(expr
: Expr
): Expr
= expr
match { case Or
(And
(p, q
), And
(r, s
)) => And
(And
(distr
(Or
(p, r
)), distr
(Or
(p, s
))),
And(distr(Or(q, r)), distr(Or(q, s))))
case Or
(p, And
(q, r
)) => And
(distr
(Or
(p, q
)), distr
(Or
(p, r
))) case Or
(And
(p, q
), r
) => And
(distr
(Or
(p, r
)), distr
(Or
(q, r
))) case e
=> e
}
case Or
(p, q
) => distr
(Or
(cnf
(p
), cnf
(q
))) case And
(p, q
) => And
(cnf
(p
), cnf
(q
)) case e
=> e
} }
def lcnf
(expr
: Expr
): List
[List
[Expr
]] = {
def lcnfH1
(list
: List
[Expr
], expr
: Expr
): List
[Expr
] = expr
match { case And
(p, q
) => list
::: lcnfH1
(List
(), p
) ::: lcnfH1
(List
(), q
) case e
=> e
:: list
} def lcnfH2
(list
: List
[Expr
])(expr
: Expr
): List
[Expr
] = expr
match { case Or
(p, q
) => list
::: lcnfH2
(List
())(p
) ::: lcnfH2
(List
())(q
) case e
=> e
:: list
}
lcnfH1(List(), expr) map lcnfH2(List())_
}
def resolution
(clist
: List
[List
[Expr
]]): Boolean
= {
def splitClause
(clause
: List
[Expr
]) = { val (pos, neg
) = clause partition
(isAtom
_) ; List
(pos, neg map
(removeNot
_)) } def isTrueClause
(clause
: List
[Expr
]) = { val List
(lits, nlits
) = splitClause
(clause
) ((lits map
((e
: Expr
) => nlits exists
( _ == e
))) :\
false)(_||
_) } def isLitTrueClause
(clause
: List
[Expr
]) = clause
match { def simplifyClause
(clause
: List
[Expr
]) = if (isTrueClause
(clause
)) List
(Const
(true)) else clause
def simplifyLcnf
(clist
: List
[List
[Expr
]]) = { (clist map (simplifyClause _)) remove (isLitTrueClause _) }
def s2l
[T
](set
: Set
[T
]) = set.
toList ; def l2s
[T
](list
: List
[T
]) = list.
toSet
def resolutionH2
(tp4
: (List
[Expr
], List
[List
[Expr
]], List
[List
[Expr
]], Boolean
), lit
: Expr
) = { val (c, cs, d, b
) = tp4
; if (d
== List
()) (c, cs, d, b
) else { val nlit
= negateLit
(lit
) ; val c1
= c diff List
(lit
) val cs1
= (cs filter
((c
: List
[Expr
]) => c exists
(_ == nlit
))) map
(_ diff List
(nlit
)) var d1
= cs1 map
(_ ::: c1
) ; val cs2h
= ((cs map l2s
) union
(d1 map l2s
)).
distinct val cs2
= (cs2h map s2l
) remove
(isTrueClause
_) ; d1
= cs2 diff cs
val b1
= if (d1.
isEmpty) false else (d1 exists
(_ == List
()) ) ; (c, cs2, d1, b1
) }
}
def resolutionH1
(c
: List
[Expr
], cs
: List
[List
[Expr
]], d
: List
[List
[Expr
]], b
: Boolean
): Boolean
= case (List
(), c
::cs, d, b
) => resolutionH1
(c, cs, d, b
) val (c1, cs1, d1, b1
) = ((c, cs, d ++ List
(List
(Atom
(""))), b
) /
: c
)(resolutionH2
) resolutionH1(List(), cs1, d1, b1) } }
case _ => resolutionH1
(List
(), simplifyLcnf
(clist
) map
((c
: List
[Expr
]) => s2l
(l2s
(c
))), List
(),
false) } }
def main
(args
: Array
[String
]) {
do { input
= readLine
() ; if (input
!= null && input
!= "") {
val output
= parseAll
(expr, input
) match { case Failure
(msg,
_) => "failure: " + msg
case Error
(msg,
_) => "error: " + msg
case Success
(parsed,
_) => { val nResRet
= resolution
(lcnf
(cnf
(simplify
(nnf
(simplify
(Not
(parsed
))))))) val pResRet
= resolution
(lcnf
(cnf
(simplify
(nnf
(simplify
(parsed
)))))) toStr(parsed, 0, 0) + "\n" +
(if (nResRet
) "true" else if (pResRet
) "false" else "it depends") }
}
println(output) }
}
}
import scala.util.parsing.combinator._

abstract class Expr
case class Const(value: Boolean) extends Expr
case class Atom(name: String) extends Expr
case class Not(expr: Expr) extends Expr
case class And(left: Expr, right: Expr) extends Expr
case class  Or(left: Expr, right: Expr) extends Expr
case class Iff(left: Expr, right: Expr) extends Expr
case class Imp(left: Expr, right: Expr) extends Expr
 
class MyParser extends JavaTokenParsers {

    def expr = chainl1(term, parseIffImp) ; def parseIffImp = "<=>" ^^^ {Iff} | "=>" ^^^ {Imp}
    def term = chainl1(factor, parseAndOr) ; def parseAndOr = "/\\" ^^^ {And} | "\\/" ^^^ {Or}
    def factor: Parser[Expr] = "~" ~> factor ^^ {Not} | primary 
    def primary = "#T" ^^^ {Const(true)} | "#F" ^^^ {Const(false)} | ident ^^ {Atom} | "(" ~> expr <~ ")"
}

object Main extends MyParser {

    def toStr(expr: Expr, p: Int, r: Int): String = {
    
        def par(p0: Int, s: String) = if (p > p0 || (p == p0 && r == 1)) s else ""
    
        expr match {
            case Const(true) => "#true" case Const(false) => "#false"
            case Atom(name) => name case Not(e) => "~" + toStr(e,3,r)
            case And(e1, e2) => par(2,"(") + toStr(e1,2,0) + "/\\" + toStr(e2,2,1) + par(2,")")
            case  Or(e1, e2) => par(2,"(") + toStr(e1,2,0) + "\\/" + toStr(e2,2,1) + par(2,")")
            case Imp(e1, e2) => par(1,"(") + toStr(e1,1,0) + "=>"  + toStr(e2,1,1) + par(1,")")
            case Iff(e1, e2) => par(1,"(") + toStr(e1,1,0) + "<=>" + toStr(e2,1,1) + par(1,")") }
    }

    def simplify(expr: Expr): Expr = {
    
        def simplifyH(expr: Expr) = expr match {
            case Not(   Const(true))  => Const(false) case Not(Const(false))    => Const(true)
            case And(p, Const(false)) => Const(false) case And(Const(false), p) => Const(false)
            case And(p, Const(true))  => p            case And(Const(true),  p) => p
            case  Or(p, Const(false)) => p            case  Or(Const(false), p) => p
            case  Or(p, Const(true))  => Const(true)  case  Or(Const(true),  p) => Const(true)
            case Imp(p, Const(true))  => Const(true)  case Imp(Const(true),  p) => p
            case Imp(p, Const(false)) => Not(p)       case Imp(Const(false), p) => Const(true)
            case Iff(p, Const(true))  => p            case Iff(Const(true),  p) => p
            case Iff(p, Const(false)) => Not(p)       case Iff(Const(false), p) => Not(p)                
            case Not(Not(p)) => p                     case e => e }
        
        expr match {
            case Not(p)    => simplifyH(Not(simplify(p)))
            case And(p, q) => simplifyH(And(simplify(p), simplify(q)))
            case  Or(p, q) => simplifyH( Or(simplify(p), simplify(q)))
            case Iff(p, q) => simplifyH(Iff(simplify(p), simplify(q)))
            case Imp(p, q) => simplifyH(Imp(simplify(p), simplify(q))) case e => e }
    }

    def nnf(expr: Expr): Expr = expr match {
        case And(p, q) => And(nnf(p), nnf(q)) case  Or(p, q) =>  Or(nnf(p), nnf(q))
        case Imp(p, q) =>  Or(nnf(Not(p)), nnf(q))
        case Iff(p, q) =>  Or(And(nnf(p), nnf(q)), And(nnf(Not(p)), nnf(Not(q))))
        case Not(Not(p)) => nnf(p)
        case Not(And(p, q)) =>  Or(nnf(Not(p)), nnf(Not(q)))
        case Not( Or(p, q)) => And(nnf(Not(p)), nnf(Not(q)))
        case Not(Imp(p, q)) => And(nnf(p), nnf(Not(q)))
        case Not(Iff(p, q)) =>  Or(And(nnf(p), nnf(Not(q))), And(nnf(Not(p)), nnf(q))) case e => e }
    
    def cnf(expr: Expr): Expr = {
    
        def distr(expr: Expr): Expr = expr match {
            case Or(And(p, q), And(r, s)) => And(And(distr(Or(p, r)), distr(Or(p, s))), 
                                                 And(distr(Or(q, r)), distr(Or(q, s))))
            case Or(p, And(q, r)) => And(distr(Or(p, q)), distr(Or(p, r)))
            case Or(And(p, q), r) => And(distr(Or(p, r)), distr(Or(q, r))) case e => e }

        expr match {
            case  Or(p, q) => distr(Or(cnf(p), cnf(q))) case And(p, q) => And(cnf(p), cnf(q)) case e => e }
    }

    def lcnf(expr: Expr): List[List[Expr]] = {
    
        def lcnfH1(list: List[Expr], expr: Expr): List[Expr] = expr match {
            case And(p, q) => list ::: lcnfH1(List(), p) ::: lcnfH1(List(), q) case e => e :: list }
        def lcnfH2(list: List[Expr])(expr: Expr): List[Expr] = expr match {
            case Or(p, q) => list ::: lcnfH2(List())(p) ::: lcnfH2(List())(q) case e => e :: list }

        lcnfH1(List(), expr) map lcnfH2(List())_
    }
    
    def resolution(clist: List[List[Expr]]): Boolean = {
    
        def isAtom(expr: Expr) = expr match { case Atom(_) => true case _ => false }
        def removeNot(expr: Expr) = expr match { case Not(p) => p 
            case Const(false) => Const(true) case Const(true) => Const(false) }
        def splitClause(clause: List[Expr]) = {
            val (pos, neg) = clause partition (isAtom _) ; List(pos, neg map (removeNot _)) }
        def isTrueClause(clause: List[Expr]) = { val List(lits, nlits) = splitClause(clause)
            ((lits map ((e: Expr) => nlits exists ( _ == e))) :\ false)(_||_) }
        def isLitTrueClause(clause: List[Expr]) = clause match {
            case List(Const(true)) => true case List(Not(Const(false))) => true case _ => false }
        def simplifyClause(clause: List[Expr]) = if (isTrueClause(clause)) List(Const(true)) else clause
        def simplifyLcnf(clist: List[List[Expr]]) = {
            (clist map (simplifyClause _)) remove (isLitTrueClause _) }
        def negateLit(e: Expr) = e match { case Not(p) => p case p => Not(p) }

        def s2l[T](set: Set[T]) = set.toList ; def l2s[T](list: List[T]) = list.toSet
        
        def resolutionH2(tp4: (List[Expr], List[List[Expr]], List[List[Expr]], Boolean), lit: Expr) = {
            val (c, cs, d, b) = tp4 ; if (d == List()) (c, cs, d, b) else {
                val nlit = negateLit(lit) ; val c1 = c diff List(lit)
                val cs1 = (cs filter ((c: List[Expr]) => c exists (_ == nlit))) map (_ diff List(nlit))
                var d1 = cs1 map (_ ::: c1) ; val cs2h = ((cs map l2s) union (d1 map l2s)).distinct
                val cs2 = (cs2h map s2l) remove (isTrueClause _) ; d1 = cs2 diff cs
                val b1 = if (d1.isEmpty) false else (d1 exists (_ == List()) ) ; (c, cs2, d1, b1) 
            }
        }
        
        def resolutionH1(c: List[Expr], cs: List[List[Expr]], d: List[List[Expr]], b: Boolean): Boolean =
            (c, cs, d, b) match {
                case (_, List(), _, _) => false case (_, _, _, true) => true
                case (List(), c::cs, d, b) => resolutionH1(c, cs, d, b)
                case (c, cs, d, b) => { 
                    val (c1, cs1, d1, b1) = ((c, cs, d ++ List(List(Atom(""))), b) /: c)(resolutionH2)
                    resolutionH1(List(), cs1, d1, b1) } }
    
        clist match {
            case List(Const(false)::rest) => true 
            case _ => resolutionH1(List(), simplifyLcnf(clist) map 
                ((c: List[Expr]) => s2l(l2s(c))), List(), false) }        
    }
    
    def main(args: Array[String]) {
 
        var input = ""
 
        do  { input = readLine() ; if (input != null && input != "") {
            
            val output = parseAll(expr, input) match {
                case Failure(msg, _) => "failure: " + msg case Error(msg, _) => "error: " + msg
                case Success(parsed, _) => {
                    val nResRet = resolution(lcnf(cnf(simplify(nnf(simplify(Not(parsed)))))))
                    val pResRet = resolution(lcnf(cnf(simplify(nnf(simplify(parsed))))))
                    toStr(parsed, 0, 0) + "\n" + 
                        (if (nResRet) "true" else if (pResRet) "false" else "it depends")
                }
            }
            
            println(output) }
        } while (input != null && input != "")
    }
}