#include <iostream>
#include <cmath>
#include <string>
#include <vector>
#include <functional>
using namespace std;
// ═══════════════════════════════════════════════════════
// СТРУКТУРА РЕЗУЛЬТАТА
// ═══════════════════════════════════════════════════════
struct QuadraticResult {
enum class Type { NO_ROOTS, ONE_ROOT, TWO_ROOTS };
Type type;
float root1 = 0.0f;
float root2 = 0.0f;
string message;
};
// ═══════════════════════════════════════════════════════
// ФУНКЦИЯ (переработана: возвращает результат)
// ═══════════════════════════════════════════════════════
QuadraticResult quadratic_equation(float a, float b, float c) {
QuadraticResult result;
float d = b * b - 4 * c * a;
if (d < 0) {
result.type = QuadraticResult::Type::NO_ROOTS;
result.message = "Корней нет!";
} else if (d == 0) {
result.type = QuadraticResult::Type::ONE_ROOT;
result.root1 = -b / (2 * a);
result.message = "Единственный Корень";
} else {
result.type = QuadraticResult::Type::TWO_ROOTS;
result.root1 = (-b + sqrt(d)) / (2 * a);
result.root2 = (-b - sqrt(d)) / (2 * a);
result.message = "Два корня";
}
return result;
}
// ═══════════════════════════════════════════════════════
// МИНИ-ФРЕЙМВОРК ДЛЯ ТЕСТОВ
// ═══════════════════════════════════════════════════════
int tests_passed = 0;
int tests_failed = 0;
void EXPECT_EQ_TYPE(const string& name, QuadraticResult::Type actual, QuadraticResult::Type expected) {
if (actual == expected) {
cout << " [OK] " << name << "\n";
tests_passed++;
} else {
cout << " [FAIL] " << name << " → ожидался тип " << (int)expected << ", получен " << (int)actual << "\n";
tests_failed++;
}
}
void EXPECT_NEAR_VAL(const string& name, float actual, float expected, float eps = 1e-3f) {
if (fabs(actual - expected) <= eps) {
cout << " [OK] " << name << "\n";
tests_passed++;
} else {
cout << " [FAIL] " << name << " → ожидалось " << expected << ", получено " << actual << "\n";
tests_failed++;
}
}
void EXPECT_EQ_STR(const string& name, const string& actual, const string& expected) {
if (actual == expected) {
cout << " [OK] " << name << "\n";
tests_passed++;
} else {
cout << " [FAIL] " << name << " → ожидалось \"" << expected << "\", получено \"" << actual << "\"\n";
tests_failed++;
}
}
void RUN_TEST(const string& group, const string& name, function<void()> fn) {
cout << "\n[TEST] " << group << " :: " << name << "\n";
fn();
}
// ═══════════════════════════════════════════════════════
// МОДУЛЬНЫЕ ТЕСТЫ
// ═══════════════════════════════════════════════════════
void test_two_roots() {
// Тест 1: исходный вызов из кода → x=-1, x=-3
RUN_TEST("TwoRoots", "BasicExample (1, 4, 3)", []() {
auto r = quadratic_equation(1, 4, 3);
EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type, QuadraticResult::Type::TWO_ROOTS);
EXPECT_NEAR_VAL("root1 == -1.0", r.root1, -1.0f);
EXPECT_NEAR_VAL("root2 == -3.0", r.root2, -3.0f);
});
// Тест 2: исходный вызов (53, 400, 100) — корни подставляются обратно
RUN_TEST("TwoRoots", "BigCoefficients (53, 400, 100)", []() {
auto r = quadratic_equation(53, 400, 100);
EXPECT_EQ_TYPE("тип = TWO_ROOTS", r.type, QuadraticResult::Type::TWO_ROOTS);
float check1 = 53*r.root1*r.root1 + 400*r.root1 + 100;
float check2 = 53*r.root2*r.root2 + 400*r.root2 + 100;
EXPECT_NEAR_VAL("a*x1^2+b*x1+c == 0", check1, 0.0f, 0.5f);
EXPECT_NEAR_VAL("a*x2^2+b*x2+c == 0", check2, 0.0f, 0.5f);
});
// Тест 3: x^2 - 5x + 6 = 0 → x=3, x=2
RUN_TEST("TwoRoots", "PositiveRoots (1, -5, 6)", []() {
auto r = quadratic_equation(1, -5, 6);
EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type, QuadraticResult::Type::TWO_ROOTS);
EXPECT_NEAR_VAL("root1 == 3.0", r.root1, 3.0f);
EXPECT_NEAR_VAL("root2 == 2.0", r.root2, 2.0f);
});
// Тест 4: 2x^2 - 4x - 6 = 0 → x=3, x=-1
RUN_TEST("TwoRoots", "LeadCoeff=2 (2, -4, -6)", []() {
auto r = quadratic_equation(2, -4, -6);
EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type, QuadraticResult::Type::TWO_ROOTS);
EXPECT_NEAR_VAL("root1 == 3.0", r.root1, 3.0f);
EXPECT_NEAR_VAL("root2 == -1.0", r.root2, -1.0f);
});
// Тест 5: x^2 - 1 = 0 → x=1, x=-1
RUN_TEST("TwoRoots", "DifferenceOfSquares (1, 0, -1)", []() {
auto r = quadratic_equation(1, 0, -1);
EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type, QuadraticResult::Type::TWO_ROOTS);
EXPECT_NEAR_VAL("root1 == 1.0", r.root1, 1.0f);
EXPECT_NEAR_VAL("root2 == -1.0", r.root2, -1.0f);
});
}
void test_one_root() {
// Тест 6: x^2 + 2x + 1 = 0 → x=-1
RUN_TEST("OneRoot", "PerfectSquare (1, 2, 1)", []() {
auto r = quadratic_equation(1, 2, 1);
EXPECT_EQ_TYPE ("тип = ONE_ROOT", r.type, QuadraticResult::Type::ONE_ROOT);
EXPECT_NEAR_VAL("root1 == -1.0", r.root1, -1.0f);
});
// Тест 7: 4x^2 - 4x + 1 = 0 → x=0.5
RUN_TEST("OneRoot", "LeadCoeff=4 (4, -4, 1)", []() {
auto r = quadratic_equation(4, -4, 1);
EXPECT_EQ_TYPE ("тип = ONE_ROOT", r.type, QuadraticResult::Type::ONE_ROOT);
EXPECT_NEAR_VAL("root1 == 0.5", r.root1, 0.5f);
});
// Тест 8: 9x^2 + 6x + 1 = 0 → x=-1/3
RUN_TEST("OneRoot", "NegativeRoot (9, 6, 1)", []() {
auto r = quadratic_equation(9, 6, 1);
EXPECT_EQ_TYPE ("тип = ONE_ROOT", r.type, QuadraticResult::Type::ONE_ROOT);
EXPECT_NEAR_VAL("root1 == -0.3333", r.root1, -1.0f/3.0f);
});
}
void test_no_roots() {
// Тест 9: исходный вызов из кода (14, 28, 320)
RUN_TEST("NoRoots", "OriginalCall (14, 28, 320)", []() {
auto r = quadratic_equation(14, 28, 320);
EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
});
// Тест 10: x^2 + x + 1 = 0
RUN_TEST("NoRoots", "Classic (1, 1, 1)", []() {
auto r = quadratic_equation(1, 1, 1);
EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
});
// Тест 11: x^2 + 1 = 0
RUN_TEST("NoRoots", "NoB (1, 0, 1)", []() {
auto r = quadratic_equation(1, 0, 1);
EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
});
// Тест 12: 3x^2 + 2x + 5 = 0
RUN_TEST("NoRoots", "LargeC (3, 2, 5)", []() {
auto r = quadratic_equation(3, 2, 5);
EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
});
}
void test_messages() {
// Тест 13
RUN_TEST("Messages", "NoRoots message", []() {
auto r = quadratic_equation(1, 0, 1);
EXPECT_EQ_STR("message == 'Корней нет!'", r.message, "Корней нет!");
});
// Тест 14
RUN_TEST("Messages", "OneRoot message", []() {
auto r = quadratic_equation(1, 2, 1);
EXPECT_EQ_STR("message == 'Единственный Корень'", r.message, "Единственный Корень");
});
// Тест 15
RUN_TEST("Messages", "TwoRoots message", []() {
auto r = quadratic_equation(1, 4, 3);
EXPECT_EQ_STR("message == 'Два корня'", r.message, "Два корня");
});
}
// ═══════════════════════════════════════════════════════
// MAIN
// ═══════════════════════════════════════════════════════
int main() {
cout << "╔══════════════════════════════════════════╗\n";
cout << "║ МОДУЛЬНЫЕ ТЕСТЫ: quadratic_equation ║\n";
cout << "╚══════════════════════════════════════════╝\n";
test_two_roots();
test_one_root();
test_no_roots();
test_messages();
cout << "\n══════════════════════════════════════════\n";
cout << " Итого: " << (tests_passed + tests_failed) << " тестов\n";
cout << " Пройдено: " << tests_passed << "\n";
cout << " Провалено: " << tests_failed << "\n";
cout << "══════════════════════════════════════════\n";
return tests_failed == 0 ? 0 : 1;
}
#include <iostream>
#include <cmath>
#include <string>
#include <vector>
#include <functional>

