// ets_tracing: off
import * as L from "../../Collections/Immutable/List/index.js"
import * as T from "../../Effect/index.js"
import type { Lazy } from "../../Function/index.js"
import * as O from "../../Option/index.js"
import * as ST from "../../Structural/index.js"
import * as AD from "../AssertionData/index.js"
import type * as AssertionM from "../AssertionM/AssertionM.js"
import type * as AR from "../AssertionResult/index.js"
import type * as ARM from "../AssertionResultM/index.js"
import * as makeAssertionValue from "../AssertionValue/makeAssertionValue.js"
import * as BA from "../BoolAlgebra/index.js"
import * as BAM from "../BoolAlgebraM/index.js"
import * as PR from "../Primitives/index.js"
import * as R from "../Render/index.js"
export class Assertion implements AssertionM.AssertionM {
readonly [PR._A]: (_: A) => void
constructor(
readonly render: () => R.Render,
readonly run: (a: Lazy) => AR.AssertResult
) {
this.runM = this.runM.bind(this)
this.toString = this.toString.bind(this)
}
runM(a: Lazy): ARM.AssertResultM {
return new BAM.BoolAlgebraM(T.succeed(this.run(a)))
}
get stringify(): string {
return this.render().toString()
}
toString(): string {
return this.stringify
}
[ST.equalsSym](that: unknown): boolean {
if (isAssertion(that)) {
return this.stringify === that.stringify
}
return false
}
get [ST.hashSym](): number {
return ST.hashString(this.stringify)
}
}
export function isAssertion(that: unknown): that is Assertion {
return that instanceof Assertion
}
export function makeAssertion(name: string, ...params: R.RenderParam[]) {
return (run: (a: Lazy) => boolean) => {
const assertion = makeAssertionDirect(
name,
...params
)((actual) => {
const actualValue = actual()
const result = (): AR.AssertResult =>
run(() => actualValue)
? BA.success(
makeAssertionValue.makeAssertionValue(
assertion,
() => actualValue,
result
)
)
: BA.failure(
makeAssertionValue.makeAssertionValue(
assertion,
() => actualValue,
result
)
)
return result()
})
return assertion
}
}
export function makeAssertionDirect(name: string, ...params: R.RenderParam[]) {
return (run: (a: Lazy) => AR.AssertResult) =>
new Assertion(() => R.function_(name, L.of(L.from(params))), run)
}
export const isFalse = makeAssertion("isFalse")((a) => !a())
export const isEmptyString = makeAssertion("isEmptyString")(
(a) => a().length === 0
)
export function equalTo(expected: A): Assertion {
return makeAssertion(
"EqualTo",
R.param(expected)
)((actual) => {
const actualValue = actual()
return ST.equals(expected, actualValue)
})
}
export function makeAssertionRec(name: string, ...params: R.RenderParam[]) {
return (assertion: Assertion) => {
return (
get: (a: Lazy) => O.Option,
orElse: (ad: AD.AssertionData) => AR.AssertResult = AD.asFailure
) => {
const resultAssertion = (): Assertion =>
makeAssertionDirect(
name,
...params
)((a) => {
const actualValue = a()
return O.fold_(
get(a),
() => orElse(AD.makeAssertionData(resultAssertion(), actualValue)),
(b) => {
const innerResult = assertion.run(() => b)
const result = (): AR.AssertResult =>
BA.isSuccess(innerResult)
? BA.success(
makeAssertionValue.makeAssertionValue(
resultAssertion(),
() => actualValue,
result
)
)
: BA.failure(
makeAssertionValue.makeAssertionValue(
resultAssertion(),
() => b as unknown as A,
() => innerResult
)
)
return result()
}
)
})
return resultAssertion()
}
}
}
export function hasProperty(
name: string,
proj: (a: A) => B,
assertion: Assertion
): Assertion {
return makeAssertionRec(
"hasField",
R.param(R.quoted(name)),
R.param(R.field(name)),
R.param(assertion)
)(assertion)((actual) => {
return O.some(proj(actual()))
})
}
export function and(self: Assertion, that: Assertion): Assertion {
return new Assertion(
() => R.infix(R.param(self), "&&", R.param(that)),
(actual) => BA.and_(self.run(actual), that.run(actual))
)
}
export function or(self: Assertion, that: Assertion): Assertion {
return new Assertion(
() => R.infix(R.param(self), "||", R.param(that)),
(actual) => BA.or_(self.run(actual), that.run(actual))
)
}