Skip to content

Commit 065f7d7

Browse files
authored
Expand and clarify docs on persistent tasks (#297)
1 parent 1b88ce9 commit 065f7d7

File tree

3 files changed

+73
-8
lines changed

3 files changed

+73
-8
lines changed

CHANGELOG.md

+9
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,13 @@ The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
66
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
77

88

9+
## Version [v0.8.8] - 2024-10-10
10+
11+
### Changed
12+
13+
- Improved the documentation of `test_persisten_tasks`. ([#297])
14+
15+
916
## Version [v0.8.7] - 2024-04-09
1017

1118
- Reverted [#285], which was originally released in [v0.8.6], but caused a regression. ([#287], [#288])
@@ -227,6 +234,7 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
227234
[v0.8.5]: https://github.com/JuliaTesting/Aqua.jl/releases/tag/v0.8.5
228235
[v0.8.6]: https://github.com/JuliaTesting/Aqua.jl/releases/tag/v0.8.6
229236
[v0.8.7]: https://github.com/JuliaTesting/Aqua.jl/releases/tag/v0.8.7
237+
[v0.8.8]: https://github.com/JuliaTesting/Aqua.jl/releases/tag/v0.8.8
230238
[#93]: https://github.com/JuliaTesting/Aqua.jl/issues/93
231239
[#103]: https://github.com/JuliaTesting/Aqua.jl/issues/103
232240
[#113]: https://github.com/JuliaTesting/Aqua.jl/issues/113
@@ -274,3 +282,4 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
274282
[#285]: https://github.com/JuliaTesting/Aqua.jl/issues/285
275283
[#287]: https://github.com/JuliaTesting/Aqua.jl/issues/287
276284
[#288]: https://github.com/JuliaTesting/Aqua.jl/issues/288
285+
[#297]: https://github.com/JuliaTesting/Aqua.jl/issues/297

Project.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
name = "Aqua"
22
uuid = "4c88cf16-eb10-579e-8560-4a9242c79595"
33
authors = ["Takafumi Arakaki <[email protected]> and contributors"]
4-
version = "0.8.7"
4+
version = "0.8.8"
55

66
[deps]
77
Compat = "34da2185-b29b-5c13-b0c7-acf172513d20"

docs/src/persistent_tasks.md

+63-7
Original file line numberDiff line numberDiff line change
@@ -8,13 +8,24 @@ One consequence is that a package that launches
88
`Task`s in its `__init__` function may precompile successfully,
99
but block precompilation of any packages that depend on it.
1010

11+
The symptom of this problem is a message
12+
```
13+
◐ MyPackage: Waiting for background task / IO / timer. Interrupt to inspect...
14+
```
15+
that may appear during precompilation, with that precompilation process
16+
"hanging" until you press Ctrl-C.
17+
18+
Aqua has checks to determine whether your package *causes* this problem.
19+
Conversely, if you're a *victim* of this problem, it also has tools to help you
20+
determine which of your dependencies is causing the problem.
21+
1122
## Example
1223

1324
Let's create a dummy package, `PkgA`, that launches a persistent `Task`:
1425

1526
```julia
1627
module PkgA
17-
const t = Ref{Any}() # to prevent the Timer from being garbage-collected
28+
const t = Ref{Timer}() # used to prevent the Timer from being garbage-collected
1829
__init__() = t[] = Timer(0.1; interval=1) # create a persistent `Timer` `Task`
1930
end
2031
```
@@ -32,16 +43,31 @@ fails to precompile: `using PkgA` runs `PkgA.__init__()`, which
3243
leaves the `Timer` `Task` running, and that causes precompilation
3344
of `PkgB` to hang.
3445

35-
## Example with `expr`
46+
Without Aqua's tests, the developers of `PkgA` might not realize that their
47+
package is essentially unusable with any other package.
48+
49+
## Checking for persistent tasks
50+
51+
Running all of Aqua's tests will automatically check whether your package falls
52+
into this trap. In addition, there are ways to manually run (or tweak) this
53+
specific test.
54+
55+
### Manually running the peristent-tasks check
3656

37-
You can test that an expression using your package finishes without leaving any persistent
38-
tasks by passing a quoted expression:
57+
[`Aqua.test_persistent_tasks(MyPackage)`](@ref) will check whether `MyPackage` blocks
58+
precompilation for any packages that depend on it.
59+
60+
### Using an `expr` to check more than just `__init__`
61+
62+
By default, `Aqua.test_persistent_tasks` only checks whether a package's
63+
`__init__` function leaves persistent tasks running. To check whether other
64+
package functions leave persistent tasks running, pass a quoted expression:
3965

4066
```julia
4167
Aqua.test_persistent_tasks(MyPackage, quote
4268
# Code to run after loading MyPackage
4369
server = MyPackage.start_server()
44-
MyPackage.stop_server!(server)
70+
MyPackage.stop_server!(server) # ideally, this this should cleanly shut everything down. Does it?
4571
end)
4672
```
4773

@@ -76,8 +102,38 @@ function __init__()
76102
end
77103
```
78104

79-
In more complex cases, you may need to set up independently-callable functions
80-
to launch the tasks and set conditions that allow them to cleanly exit.
105+
In more complex cases, you may need to modify the task to support a clean
106+
shutdown. For example, if you have a `Task` that runs a never-terminating
107+
`while` loop, you could change
108+
109+
```
110+
while true
111+
112+
end
113+
```
114+
115+
to
116+
117+
```
118+
while task_should_run[]
119+
120+
end
121+
```
122+
123+
where
124+
125+
```
126+
const task_should_run = Ref(true)
127+
```
128+
129+
is a global constant in your module. Setting `task_should_run[] = false` from
130+
outside that `while` loop will cause it to terminate on its next iteration,
131+
allowing the `Task` to finish.
132+
133+
## Additional information
134+
135+
[Julia's devdocs](https://docs.julialang.org/en/v1/devdocs/precompile_hang/)
136+
also discuss this issue.
81137

82138
## [Test functions](@id test_persistent_tasks)
83139

0 commit comments

Comments
 (0)