using namespace std;

// ═══════════════════════════════════════════════════════
//  СТРУКТУРА РЕЗУЛЬТАТА
// ═══════════════════════════════════════════════════════

struct QuadraticResult {
    enum class Type { NO_ROOTS, ONE_ROOT, TWO_ROOTS };
    Type type;
    float root1 = 0.0f;
    float root2 = 0.0f;
    string message;
};

// ═══════════════════════════════════════════════════════
//  ФУНКЦИЯ (переработана: возвращает результат)
// ═══════════════════════════════════════════════════════

QuadraticResult quadratic_equation(float a, float b, float c) {
    QuadraticResult result;
    float d = b * b - 4 * c * a;

    if (d < 0) {
        result.type    = QuadraticResult::Type::NO_ROOTS;
        result.message = "Корней нет!";
    } else if (d == 0) {
        result.type    = QuadraticResult::Type::ONE_ROOT;
        result.root1   = -b / (2 * a);
        result.message = "Единственный Корень";
    } else {
        result.type    = QuadraticResult::Type::TWO_ROOTS;
        result.root1   = (-b + sqrt(d)) / (2 * a);
        result.root2   = (-b - sqrt(d)) / (2 * a);
        result.message = "Два корня";
    }
    return result;
}

// ═══════════════════════════════════════════════════════
//  МИНИ-ФРЕЙМВОРК ДЛЯ ТЕСТОВ
// ═══════════════════════════════════════════════════════

