Skip to content

Commit 0085bab

Browse files
committed
add @require, like @Assert but thows ArgumentError, e.g.
function split{T}(c::T, n::Int) @require n > 0 return SplitIterator{T}(c, n) end split("xxx", 0) ERROR: ArgumentError: split(::ASCIIString, ::Int64) requires n > 0
1 parent 77936d0 commit 0085bab

File tree

5 files changed

+56
-0
lines changed

5 files changed

+56
-0
lines changed

base/contracts.jl

+22
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,22 @@
1+
# This file is a part of Julia. License is MIT: http://julialang.org/license
2+
3+
4+
@noinline function precondition_error(msg)
5+
io = IOBuffer()
6+
StackTraces.show_spec_linfo(io, StackTraces.lookup(backtrace()[3]))
7+
throw(ArgumentError("$(takebuf_string(io)) requires $msg"))
8+
end
9+
10+
11+
"""
12+
@require precondition [message]
13+
14+
Throw `ArgumentError` if `precondition` is false.
15+
"""
16+
macro require(precondition, msgs...)
17+
msg = isempty(msgs) ? string(precondition) : msgs[1]
18+
:(if ! $(esc(precondition)) precondition_error($msg) end)
19+
end
20+
# FIXME
21+
# Should this have a branch-prediction hint? (same for @assert?)
22+
# http://llvm.org/docs/BranchWeightMetadata.html#built-in-expect-instructions

base/exports.jl

+1
Original file line numberDiff line numberDiff line change
@@ -1436,6 +1436,7 @@ export
14361436
@noinline,
14371437

14381438
@assert,
1439+
@require,
14391440
@enum,
14401441
@label,
14411442
@goto,

base/sysimg.jl

+2
Original file line numberDiff line numberDiff line change
@@ -90,6 +90,8 @@ include("arraymath.jl")
9090
include("simdloop.jl")
9191
importall .SimdLoop
9292

93+
include("contracts.jl")
94+
9395
# map-reduce operators
9496
include("reduce.jl")
9597

doc/stdlib/base.rst

+6
Original file line numberDiff line numberDiff line change
@@ -1068,6 +1068,12 @@ Errors
10681068
10691069
Throw an ``AssertionError`` if ``cond`` is ``false``\ . Preferred syntax for writing assertions. Message ``text`` is optionally displayed upon assertion failure.
10701070

1071+
.. function:: @require precondition [message]
1072+
1073+
.. Docstring generated from Julia source
1074+
1075+
Throw ``ArgumentError`` if ``precondition`` is false.
1076+
10711077
.. function:: ArgumentError(msg)
10721078

10731079
.. Docstring generated from Julia source

test/misc.jl

+25
Original file line numberDiff line numberDiff line change
@@ -101,6 +101,31 @@ let deepthought(x, y) = 42
101101
end
102102
end
103103

104+
# test @require
105+
let
106+
f1(n) = begin @require n > 0; n end
107+
f2(n) = begin @require n != 0 "non-zero n"; n end
108+
109+
@test f1(1) == 1
110+
@test f2(1) == 1
111+
112+
try
113+
f1(0)
114+
error("unexpected")
115+
catch ex
116+
@test isa(ex, ArgumentError)
117+
@test ismatch(r"f1(.*) requires n > 0", ex.msg)
118+
end
119+
120+
try
121+
f2(0)
122+
error("unexpected")
123+
catch ex
124+
@test isa(ex, ArgumentError)
125+
@test ismatch(r"f2(.*) requires non-zero n", ex.msg)
126+
end
127+
end
128+
104129
let # test the process title functions, issue #9957
105130
oldtitle = Sys.get_process_title()
106131
Sys.set_process_title("julia0x1")

0 commit comments

Comments
 (0)