Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Make actor.Stop respect the given exit reason #88

Merged
merged 4 commits into from
Feb 23, 2025
Merged
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
5 changes: 5 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
@@ -1,5 +1,10 @@
# Changelog

## Unreleased

- Fixed a bug where using `actor.Stop()` with other reasons than `Normal`
would stop the process with `Normal`.

## v0.16.1 - 2025-01-20

- Fixed a bug where the static supervisor would return the incorrect value if
15 changes: 10 additions & 5 deletions src/gleam/otp/actor.gleam
Original file line number Diff line number Diff line change
@@ -139,7 +139,7 @@ import gleam/dynamic.{type Dynamic}
import gleam/erlang/atom
import gleam/erlang/charlist.{type Charlist}
import gleam/erlang/process.{
type ExitReason, type Pid, type Selector, type Subject, Abnormal,
type ExitReason, type Pid, type Selector, type Subject, Abnormal, Killed,
}
import gleam/option.{type Option, None, Some}
import gleam/otp/system.{
@@ -189,7 +189,7 @@ pub fn with_selector(
) -> Next(message, state) {
case value {
Continue(state, _) -> Continue(state, Some(selector))
_ -> value
Stop(_) -> value
}
}

@@ -257,7 +257,12 @@ pub type Spec(state, msg) {

// TODO: Check needed functionality here to be OTP compatible
fn exit_process(reason: ExitReason) -> ExitReason {
// TODO
case reason {
Abnormal(reason) -> process.send_abnormal_exit(process.self(), reason)
Killed -> process.kill(process.self())
_ -> Nil
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

No catch all patterns ever please 🙏

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Replaced it with Stop(_)👍

I also removed the dependency to gleam-lang/erlang#65 / gleam-lang/erlang#67, even though one of the tests is a bit wonky now 🤷

}

reason
}

@@ -404,10 +409,10 @@ fn initialise_actor(
loop(self)
}

// The init failed. Exit with an error.
// The init failed. Send the reason back to the parent, but exit normally.
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why do we now exit normally?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

failed_init_test would otherwise fail with the test process being killed. It's also what I'd assume to be correct based on the docs:

Failed(String)
The actor has failed to initialise. The actor shuts down and an error is returned to the parent process.

"shuts down" implies a clean exit to me.

The reason the test wasn't failing before is that exit_process didn't respect the reason given to it.

If we don't want a clean exit, then I think actor.Failed shouldn't exist, or it should at least have clearer documentation (similar reasoning to what I said here).

Failed(reason) -> {
process.send(ack, Error(Abnormal(reason)))
exit_process(Abnormal(reason))
exit_process(process.Normal)
}
}
}
48 changes: 48 additions & 0 deletions test/gleam/otp/actor_test.gleam
Original file line number Diff line number Diff line change
@@ -231,6 +231,54 @@ pub fn replace_selector_test() {
|> should.equal(dynamic.from("unknown message: String"))
}

pub fn abnormal_exit_can_be_trapped_test() {
process.trap_exits(True)
let exits =
process.new_selector()
|> process.selecting_trapped_exits(function.identity)

// Make an actor exit with an abnormal reason
let assert Ok(subject) =
actor.start(Nil, fn(_, _) { actor.Stop(process.Abnormal("reason")) })
process.send(subject, Nil)

let trapped_reason = process.select(exits, 10)

// Stop trapping exits, as otherwise other tests fail
process.trap_exits(False)

// The weird reason below is because of https://github.com/gleam-lang/erlang/issues/66
trapped_reason
|> should.equal(
Ok(process.ExitMessage(
process.subject_owner(subject),
process.Abnormal("Abnormal(\"reason\")"),
)),
)
}

pub fn killed_exit_can_be_trapped_test() {
process.trap_exits(True)
let exits =
process.new_selector()
|> process.selecting_trapped_exits(function.identity)

// Make an actor exit with a killed reason
let assert Ok(subject) =
actor.start(Nil, fn(_, _) { actor.Stop(process.Killed) })
process.send(subject, Nil)

let trapped_reason = process.select(exits, 10)

// Stop trapping exits, as otherwise other tests fail
process.trap_exits(False)

trapped_reason
|> should.equal(
Ok(process.ExitMessage(process.subject_owner(subject), process.Killed)),
)
}

fn mapped_selector(mapper: fn(a) -> ActorMessage) {
let subject = process.new_subject()