1use super::order_normalize::normalize_positive_order_atomic_fact;
2use crate::prelude::*;
3
4pub enum NumberCompareResult {
5 Less,
6 Equal,
7 Greater,
8}
9
10pub fn compare_normalized_number_str_to_zero(number_value: &str) -> NumberCompareResult {
12 compare_number_strings(number_value.trim(), "0")
13}
14
15fn parse_number_parts_for_comparison(number_value: &str) -> (bool, Vec<u8>, Vec<u8>) {
16 let trimmed_number_value = number_value.trim();
17 let (is_negative, magnitude_string) = if trimmed_number_value.starts_with('-') {
18 (true, trimmed_number_value[1..].trim())
19 } else {
20 (false, trimmed_number_value)
21 };
22
23 let (integer_part_string, fractional_part_string) = match magnitude_string.find('.') {
24 Some(dot_index) => (
25 &magnitude_string[..dot_index],
26 &magnitude_string[dot_index + 1..],
27 ),
28 None => (magnitude_string, ""),
29 };
30
31 let mut integer_digits: Vec<u8> = Vec::new();
32 if integer_part_string.is_empty() {
33 integer_digits.push(0);
34 } else {
35 for current_char in integer_part_string.chars() {
36 if current_char.is_ascii_digit() {
37 integer_digits.push(current_char as u8 - b'0');
38 }
39 }
40 if integer_digits.is_empty() {
41 integer_digits.push(0);
42 }
43 }
44
45 let mut fractional_digits: Vec<u8> = Vec::new();
46 for current_char in fractional_part_string.chars() {
47 if current_char.is_ascii_digit() {
48 fractional_digits.push(current_char as u8 - b'0');
49 }
50 }
51
52 (is_negative, integer_digits, fractional_digits)
53}
54
55fn digits_are_all_zero(digits: &[u8]) -> bool {
56 for digit in digits {
57 if *digit != 0 {
58 return false;
59 }
60 }
61 true
62}
63
64fn normalized_decimal_string_is_integer(number_value: &str) -> bool {
65 let (_, _integer_digits, fractional_digits) = parse_number_parts_for_comparison(number_value);
66 digits_are_all_zero(&fractional_digits)
67}
68
69fn normalized_decimal_string_is_even_integer(number_value: &str) -> bool {
70 if !normalized_decimal_string_is_integer(number_value) {
71 return false;
72 }
73 let (_is_negative, integer_digits, _fractional_digits) =
74 parse_number_parts_for_comparison(number_value);
75 let last_digit = integer_digits.last().copied().unwrap_or(0);
76 last_digit % 2 == 0
77}
78
79fn first_non_zero_integer_digit_index(integer_digits: &[u8]) -> usize {
80 let mut current_index = 0;
81 while current_index + 1 < integer_digits.len() && integer_digits[current_index] == 0 {
82 current_index += 1;
83 }
84 current_index
85}
86
87fn compare_non_negative_decimal_parts(
88 left_integer_digits: &[u8],
89 left_fractional_digits: &[u8],
90 right_integer_digits: &[u8],
91 right_fractional_digits: &[u8],
92) -> NumberCompareResult {
93 let left_integer_start_index = first_non_zero_integer_digit_index(left_integer_digits);
94 let right_integer_start_index = first_non_zero_integer_digit_index(right_integer_digits);
95
96 let left_effective_integer_len = left_integer_digits.len() - left_integer_start_index;
97 let right_effective_integer_len = right_integer_digits.len() - right_integer_start_index;
98 if left_effective_integer_len < right_effective_integer_len {
99 return NumberCompareResult::Less;
100 }
101 if left_effective_integer_len > right_effective_integer_len {
102 return NumberCompareResult::Greater;
103 }
104
105 let mut integer_index = 0;
106 while integer_index < left_effective_integer_len {
107 let left_digit = left_integer_digits[left_integer_start_index + integer_index];
108 let right_digit = right_integer_digits[right_integer_start_index + integer_index];
109 if left_digit < right_digit {
110 return NumberCompareResult::Less;
111 }
112 if left_digit > right_digit {
113 return NumberCompareResult::Greater;
114 }
115 integer_index += 1;
116 }
117
118 let fractional_compare_len = if left_fractional_digits.len() > right_fractional_digits.len() {
119 left_fractional_digits.len()
120 } else {
121 right_fractional_digits.len()
122 };
123 let mut fractional_index = 0;
124 while fractional_index < fractional_compare_len {
125 let left_digit = match left_fractional_digits.get(fractional_index) {
126 Some(digit) => *digit,
127 None => 0,
128 };
129 let right_digit = match right_fractional_digits.get(fractional_index) {
130 Some(digit) => *digit,
131 None => 0,
132 };
133 if left_digit < right_digit {
134 return NumberCompareResult::Less;
135 }
136 if left_digit > right_digit {
137 return NumberCompareResult::Greater;
138 }
139 fractional_index += 1;
140 }
141
142 NumberCompareResult::Equal
143}
144
145pub fn compare_number_strings(
146 left_number_value: &str,
147 right_number_value: &str,
148) -> NumberCompareResult {
149 let (left_is_negative, left_integer_digits, left_fractional_digits) =
150 parse_number_parts_for_comparison(left_number_value);
151 let (right_is_negative, right_integer_digits, right_fractional_digits) =
152 parse_number_parts_for_comparison(right_number_value);
153
154 let left_is_zero =
155 digits_are_all_zero(&left_integer_digits) && digits_are_all_zero(&left_fractional_digits);
156 let right_is_zero =
157 digits_are_all_zero(&right_integer_digits) && digits_are_all_zero(&right_fractional_digits);
158 if left_is_zero && right_is_zero {
159 return NumberCompareResult::Equal;
160 }
161
162 if left_is_negative && !left_is_zero && !right_is_negative {
163 return NumberCompareResult::Less;
164 }
165 if right_is_negative && !right_is_zero && !left_is_negative {
166 return NumberCompareResult::Greater;
167 }
168
169 let non_negative_compare_result = compare_non_negative_decimal_parts(
170 &left_integer_digits,
171 &left_fractional_digits,
172 &right_integer_digits,
173 &right_fractional_digits,
174 );
175 if left_is_negative && !left_is_zero && right_is_negative && !right_is_zero {
176 return match non_negative_compare_result {
177 NumberCompareResult::Less => NumberCompareResult::Greater,
178 NumberCompareResult::Equal => NumberCompareResult::Equal,
179 NumberCompareResult::Greater => NumberCompareResult::Less,
180 };
181 }
182
183 non_negative_compare_result
184}
185
186impl Runtime {
187 pub(crate) fn verify_non_equational_known_then_builtin_rules_only(
191 &mut self,
192 atomic_fact: &AtomicFact,
193 verify_state: &VerifyState,
194 ) -> Result<StmtResult, RuntimeError> {
195 let r = self.verify_non_equational_atomic_fact_with_known_atomic_facts(atomic_fact)?;
196 if r.is_true() {
197 return Ok(r);
198 }
199 self.verify_non_equational_atomic_fact_with_builtin_rules(atomic_fact, verify_state)
200 }
201
202 fn verify_zero_order_on_sub_expr(
203 &mut self,
204 zero: &Obj,
205 sub_expr: &Obj,
206 weak: bool,
207 line_file: &LineFile,
208 ) -> Result<StmtResult, RuntimeError> {
209 let fact: AtomicFact = if weak {
210 LessEqualFact::new(zero.clone(), sub_expr.clone(), line_file.clone()).into()
211 } else {
212 LessFact::new(zero.clone(), sub_expr.clone(), line_file.clone()).into()
213 };
214 let mut result = self.verify_non_equational_atomic_fact_with_known_atomic_facts(&fact)?;
215 if !result.is_true() {
216 result = self.verify_order_atomic_fact_numeric_builtin_only(&fact)?;
217 }
218 Ok(result)
219 }
220
221 fn try_verify_order_nonnegative_from_membership_in_n(
223 &mut self,
224 atomic_fact: &AtomicFact,
225 verify_state: &VerifyState,
226 ) -> Result<Option<StmtResult>, RuntimeError> {
227 let (n, line_file) = match atomic_fact {
228 AtomicFact::GreaterEqualFact(f) => {
229 let Some(z) = self.resolve_obj_to_number(&f.right) else {
230 return Ok(None);
231 };
232 if !matches!(
233 compare_normalized_number_str_to_zero(&z.normalized_value),
234 NumberCompareResult::Equal
235 ) {
236 return Ok(None);
237 }
238 (f.left.clone(), f.line_file.clone())
239 }
240 AtomicFact::LessEqualFact(f) => {
241 let Some(z) = self.resolve_obj_to_number(&f.left) else {
242 return Ok(None);
243 };
244 if !matches!(
245 compare_normalized_number_str_to_zero(&z.normalized_value),
246 NumberCompareResult::Equal
247 ) {
248 return Ok(None);
249 }
250 (f.right.clone(), f.line_file.clone())
251 }
252 _ => return Ok(None),
253 };
254 let in_n: AtomicFact = InFact::new(n, StandardSet::N.into(), line_file.clone()).into();
255 if self
256 .verify_non_equational_known_then_builtin_rules_only(&in_n, verify_state)?
257 .is_true()
258 {
259 return Ok(Some(StmtResult::FactualStmtSuccess(
260 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
261 atomic_fact.clone().into(),
262 "n >= 0 from n $in N".to_string(),
263 Vec::new(),
264 ),
265 )));
266 }
267 Ok(None)
268 }
269
270 fn verify_zero_le_abs_builtin_rule(
271 &mut self,
272 atomic_fact: &AtomicFact,
273 ) -> Result<Option<StmtResult>, RuntimeError> {
274 let Some(norm) = normalize_positive_order_atomic_fact(atomic_fact) else {
275 return Ok(None);
276 };
277 let AtomicFact::LessEqualFact(f) = &norm else {
278 return Ok(None);
279 };
280 if f.left.to_string() != "0" {
281 return Ok(None);
282 }
283 if !matches!(&f.right, Obj::Abs(_)) {
284 return Ok(None);
285 }
286 Ok(Some(StmtResult::FactualStmtSuccess(
287 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
288 atomic_fact.clone().into(),
289 "0 <= abs(x) for x in R".to_string(),
290 Vec::new(),
291 ),
292 )))
293 }
294
295 fn try_verify_order_opposite_sign_mul_minus_one(
298 &mut self,
299 atomic_fact: &AtomicFact,
300 verify_state: &VerifyState,
301 ) -> Result<Option<StmtResult>, RuntimeError> {
302 let z: Obj = Number::new("0".to_string()).into();
303 let success = |msg: &'static str| {
304 Ok(Some(StmtResult::FactualStmtSuccess(
305 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
306 atomic_fact.clone().into(),
307 msg.to_string(),
308 Vec::new(),
309 ),
310 )))
311 };
312 match atomic_fact {
313 AtomicFact::GreaterEqualFact(f) if self.obj_is_resolved_zero(&f.right) => {
314 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.left) {
315 let le: AtomicFact =
316 LessEqualFact::new(x.clone(), z.clone(), f.line_file.clone()).into();
317 if self
318 .verify_non_equational_known_then_builtin_rules_only(&le, verify_state)?
319 .is_true()
320 {
321 return success("order: (-1)*x >= 0 from x <= 0");
322 }
323 let lt: AtomicFact = LessFact::new(x, z.clone(), f.line_file.clone()).into();
324 if self
325 .verify_non_equational_known_then_builtin_rules_only(<, verify_state)?
326 .is_true()
327 {
328 return success("order: (-1)*x >= 0 from x < 0");
329 }
330 }
331 Ok(None)
332 }
333 AtomicFact::GreaterFact(f) if self.obj_is_resolved_zero(&f.right) => {
334 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.left) {
335 let lt: AtomicFact = LessFact::new(x, z.clone(), f.line_file.clone()).into();
336 if self
337 .verify_non_equational_known_then_builtin_rules_only(<, verify_state)?
338 .is_true()
339 {
340 return success("order: (-1)*x > 0 from x < 0");
341 }
342 }
343 Ok(None)
344 }
345 AtomicFact::LessEqualFact(f) if self.obj_is_resolved_zero(&f.right) => {
346 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.left) {
347 let ge: AtomicFact =
348 GreaterEqualFact::new(x.clone(), z.clone(), f.line_file.clone()).into();
349 if self
350 .verify_non_equational_known_then_builtin_rules_only(&ge, verify_state)?
351 .is_true()
352 {
353 return success("order: (-1)*x <= 0 from x >= 0");
354 }
355 let gt: AtomicFact = GreaterFact::new(x, z.clone(), f.line_file.clone()).into();
356 if self
357 .verify_non_equational_known_then_builtin_rules_only(>, verify_state)?
358 .is_true()
359 {
360 return success("order: (-1)*x <= 0 from x > 0");
361 }
362 }
363 Ok(None)
364 }
365 AtomicFact::LessFact(f) if self.obj_is_resolved_zero(&f.right) => {
366 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.left) {
367 let gt: AtomicFact = GreaterFact::new(x, z.clone(), f.line_file.clone()).into();
368 if self
369 .verify_non_equational_known_then_builtin_rules_only(>, verify_state)?
370 .is_true()
371 {
372 return success("order: (-1)*x < 0 from x > 0");
373 }
374 }
375 Ok(None)
376 }
377 AtomicFact::LessEqualFact(f) if self.obj_is_resolved_zero(&f.left) => {
378 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.right) {
379 let le: AtomicFact =
380 LessEqualFact::new(x.clone(), z.clone(), f.line_file.clone()).into();
381 if self
382 .verify_non_equational_known_then_builtin_rules_only(&le, verify_state)?
383 .is_true()
384 {
385 return success("order: 0 <= (-1)*x from x <= 0");
386 }
387 let lt: AtomicFact = LessFact::new(x, z.clone(), f.line_file.clone()).into();
388 if self
389 .verify_non_equational_known_then_builtin_rules_only(<, verify_state)?
390 .is_true()
391 {
392 return success("order: 0 <= (-1)*x from x < 0");
393 }
394 }
395 Ok(None)
396 }
397 AtomicFact::LessFact(f) if self.obj_is_resolved_zero(&f.left) => {
398 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.right) {
399 let lt: AtomicFact = LessFact::new(x, z.clone(), f.line_file.clone()).into();
400 if self
401 .verify_non_equational_known_then_builtin_rules_only(<, verify_state)?
402 .is_true()
403 {
404 return success("order: 0 < (-1)*x from x < 0");
405 }
406 }
407 Ok(None)
408 }
409 AtomicFact::GreaterEqualFact(f) if self.obj_is_resolved_zero(&f.left) => {
410 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.right) {
411 let ge: AtomicFact =
412 GreaterEqualFact::new(x.clone(), z.clone(), f.line_file.clone()).into();
413 if self
414 .verify_non_equational_known_then_builtin_rules_only(&ge, verify_state)?
415 .is_true()
416 {
417 return success("order: 0 >= (-1)*x from x >= 0");
418 }
419 let gt: AtomicFact = GreaterFact::new(x, z.clone(), f.line_file.clone()).into();
420 if self
421 .verify_non_equational_known_then_builtin_rules_only(>, verify_state)?
422 .is_true()
423 {
424 return success("order: 0 >= (-1)*x from x > 0");
425 }
426 }
427 Ok(None)
428 }
429 AtomicFact::GreaterFact(f) if self.obj_is_resolved_zero(&f.left) => {
430 if let Some(x) = self.peel_mul_by_literal_neg_one(&f.right) {
431 let gt: AtomicFact = GreaterFact::new(x, z.clone(), f.line_file.clone()).into();
432 if self
433 .verify_non_equational_known_then_builtin_rules_only(>, verify_state)?
434 .is_true()
435 {
436 return success("order: 0 > (-1)*x from x > 0");
437 }
438 }
439 Ok(None)
440 }
441 _ => Ok(None),
442 }
443 }
444
445 pub fn verify_order_atomic_fact_numeric_builtin_only(
460 &mut self,
461 atomic_fact: &AtomicFact,
462 ) -> Result<StmtResult, RuntimeError> {
463 let vs = VerifyState::new(0, true);
464 if let Some(result) =
465 self.try_verify_order_nonnegative_from_membership_in_n(atomic_fact, &vs)?
466 {
467 return Ok(result);
468 }
469 if let Some(result) = self.try_verify_order_opposite_sign_mul_minus_one(atomic_fact, &vs)? {
470 return Ok(result);
471 }
472 if let Some(result) = self.verify_order_from_known_negated_complement(atomic_fact)? {
473 return Ok(result);
474 }
475 if let Some(result) = self.verify_negated_order_from_known_equivalent_order(atomic_fact)? {
476 return Ok(result);
477 }
478 if let Some(result) = self.verify_order_algebra_structural_builtin_rule(atomic_fact)? {
479 return Ok(result);
480 }
481 if let Some(result) = self.verify_zero_le_abs_builtin_rule(atomic_fact)? {
482 return Ok(result);
483 }
484 if let Some(result) =
485 self.verify_zero_order_on_sub_from_two_sided_order_builtin_rule(atomic_fact)?
486 {
487 return Ok(result);
488 }
489 if let Some(result) =
490 self.verify_zero_le_add_from_known_atomic_facts_builtin_rule(atomic_fact)?
491 {
492 return Ok(result);
493 }
494 if let Some(result) =
495 self.verify_zero_lt_add_from_known_atomic_facts_builtin_rule(atomic_fact)?
496 {
497 return Ok(result);
498 }
499 if let Some(result) = self.verify_zero_le_even_integer_pow_builtin_rule(atomic_fact)? {
500 return Ok(result);
501 }
502 if let Some(result) =
503 self.verify_zero_lt_even_integer_pow_from_base_nonzero_builtin_rule(atomic_fact)?
504 {
505 return Ok(result);
506 }
507 if let Some(result) =
508 self.verify_zero_lt_pow_from_positive_base_real_exp_builtin_rule(atomic_fact)?
509 {
510 return Ok(result);
511 }
512 if let Some(result) =
513 self.verify_zero_le_pow_from_positive_base_real_exp_builtin_rule(atomic_fact)?
514 {
515 return Ok(result);
516 }
517 if let Some(result) =
518 self.verify_zero_le_pow_integer_exponent_from_nonneg_base_builtin_rule(atomic_fact)?
519 {
520 return Ok(result);
521 }
522 if let Some(result) =
523 self.verify_zero_le_mul_from_known_atomic_facts_builtin_rule(atomic_fact)?
524 {
525 return Ok(result);
526 }
527 if let Some(result) =
528 self.verify_zero_lt_mul_from_known_atomic_facts_builtin_rule(atomic_fact)?
529 {
530 return Ok(result);
531 }
532 if let Some(result) =
533 self.verify_zero_le_div_from_known_atomic_facts_builtin_rule(atomic_fact)?
534 {
535 return Ok(result);
536 }
537 if let Some(result) =
538 self.verify_zero_lt_div_from_known_atomic_facts_builtin_rule(atomic_fact)?
539 {
540 return Ok(result);
541 }
542
543 if let AtomicFact::LessEqualFact(less_equal_fact) = atomic_fact {
544 if less_equal_fact.left.to_string() == less_equal_fact.right.to_string() {
545 return Ok(StmtResult::FactualStmtSuccess(
546 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
547 less_equal_fact.clone().into(),
548 "less_equal_fact_equal".to_string(),
549 Vec::new(),
550 ),
551 ));
552 }
553 let strict_fact: Fact = LessFact::new(
554 less_equal_fact.left.clone(),
555 less_equal_fact.right.clone(),
556 less_equal_fact.line_file.clone(),
557 )
558 .into();
559 let strict_key = strict_fact.to_string();
560 let (cache_ok, cache_line_file) = self.cache_known_facts_contains(&strict_key);
561 if cache_ok {
562 return Ok(StmtResult::FactualStmtSuccess(
563 FactualStmtSuccess::new_with_verified_by_known_fact_source_recording_facts(
564 less_equal_fact.clone().into(),
565 strict_key,
566 Some(strict_fact),
567 Some(cache_line_file),
568 Vec::new(),
569 ),
570 ));
571 }
572 }
573 if let AtomicFact::GreaterEqualFact(greater_equal_fact) = atomic_fact {
574 if greater_equal_fact.left.to_string() == greater_equal_fact.right.to_string() {
575 return Ok(StmtResult::FactualStmtSuccess(
576 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
577 greater_equal_fact.clone().into(),
578 "greater_equal_fact_equal".to_string(),
579 Vec::new(),
580 ),
581 ));
582 }
583 }
584 if let Some(true) = self.verify_number_comparison_builtin_rule(atomic_fact) {
585 Ok(StmtResult::FactualStmtSuccess(
586 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
587 atomic_fact.clone().into(),
588 "number comparison".to_string(),
589 Vec::new(),
590 ),
591 ))
592 } else {
593 Ok(StmtResult::StmtUnknown(StmtUnknown::new()))
594 }
595 }
596
597 fn verify_order_from_known_negated_complement(
599 &mut self,
600 atomic_fact: &AtomicFact,
601 ) -> Result<Option<StmtResult>, RuntimeError> {
602 let negated: Option<AtomicFact> = match atomic_fact {
603 AtomicFact::GreaterFact(f) => {
604 Some(NotLessEqualFact::new(f.left.clone(), f.right.clone(), f.line_file.clone()).into())
605 }
606 AtomicFact::LessFact(f) => Some(
607 NotGreaterEqualFact::new(f.left.clone(), f.right.clone(), f.line_file.clone()).into(),
608 ),
609 AtomicFact::GreaterEqualFact(f) => {
610 Some(NotLessFact::new(f.left.clone(), f.right.clone(), f.line_file.clone()).into())
611 }
612 AtomicFact::LessEqualFact(f) => {
613 Some(NotGreaterFact::new(f.left.clone(), f.right.clone(), f.line_file.clone()).into())
614 }
615 _ => None,
616 };
617 let Some(neg) = negated else {
618 return Ok(None);
619 };
620 let sub = self.verify_non_equational_atomic_fact_with_known_atomic_facts(&neg)?;
621 if sub.is_true() {
622 return Ok(Some(
623 FactualStmtSuccess::new_with_verified_by_builtin_rules(
624 atomic_fact.clone().into(),
625 InferResult::new(),
626 "order_from_known_negated_complement".to_string(),
627 vec![sub],
628 )
629 .into(),
630 ));
631 }
632 Ok(None)
633 }
634
635 fn verify_negated_order_from_known_equivalent_order(
637 &mut self,
638 atomic_fact: &AtomicFact,
639 ) -> Result<Option<StmtResult>, RuntimeError> {
640 let candidates: Vec<AtomicFact> = match atomic_fact {
641 AtomicFact::NotLessFact(f) => {
642 let lf = f.line_file.clone();
643 vec![
644 LessEqualFact::new(f.right.clone(), f.left.clone(), lf.clone()).into(),
645 GreaterEqualFact::new(f.left.clone(), f.right.clone(), lf).into(),
646 ]
647 }
648 AtomicFact::NotGreaterFact(f) => {
649 let lf = f.line_file.clone();
650 vec![
651 LessEqualFact::new(f.left.clone(), f.right.clone(), lf.clone()).into(),
652 GreaterEqualFact::new(f.right.clone(), f.left.clone(), lf).into(),
653 ]
654 }
655 AtomicFact::NotLessEqualFact(f) => {
656 let lf = f.line_file.clone();
657 vec![
658 LessFact::new(f.right.clone(), f.left.clone(), lf.clone()).into(),
659 GreaterFact::new(f.left.clone(), f.right.clone(), lf).into(),
660 ]
661 }
662 AtomicFact::NotGreaterEqualFact(f) => {
663 let lf = f.line_file.clone();
664 vec![
665 LessFact::new(f.left.clone(), f.right.clone(), lf.clone()).into(),
666 GreaterFact::new(f.right.clone(), f.left.clone(), lf).into(),
667 ]
668 }
669 _ => return Ok(None),
670 };
671 for candidate in candidates {
672 let sub = self.verify_non_equational_atomic_fact_with_known_atomic_facts(&candidate)?;
673 if sub.is_true() {
674 return Ok(Some(
675 FactualStmtSuccess::new_with_verified_by_builtin_rules(
676 atomic_fact.clone().into(),
677 InferResult::new(),
678 "negated_order_from_known_equivalent_order".to_string(),
679 vec![sub],
680 )
681 .into(),
682 ));
683 }
684 }
685 Ok(None)
686 }
687
688 fn verify_zero_order_on_sub_from_two_sided_order_builtin_rule(
690 &mut self,
691 atomic_fact: &AtomicFact,
692 ) -> Result<Option<StmtResult>, RuntimeError> {
693 let Some(norm) = normalize_positive_order_atomic_fact(atomic_fact) else {
694 return Ok(None);
695 };
696 match &norm {
697 AtomicFact::LessEqualFact(f) if f.left.to_string() == "0" => {
698 let Obj::Sub(sub) = &f.right else {
699 return Ok(None);
700 };
701 let derived: AtomicFact = LessEqualFact::new(
702 sub.right.as_ref().clone(),
703 sub.left.as_ref().clone(),
704 f.line_file.clone(),
705 )
706 .into();
707 let mut result =
708 self.verify_non_equational_atomic_fact_with_known_atomic_facts(&derived)?;
709 if !result.is_true() {
710 result = self.verify_order_atomic_fact_numeric_builtin_only(&derived)?;
711 }
712 if result.is_true() {
713 Ok(Some(StmtResult::FactualStmtSuccess(
714 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
715 atomic_fact.clone().into(),
716 "0 <= u - v from v <= u".to_string(),
717 vec![result],
718 ),
719 )))
720 } else {
721 Ok(None)
722 }
723 }
724 AtomicFact::LessFact(f) if f.left.to_string() == "0" => {
725 let Obj::Sub(sub) = &f.right else {
726 return Ok(None);
727 };
728 let derived: AtomicFact = LessFact::new(
729 sub.right.as_ref().clone(),
730 sub.left.as_ref().clone(),
731 f.line_file.clone(),
732 )
733 .into();
734 let mut result =
735 self.verify_non_equational_atomic_fact_with_known_atomic_facts(&derived)?;
736 if !result.is_true() {
737 result = self.verify_order_atomic_fact_numeric_builtin_only(&derived)?;
738 }
739 if result.is_true() {
740 Ok(Some(StmtResult::FactualStmtSuccess(
741 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
742 atomic_fact.clone().into(),
743 "0 < u - v from v < u".to_string(),
744 vec![result],
745 ),
746 )))
747 } else {
748 Ok(None)
749 }
750 }
751 _ => Ok(None),
752 }
753 }
754
755 fn verify_zero_le_add_from_known_atomic_facts_builtin_rule(
756 &mut self,
757 atomic_fact: &AtomicFact,
758 ) -> Result<Option<StmtResult>, RuntimeError> {
759 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
760 return Ok(None);
761 };
762 let AtomicFact::LessEqualFact(less_equal_fact) = normalized_fact else {
763 return Ok(None);
764 };
765 if less_equal_fact.left.to_string() != "0" {
766 return Ok(None);
767 }
768 let Obj::Add(add_obj) = &less_equal_fact.right else {
769 return Ok(None);
770 };
771
772 let zero = &less_equal_fact.left;
773 let line_file = &less_equal_fact.line_file;
774 let left_verify_result =
775 self.verify_zero_order_on_sub_expr(zero, add_obj.left.as_ref(), true, line_file)?;
776 if !left_verify_result.is_true() {
777 return Ok(None);
778 }
779 let right_verify_result =
780 self.verify_zero_order_on_sub_expr(zero, add_obj.right.as_ref(), true, line_file)?;
781 if !right_verify_result.is_true() {
782 return Ok(None);
783 }
784
785 Ok(Some(StmtResult::FactualStmtSuccess(
786 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
787 atomic_fact.clone().into(),
788 "0 <= a + b from known atomic facts 0 <= a and 0 <= b".to_string(),
789 vec![left_verify_result, right_verify_result],
790 ),
791 )))
792 }
793
794 fn verify_zero_lt_add_from_known_atomic_facts_builtin_rule(
795 &mut self,
796 atomic_fact: &AtomicFact,
797 ) -> Result<Option<StmtResult>, RuntimeError> {
798 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
799 return Ok(None);
800 };
801 let AtomicFact::LessFact(less_fact) = normalized_fact else {
802 return Ok(None);
803 };
804 if less_fact.left.to_string() != "0" {
805 return Ok(None);
806 }
807 let Obj::Add(add_obj) = &less_fact.right else {
808 return Ok(None);
809 };
810
811 let zero = &less_fact.left;
812 let line_file = &less_fact.line_file;
813
814 let strict_then_weak = |this: &mut Self| -> Result<Option<StmtResult>, RuntimeError> {
815 let left_result =
816 this.verify_zero_order_on_sub_expr(zero, add_obj.left.as_ref(), false, line_file)?;
817 if !left_result.is_true() {
818 return Ok(None);
819 }
820 let right_result =
821 this.verify_zero_order_on_sub_expr(zero, add_obj.right.as_ref(), true, line_file)?;
822 if !right_result.is_true() {
823 return Ok(None);
824 }
825 Ok(Some(StmtResult::FactualStmtSuccess(
826 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
827 atomic_fact.clone().into(),
828 "0 < a + b from (0 < a and 0 <= b)".to_string(),
829 vec![left_result, right_result],
830 ),
831 )))
832 };
833 let weak_then_strict = |this: &mut Self| -> Result<Option<StmtResult>, RuntimeError> {
834 let left_result =
835 this.verify_zero_order_on_sub_expr(zero, add_obj.left.as_ref(), true, line_file)?;
836 if !left_result.is_true() {
837 return Ok(None);
838 }
839 let right_result =
840 this.verify_zero_order_on_sub_expr(zero, add_obj.right.as_ref(), false, line_file)?;
841 if !right_result.is_true() {
842 return Ok(None);
843 }
844 Ok(Some(StmtResult::FactualStmtSuccess(
845 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
846 atomic_fact.clone().into(),
847 "0 < a + b from (0 <= a and 0 < b)".to_string(),
848 vec![left_result, right_result],
849 ),
850 )))
851 };
852
853 if let Some(success) = strict_then_weak(self)? {
854 return Ok(Some(success));
855 }
856 weak_then_strict(self)
857 }
858
859 fn verify_zero_le_even_integer_pow_builtin_rule(
860 &mut self,
861 atomic_fact: &AtomicFact,
862 ) -> Result<Option<StmtResult>, RuntimeError> {
863 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
864 return Ok(None);
865 };
866 let AtomicFact::LessEqualFact(less_equal_fact) = normalized_fact else {
867 return Ok(None);
868 };
869 if less_equal_fact.left.to_string() != "0" {
870 return Ok(None);
871 }
872 let right = &less_equal_fact.right;
873 let is_equal_factors_mul = match right {
874 Obj::Mul(mul_obj) => mul_obj.left.to_string() == mul_obj.right.to_string(),
875 _ => false,
876 };
877 let is_even_pow = match right {
878 Obj::Pow(pow_obj) => match pow_obj.exponent.as_ref() {
879 Obj::Number(n) => normalized_decimal_string_is_even_integer(&n.normalized_value),
880 _ => false,
881 },
882 _ => false,
883 };
884 if !is_equal_factors_mul && !is_even_pow {
885 return Ok(None);
886 }
887 let msg = if is_equal_factors_mul {
888 "0 <= a * a from even integer exponent (here 2) (forall a R)".to_string()
889 } else {
890 "0 <= a^n for even integer n (forall a R)".to_string()
891 };
892 Ok(Some(StmtResult::FactualStmtSuccess(
893 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
894 atomic_fact.clone().into(),
895 msg,
896 Vec::new(),
897 ),
898 )))
899 }
900
901 fn verify_zero_lt_even_integer_pow_from_base_nonzero_builtin_rule(
902 &mut self,
903 atomic_fact: &AtomicFact,
904 ) -> Result<Option<StmtResult>, RuntimeError> {
905 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
906 return Ok(None);
907 };
908 let AtomicFact::LessFact(less_fact) = normalized_fact else {
909 return Ok(None);
910 };
911 if less_fact.left.to_string() != "0" {
912 return Ok(None);
913 }
914 let Obj::Pow(pow_obj) = &less_fact.right else {
915 return Ok(None);
916 };
917 let Obj::Number(exp_num) = pow_obj.exponent.as_ref() else {
918 return Ok(None);
919 };
920 if !normalized_decimal_string_is_even_integer(&exp_num.normalized_value) {
921 return Ok(None);
922 }
923
924 let line_file = less_fact.line_file.clone();
925 let base = pow_obj.base.as_ref().clone();
926 let zero_obj: Obj = Number::new("0".to_string()).into();
927 let base_neq_zero: AtomicFact = NotEqualFact::new(base, zero_obj, line_file.clone()).into();
928
929 let neq_result = self.verify_non_equational_known_then_builtin_rules_only(
930 &base_neq_zero,
931 &VerifyState::new(0, true),
932 )?;
933 if !neq_result.is_true() {
934 return Ok(None);
935 }
936
937 Ok(Some(StmtResult::FactualStmtSuccess(
938 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
939 atomic_fact.clone().into(),
940 "0 < a^n for even integer n from a != 0".to_string(),
941 vec![neq_result],
942 ),
943 )))
944 }
945
946 fn verify_zero_lt_pow_from_positive_base_real_exp_builtin_rule(
948 &mut self,
949 atomic_fact: &AtomicFact,
950 ) -> Result<Option<StmtResult>, RuntimeError> {
951 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
952 return Ok(None);
953 };
954 let AtomicFact::LessFact(less_fact) = normalized_fact else {
955 return Ok(None);
956 };
957 if less_fact.left.to_string() != "0" {
958 return Ok(None);
959 }
960 let Obj::Pow(pow_obj) = &less_fact.right else {
961 return Ok(None);
962 };
963 let zero = &less_fact.left;
964 let line_file = &less_fact.line_file;
965 let base = pow_obj.base.as_ref();
966 let base_result = self.verify_zero_order_on_sub_expr(zero, base, false, line_file)?;
967 if !base_result.is_true() {
968 return Ok(None);
969 }
970 let in_r: AtomicFact = InFact::new(
971 (*pow_obj.exponent).clone(),
972 StandardSet::R.into(),
973 line_file.clone(),
974 )
975 .into();
976 let in_r_result = self.verify_non_equational_known_then_builtin_rules_only(
977 &in_r,
978 &VerifyState::new(0, true),
979 )?;
980 if !in_r_result.is_true() {
981 return Ok(None);
982 }
983 Ok(Some(StmtResult::FactualStmtSuccess(
984 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
985 atomic_fact.clone().into(),
986 "0 < a^b from 0 < a and b in R".to_string(),
987 vec![base_result, in_r_result],
988 ),
989 )))
990 }
991
992 fn verify_zero_le_pow_from_positive_base_real_exp_builtin_rule(
995 &mut self,
996 atomic_fact: &AtomicFact,
997 ) -> Result<Option<StmtResult>, RuntimeError> {
998 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
999 return Ok(None);
1000 };
1001 let AtomicFact::LessEqualFact(less_equal_fact) = normalized_fact else {
1002 return Ok(None);
1003 };
1004 if less_equal_fact.left.to_string() != "0" {
1005 return Ok(None);
1006 }
1007 let Obj::Pow(pow_obj) = &less_equal_fact.right else {
1008 return Ok(None);
1009 };
1010 let zero = &less_equal_fact.left;
1011 let line_file = &less_equal_fact.line_file;
1012 let base = pow_obj.base.as_ref();
1013 let base_result = self.verify_zero_order_on_sub_expr(zero, base, false, line_file)?;
1014 if !base_result.is_true() {
1015 return Ok(None);
1016 }
1017 let in_r: AtomicFact = InFact::new(
1018 (*pow_obj.exponent).clone(),
1019 StandardSet::R.into(),
1020 line_file.clone(),
1021 )
1022 .into();
1023 let in_r_result = self.verify_non_equational_known_then_builtin_rules_only(
1024 &in_r,
1025 &VerifyState::new(0, true),
1026 )?;
1027 if !in_r_result.is_true() {
1028 return Ok(None);
1029 }
1030 Ok(Some(StmtResult::FactualStmtSuccess(
1031 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
1032 atomic_fact.clone().into(),
1033 "0 <= a^b from 0 < a and b in R".to_string(),
1034 vec![base_result, in_r_result],
1035 ),
1036 )))
1037 }
1038
1039 fn verify_zero_le_pow_integer_exponent_from_nonneg_base_builtin_rule(
1040 &mut self,
1041 atomic_fact: &AtomicFact,
1042 ) -> Result<Option<StmtResult>, RuntimeError> {
1043 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
1044 return Ok(None);
1045 };
1046 let AtomicFact::LessEqualFact(less_equal_fact) = normalized_fact else {
1047 return Ok(None);
1048 };
1049 if less_equal_fact.left.to_string() != "0" {
1050 return Ok(None);
1051 }
1052 let Obj::Pow(pow_obj) = &less_equal_fact.right else {
1053 return Ok(None);
1054 };
1055 let Obj::Number(exp_num) = pow_obj.exponent.as_ref() else {
1056 return Ok(None);
1057 };
1058 if !normalized_decimal_string_is_integer(&exp_num.normalized_value) {
1059 return Ok(None);
1060 }
1061
1062 let zero = &less_equal_fact.left;
1063 let line_file = &less_equal_fact.line_file;
1064 let base = pow_obj.base.as_ref();
1065
1066 let exponent_vs_zero = compare_normalized_number_str_to_zero(&exp_num.normalized_value);
1067 let base_result = match exponent_vs_zero {
1068 NumberCompareResult::Less => {
1069 self.verify_zero_order_on_sub_expr(zero, base, false, line_file)?
1070 }
1071 NumberCompareResult::Equal | NumberCompareResult::Greater => {
1072 self.verify_zero_order_on_sub_expr(zero, base, true, line_file)?
1073 }
1074 };
1075 if !base_result.is_true() {
1076 return Ok(None);
1077 }
1078
1079 let msg = match exponent_vs_zero {
1080 NumberCompareResult::Less => "0 <= a^n from 0 < a and integer n < 0".to_string(),
1081 _ => "0 <= a^n from 0 <= a and integer n".to_string(),
1082 };
1083
1084 Ok(Some(StmtResult::FactualStmtSuccess(
1085 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
1086 atomic_fact.clone().into(),
1087 msg,
1088 vec![base_result],
1089 ),
1090 )))
1091 }
1092
1093 fn verify_zero_le_mul_from_known_atomic_facts_builtin_rule(
1094 &mut self,
1095 atomic_fact: &AtomicFact,
1096 ) -> Result<Option<StmtResult>, RuntimeError> {
1097 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
1098 return Ok(None);
1099 };
1100 let AtomicFact::LessEqualFact(less_equal_fact) = normalized_fact else {
1101 return Ok(None);
1102 };
1103 if less_equal_fact.left.to_string() != "0" {
1104 return Ok(None);
1105 }
1106 let Obj::Mul(mul_obj) = &less_equal_fact.right else {
1107 return Ok(None);
1108 };
1109
1110 let zero = &less_equal_fact.left;
1111 let line_file = &less_equal_fact.line_file;
1112 let left_verify_result =
1113 self.verify_zero_order_on_sub_expr(zero, mul_obj.left.as_ref(), true, line_file)?;
1114 if !left_verify_result.is_true() {
1115 return Ok(None);
1116 }
1117 let right_verify_result =
1118 self.verify_zero_order_on_sub_expr(zero, mul_obj.right.as_ref(), true, line_file)?;
1119 if !right_verify_result.is_true() {
1120 return Ok(None);
1121 }
1122
1123 Ok(Some(StmtResult::FactualStmtSuccess(
1124 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
1125 atomic_fact.clone().into(),
1126 "0 <= a * b from 0 <= a and 0 <= b".to_string(),
1127 vec![left_verify_result, right_verify_result],
1128 ),
1129 )))
1130 }
1131
1132 fn verify_zero_lt_mul_from_known_atomic_facts_builtin_rule(
1133 &mut self,
1134 atomic_fact: &AtomicFact,
1135 ) -> Result<Option<StmtResult>, RuntimeError> {
1136 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
1137 return Ok(None);
1138 };
1139 let AtomicFact::LessFact(less_fact) = normalized_fact else {
1140 return Ok(None);
1141 };
1142 if less_fact.left.to_string() != "0" {
1143 return Ok(None);
1144 }
1145 let Obj::Mul(mul_obj) = &less_fact.right else {
1146 return Ok(None);
1147 };
1148
1149 let zero = &less_fact.left;
1150 let line_file = &less_fact.line_file;
1151 let left_verify_result =
1152 self.verify_zero_order_on_sub_expr(zero, mul_obj.left.as_ref(), false, line_file)?;
1153 if !left_verify_result.is_true() {
1154 return Ok(None);
1155 }
1156 let right_verify_result =
1157 self.verify_zero_order_on_sub_expr(zero, mul_obj.right.as_ref(), false, line_file)?;
1158 if !right_verify_result.is_true() {
1159 return Ok(None);
1160 }
1161
1162 Ok(Some(StmtResult::FactualStmtSuccess(
1163 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
1164 atomic_fact.clone().into(),
1165 "0 < a * b from 0 < a and 0 < b".to_string(),
1166 vec![left_verify_result, right_verify_result],
1167 ),
1168 )))
1169 }
1170
1171 fn verify_zero_le_div_from_known_atomic_facts_builtin_rule(
1172 &mut self,
1173 atomic_fact: &AtomicFact,
1174 ) -> Result<Option<StmtResult>, RuntimeError> {
1175 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
1176 return Ok(None);
1177 };
1178 let AtomicFact::LessEqualFact(less_equal_fact) = normalized_fact else {
1179 return Ok(None);
1180 };
1181 if less_equal_fact.left.to_string() != "0" {
1182 return Ok(None);
1183 }
1184 let Obj::Div(div_obj) = &less_equal_fact.right else {
1185 return Ok(None);
1186 };
1187
1188 let zero = &less_equal_fact.left;
1189 let line_file = &less_equal_fact.line_file;
1190 let numer_result =
1191 self.verify_zero_order_on_sub_expr(zero, div_obj.left.as_ref(), true, line_file)?;
1192 if !numer_result.is_true() {
1193 return Ok(None);
1194 }
1195 let denom_result =
1196 self.verify_zero_order_on_sub_expr(zero, div_obj.right.as_ref(), false, line_file)?;
1197 if !denom_result.is_true() {
1198 return Ok(None);
1199 }
1200
1201 Ok(Some(StmtResult::FactualStmtSuccess(
1202 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
1203 atomic_fact.clone().into(),
1204 "0 <= a / b from 0 <= a and 0 < b".to_string(),
1205 vec![numer_result, denom_result],
1206 ),
1207 )))
1208 }
1209
1210 fn verify_zero_lt_div_from_known_atomic_facts_builtin_rule(
1211 &mut self,
1212 atomic_fact: &AtomicFact,
1213 ) -> Result<Option<StmtResult>, RuntimeError> {
1214 let Some(normalized_fact) = normalize_positive_order_atomic_fact(atomic_fact) else {
1215 return Ok(None);
1216 };
1217 let AtomicFact::LessFact(less_fact) = normalized_fact else {
1218 return Ok(None);
1219 };
1220 if less_fact.left.to_string() != "0" {
1221 return Ok(None);
1222 }
1223 let Obj::Div(div_obj) = &less_fact.right else {
1224 return Ok(None);
1225 };
1226
1227 let zero = &less_fact.left;
1228 let line_file = &less_fact.line_file;
1229 let numer_result =
1230 self.verify_zero_order_on_sub_expr(zero, div_obj.left.as_ref(), false, line_file)?;
1231 if !numer_result.is_true() {
1232 return Ok(None);
1233 }
1234 let denom_result =
1235 self.verify_zero_order_on_sub_expr(zero, div_obj.right.as_ref(), false, line_file)?;
1236 if !denom_result.is_true() {
1237 return Ok(None);
1238 }
1239
1240 Ok(Some(StmtResult::FactualStmtSuccess(
1241 FactualStmtSuccess::new_with_verified_by_builtin_rules_recording_stmt(
1242 atomic_fact.clone().into(),
1243 "0 < a / b from 0 < a and 0 < b".to_string(),
1244 vec![numer_result, denom_result],
1245 ),
1246 )))
1247 }
1248
1249 pub fn verify_number_comparison_builtin_rule(&self, atomic_fact: &AtomicFact) -> Option<bool> {
1250 let normalized = normalize_positive_order_atomic_fact(atomic_fact)?;
1251 match normalized {
1252 AtomicFact::LessFact(less_fact) => {
1253 if let Some(calculated_number_string_pair) =
1254 self.calculate_obj_pair_to_number_strings(&less_fact.left, &less_fact.right)
1255 {
1256 return Some(matches!(
1257 compare_number_strings(
1258 &calculated_number_string_pair.0,
1259 &calculated_number_string_pair.1
1260 ),
1261 NumberCompareResult::Less
1262 ));
1263 }
1264 self.try_verify_numeric_order_via_div_elimination(
1265 &less_fact.left,
1266 &less_fact.right,
1267 false,
1268 )
1269 }
1270 AtomicFact::LessEqualFact(less_equal_fact) => {
1271 if let Some(calculated_number_string_pair) = self
1272 .calculate_obj_pair_to_number_strings(
1273 &less_equal_fact.left,
1274 &less_equal_fact.right,
1275 )
1276 {
1277 let compare_result = compare_number_strings(
1278 &calculated_number_string_pair.0,
1279 &calculated_number_string_pair.1,
1280 );
1281 return Some(matches!(
1282 compare_result,
1283 NumberCompareResult::Less | NumberCompareResult::Equal
1284 ));
1285 }
1286 self.try_verify_numeric_order_via_div_elimination(
1287 &less_equal_fact.left,
1288 &less_equal_fact.right,
1289 true,
1290 )
1291 }
1292 _ => None,
1293 }
1294 }
1295
1296 fn calculate_obj_pair_to_number_strings(
1297 &self,
1298 left_obj: &Obj,
1299 right_obj: &Obj,
1300 ) -> Option<(String, String)> {
1301 let left_number = self.resolve_obj_to_number_resolved(left_obj)?;
1302 let right_number = self.resolve_obj_to_number_resolved(right_obj)?;
1303 Some((left_number.normalized_value, right_number.normalized_value))
1304 }
1305}