int tests_passed = 0;
int tests_failed = 0;

void EXPECT_EQ_TYPE(const string& name, QuadraticResult::Type actual, QuadraticResult::Type expected) {
    if (actual == expected) {
        cout << "  [OK] " << name << "\n";
        tests_passed++;
    } else {
        cout << "  [FAIL] " << name << " → ожидался тип " << (int)expected << ", получен " << (int)actual << "\n";
        tests_failed++;
    }
}

void EXPECT_NEAR_VAL(const string& name, float actual, float expected, float eps = 1e-3f) {
    if (fabs(actual - expected) <= eps) {
        cout << "  [OK] " << name << "\n";
        tests_passed++;
    } else {
        cout << "  [FAIL] " << name << " → ожидалось " << expected << ", получено " << actual << "\n";
        tests_failed++;
    }
}

void EXPECT_EQ_STR(const string& name, const string& actual, const string& expected) {
    if (actual == expected) {
        cout << "  [OK] " << name << "\n";
        tests_passed++;
    } else {
        cout << "  [FAIL] " << name << " → ожидалось \"" << expected << "\", получено \"" << actual << "\"\n";
        tests_failed++;
    }
}

void RUN_TEST(const string& group, const string& name, function<void()> fn) {
    cout << "\n[TEST] " << group << " :: " << name << "\n";
    fn();
}

