1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
|
pragma solidity >=0.0;
import "../Oracles/Oracle.sol";
/// @title Signed message oracle contract - Allows to set an outcome with a signed message
/// @author Stefan George - <stefan@gnosis.pm>
contract SignedMessageOracle is Oracle {
/*
* Events
*/
event SignerReplacement(address indexed newSigner);
event OutcomeAssignment(int outcome);
/*
* Storage
*/
address public signer;
bytes32 public descriptionHash;
uint nonce;
bool public isSet;
int public outcome;
/*
* Modifiers
*/
modifier isSigner () {
// Only signer is allowed to proceed
require(msg.sender == signer);
_;
}
/*
* Public functions
*/
/// @dev Constructor sets signer address based on signature
/// @param _descriptionHash Hash identifying off chain event description
/// @param v Signature parameter
/// @param r Signature parameter
/// @param s Signature parameter
constructor(bytes32 _descriptionHash, uint8 v, bytes32 r, bytes32 s)
public
{
signer = ecrecover(_descriptionHash, v, r, s);
descriptionHash = _descriptionHash;
}
/// @dev Replaces signer
/// @param newSigner New signer
/// @param _nonce Unique nonce to prevent replay attacks
/// @param v Signature parameter
/// @param r Signature parameter
/// @param s Signature parameter
function replaceSigner(address newSigner, uint _nonce, uint8 v, bytes32 r, bytes32 s)
public
isSigner
{
// Result is not set yet and nonce and signer are valid
require( !isSet
&& _nonce > nonce
&& signer == ecrecover(keccak256(abi.encodePacked(descriptionHash, newSigner, _nonce)), v, r, s));
nonce = _nonce;
signer = newSigner;
emit SignerReplacement(newSigner);
}
/// @dev Sets outcome based on signed message
/// @param _outcome Signed event outcome
/// @param v Signature parameter
/// @param r Signature parameter
/// @param s Signature parameter
function setOutcome(int _outcome, uint8 v, bytes32 r, bytes32 s)
public
{
// Result is not set yet and signer is valid
require( !isSet
&& signer == ecrecover(keccak256(abi.encodePacked(descriptionHash, _outcome)), v, r, s));
isSet = true;
outcome = _outcome;
emit OutcomeAssignment(_outcome);
}
/// @dev Returns if winning outcome
/// @return Is outcome set?
function isOutcomeSet()
public
view
returns (bool)
{
return isSet;
}
/// @dev Returns winning outcome
/// @return Outcome
function getOutcome()
public
view
returns (int)
{
return outcome;
}
}
|