// ═══════════════════════════════════════════════════════
//  МОДУЛЬНЫЕ ТЕСТЫ
// ═══════════════════════════════════════════════════════

void test_two_roots() {
    // Тест 1: исходный вызов из кода → x=-1, x=-3
    RUN_TEST("TwoRoots", "BasicExample (1, 4, 3)", []() {
        auto r = quadratic_equation(1, 4, 3);
        EXPECT_EQ_TYPE ("тип = TWO_ROOTS",   r.type,  QuadraticResult::Type::TWO_ROOTS);
        EXPECT_NEAR_VAL("root1 == -1.0",     r.root1, -1.0f);
        EXPECT_NEAR_VAL("root2 == -3.0",     r.root2, -3.0f);
    });

    // Тест 2: исходный вызов (53, 400, 100) — корни подставляются обратно
    RUN_TEST("TwoRoots", "BigCoefficients (53, 400, 100)", []() {
        auto r = quadratic_equation(53, 400, 100);
        EXPECT_EQ_TYPE("тип = TWO_ROOTS", r.type, QuadraticResult::Type::TWO_ROOTS);
        float check1 = 53*r.root1*r.root1 + 400*r.root1 + 100;
        float check2 = 53*r.root2*r.root2 + 400*r.root2 + 100;
        EXPECT_NEAR_VAL("a*x1^2+b*x1+c == 0", check1, 0.0f, 0.5f);
        EXPECT_NEAR_VAL("a*x2^2+b*x2+c == 0", check2, 0.0f, 0.5f);
    });

    // Тест 3: x^2 - 5x + 6 = 0 → x=3, x=2
    RUN_TEST("TwoRoots", "PositiveRoots (1, -5, 6)", []() {
        auto r = quadratic_equation(1, -5, 6);
        EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type,  QuadraticResult::Type::TWO_ROOTS);
        EXPECT_NEAR_VAL("root1 == 3.0",    r.root1, 3.0f);
        EXPECT_NEAR_VAL("root2 == 2.0",    r.root2, 2.0f);
    });

    // Тест 4: 2x^2 - 4x - 6 = 0 → x=3, x=-1
    RUN_TEST("TwoRoots", "LeadCoeff=2 (2, -4, -6)", []() {
        auto r = quadratic_equation(2, -4, -6);
        EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type,  QuadraticResult::Type::TWO_ROOTS);
        EXPECT_NEAR_VAL("root1 == 3.0",    r.root1, 3.0f);
        EXPECT_NEAR_VAL("root2 == -1.0",   r.root2, -1.0f);
    });

    // Тест 5: x^2 - 1 = 0 → x=1, x=-1
    RUN_TEST("TwoRoots", "DifferenceOfSquares (1, 0, -1)", []() {
        auto r = quadratic_equation(1, 0, -1);
        EXPECT_EQ_TYPE ("тип = TWO_ROOTS", r.type,  QuadraticResult::Type::TWO_ROOTS);
        EXPECT_NEAR_VAL("root1 ==  1.0",   r.root1,  1.0f);
        EXPECT_NEAR_VAL("root2 == -1.0",   r.root2, -1.0f);
    });
}

void test_one_root() {
    // Тест 6: x^2 + 2x + 1 = 0 → x=-1
    RUN_TEST("OneRoot", "PerfectSquare (1, 2, 1)", []() {
        auto r = quadratic_equation(1, 2, 1);
        EXPECT_EQ_TYPE ("тип = ONE_ROOT",  r.type,  QuadraticResult::Type::ONE_ROOT);
        EXPECT_NEAR_VAL("root1 == -1.0",   r.root1, -1.0f);
    });

    // Тест 7: 4x^2 - 4x + 1 = 0 → x=0.5
    RUN_TEST("OneRoot", "LeadCoeff=4 (4, -4, 1)", []() {
        auto r = quadratic_equation(4, -4, 1);
        EXPECT_EQ_TYPE ("тип = ONE_ROOT",  r.type,  QuadraticResult::Type::ONE_ROOT);
        EXPECT_NEAR_VAL("root1 == 0.5",    r.root1, 0.5f);
    });

    // Тест 8: 9x^2 + 6x + 1 = 0 → x=-1/3
    RUN_TEST("OneRoot", "NegativeRoot (9, 6, 1)", []() {
        auto r = quadratic_equation(9, 6, 1);
        EXPECT_EQ_TYPE ("тип = ONE_ROOT",      r.type,  QuadraticResult::Type::ONE_ROOT);
        EXPECT_NEAR_VAL("root1 == -0.3333",    r.root1, -1.0f/3.0f);
    });
}

void test_no_roots() {
    // Тест 9: исходный вызов из кода (14, 28, 320)
    RUN_TEST("NoRoots", "OriginalCall (14, 28, 320)", []() {
        auto r = quadratic_equation(14, 28, 320);
        EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
    });

    // Тест 10: x^2 + x + 1 = 0
    RUN_TEST("NoRoots", "Classic (1, 1, 1)", []() {
        auto r = quadratic_equation(1, 1, 1);
        EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
    });

    // Тест 11: x^2 + 1 = 0
    RUN_TEST("NoRoots", "NoB (1, 0, 1)", []() {
        auto r = quadratic_equation(1, 0, 1);
        EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
    });

    // Тест 12: 3x^2 + 2x + 5 = 0
    RUN_TEST("NoRoots", "LargeC (3, 2, 5)", []() {
        auto r = quadratic_equation(3, 2, 5);
        EXPECT_EQ_TYPE("тип = NO_ROOTS", r.type, QuadraticResult::Type::NO_ROOTS);
    });
}

void test_messages() {
    // Тест 13
    RUN_TEST("Messages", "NoRoots message", []() {
        auto r = quadratic_equation(1, 0, 1);
        EXPECT_EQ_STR("message == 'Корней нет!'", r.message, "Корней нет!");
    });

    // Тест 14
    RUN_TEST("Messages", "OneRoot message", []() {
        auto r = quadratic_equation(1, 2, 1);
        EXPECT_EQ_STR("message == 'Единственный Корень'", r.message, "Единственный Корень");
    });

    // Тест 15
    RUN_TEST("Messages", "TwoRoots message", []() {
        auto r = quadratic_equation(1, 4, 3);
        EXPECT_EQ_STR("message == 'Два корня'", r.message, "Два корня");
    });
}

// ═══════════════════════════════════════════════════════
//  MAIN
// ═══════════════════════════════════════════════════════

int main() {
    cout << "╔══════════════════════════════════════════╗\n";
    cout << "║    МОДУЛЬНЫЕ ТЕСТЫ: quadratic_equation   ║\n";
    cout << "╚══════════════════════════════════════════╝\n";

    test_two_roots();
    test_one_root();
    test_no_roots();
    test_messages();

    cout << "\n══════════════════════════════════════════\n";
    cout << "  Итого: " << (tests_passed + tests_failed) << " тестов\n";
    cout << "  Пройдено: " << tests_passed << "\n";
    cout << "  Провалено: " << tests_failed << "\n";
    cout << "══════════════════════════════════════════\n";

    return tests_failed == 0 ? 0 : 1;
}
