-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathevents.sol
342 lines (300 loc) · 15.5 KB
/
events.sol
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
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
pragma solidity^0.5.0;
/// @title UniCoin smart contract
/// @author Chris Maree, Helda Mandlate, Luke Meiklejohn
/// @dev import contracts from openzeppelin related to ownable and ERC20, ERC721 tokens
import "openzeppelin-solidity/contracts/ownership/Ownable.sol";
import "openzeppelin-solidity/contracts/token/ERC20/ERC20.sol";
import "openzeppelin-solidity/contracts/token/ERC721/ERC721Metadata.sol";
/// @notice contract begins here
contract UnicoinRegistry is ERC721Metadata {
/// @notice Creates a struct for users of the plaform, needs their Ethereum address and profile URL
struct User {
address owned_address;
string profile_uri;
}
/// @notice Creates an array of users that a registered
User[] public users;
/// @notice The mapping below maps all users' addresses to their userID
mapping (address => uint256) public userAddresses ;
/// @notice Creates user defined type
enum bidStatus {Pending, Accepted, Rejected, Sale, Cancelled}
/// @notice Creates a struct for all bids, takes in the offer (amount of the bid), one of the enum parameters, publication Id and owner Id
struct Bid {
uint256 offer;
bidStatus status;
uint256 publication_Id;
uint256 owner_Id; /// @dev owner of the bid
}
/// @notice Creates an array of bids that have been placed
Bid[] public bids;
/// @notice The mapping below maps all bidders' IDs to their userID
mapping(uint256 => uint256[]) public bidOwners;
/// @notice Creates a struct for all publications
/// @param author_Id The array below will contain all bids received for that work
/// @param publication_uri If the researcher has chosen the auction pricing structure, the below is TRUE.
/// @param publication_bids If the auction is still running, the below is TRUE, because the researcher can choose to stop the auction at any point.
/// @param isAuction If both of the booleans above are FALSE, the price below is the flat sale price of the work.
/// @param isRunning The value of the publication
/// @param sell_price The co-authors/ contributors of the publication
/// @param contributors_weightings The contributor's respective weighting contributions
struct Publication {
uint256 author_Id;
string publication_uri;
uint256[] publication_bids;
bool isAuction;
bool isRunning;
uint256 sell_price;
uint256[] contributors;
uint256[] contributors_weightings;
}
/// @notice Creates an array of publications for every published document
Publication[] public publications;
/// @notice The mapping below will map the addresses of all the successful bidders' addresses to the ID of their owned publications
mapping(uint256 => uint256[]) public publicationOwners;
/// @notice Creates a struct for licencing
/// @param buyer_Id of each publication's buyer
/// @param publication_Id of the publication being bought
/// @param bid_Id The bid's Id for the publication
struct LicenceDesign {
uint256 buyer_Id;
uint256 Publication_Id;
uint256 bid_Id;
}
/// @notice Creates an array of purchased licences
LicenceDesign[] public licences;
/// @notice Mapping of licence Id to get the licence owners
mapping(uint256 => uint256) public licenceOwners;
/// @notice Mapping of licence Id to get the publication Id
mapping(uint256 => uint256[]) public publicationLicences;
/// @dev ERC20 is now daiContract
ERC20 daiContract;
/// @dev The constructor below reserves user 0 for all unregistered users
/// @param _daiContractAddress DAI contract address
constructor(address _daiContractAddress) public ERC721Metadata("UniCoin Licence", "UNIC"){
users.push(User(address(0), ""));
licences.push(LicenceDesign(0, 0, 0));
daiContract = ERC20(_daiContractAddress);
}
/// @notice This function registers a user on the platform by taking in their profile URL
/// @param _profile_uri user profile url
/// @dev If the user's address is in position 0 of the userAddresses array, they are unregistered
/// @dev Create an instance of the user and add the Id to their address
function registerUser(string memory _profile_uri) public {
require(bytes(_profile_uri).length > 0, "Profile URI should not be empty.");
require(userAddresses[msg.sender]==0,"User already registered.");
uint256 id = users.push(User(msg.sender,_profile_uri));
userAddresses[msg.sender] = id - 1;
}
/// @notice This function creates a publication on the system, with blank arrays for publication bids and owners,
/// @notice since no one has bidded for or bought a licence yet
/// @dev The researcher only specifies the flat rate if they have chosen not to auction the work
/// @dev Add instance to the respective arrays
event NewPublication(
string _publication_uri,
bool _isAuction,
uint256 _sell_price,
uint256[] _contributors
);
function createPublication(
string memory _publication_uri,
bool _isAuction,
bool _isRunning,
uint256 _sell_price,
uint256[] memory _contributors,
uint256[] memory _contributors_weightings) public {
require(bytes(_publication_uri).length > 0, "Publication URI should not be empty.");
require(userAddresses[msg.sender] != 0, "User address is not registered.");
if(_isAuction) {
require(_sell_price == 0, "Should not specify sell price for auction.");
}
else {
require(_sell_price > 0, "Sell price not specified.");
}
uint256 _author_Id = userAddresses[msg.sender];
uint256[] memory _publication_bids;
Publication memory _publication = Publication(
_author_Id,
_publication_uri,
_publication_bids,
_isAuction,
_isRunning,
_sell_price,
_contributors,
_contributors_weightings);
uint256 _id = publications.push(_publication);
publicationOwners[_author_Id].push(_id - 1);
}
event NewBid(
uint256 _publication_Id,
uint256 _offer
);
/// @notice This function creates a new bid for a particular publication
/// @param _offer for the research
/// @param _publication_Id is the index for the publication Id
/// @dev The bidder should only be able to submit a bid if the publication's pricing structure is an auction and the auction is running
/// @dev By default the bid will have a status of Pending until it is accepted or rejected by the author
/// @dev If the author has specified a flat rate, the buyer doesn't submit a bid but just sends the funds.
/// @dev The funds sent should match the sale price specified by the author.
/// @dev This 'bid' has a status of sale because the author does not need to evaluate and accept/reject these bids.
/// @dev Transfer Dai from buyer to seller
/// @dev parameters of licence design: buyer_id, publication id, bid_id
function makeBid(uint256 _offer, uint256 _publication_Id) public {
require(publications[_publication_Id].author_Id != 0, "Publication not enlisted.");
require(userAddresses[msg.sender] != 0, "Bidder address is not registered.");
if(publications[_publication_Id].isAuction) {
require(publications[_publication_Id].isRunning, "Auction is not running.");
uint256 _id = bids.push(Bid(_offer, bidStatus.Pending, _publication_Id, userAddresses[msg.sender]));
publications[_publication_Id].publication_bids.push(_id - 1);
bidOwners[userAddresses[msg.sender]].push(_id - 1);
}
if(!publications[_publication_Id].isAuction) {
require(_offer == publications[_publication_Id].sell_price, "Incorrect funds sent.");
uint256 _id = bids.push(Bid(_offer, bidStatus.Sale, _publication_Id, userAddresses[msg.sender])) - 1;
publications[_publication_Id].publication_bids.push(_id);
bidOwners[userAddresses[msg.sender]].push(_id);
// require(daiContract.allowance(msg.sender, address(this)) >= _offer, "Insufficient fund allowance");
// address publisherAddress = users[publications[_publication_Id].author_Id].owned_address;
// daiContract.transferFrom(msg.sender, publisherAddress, _offer);
uint256 _licence_Id = licences.push(LicenceDesign(bids[_id].owner_Id, _publication_Id, _id));
licenceOwners[bids[_id].owner_Id] = _licence_Id;
publicationLicences[_publication_Id].push(_licence_Id);
_mint(users[bids[_id].owner_Id].owned_address, _licence_Id);
}
}
event AcceptedBid(
uint256 _id
);
/// @notice This function allows the auctioneer to accept the bids
/// @dev parameters of licence design: buyer_id, publication id, bid_id
/// @notice This function allows the auctioneer to reject the bids
/// @param _id is the bid Id
function acceptBid(uint256 _id) public {
uint256 _publication_Id = bids[_id].publication_Id;
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
require(publications[_publication_Id].isAuction, "Publication not an auction.");
require(publications[_publication_Id].isRunning, "Auction is not running.");
bids[_id].status = bidStatus.Accepted;
uint256 _licence_Id = licences.push(LicenceDesign(bids[_id].owner_Id, _publication_Id, _id));
licenceOwners[bids[_id].owner_Id] = _licence_Id;
publicationLicences[_publication_Id].push(_licence_Id);
_mint(users[bids[_id].owner_Id].owned_address, _licence_Id);
}
event RejectedBid(
uint256 _id
);
/// @notice This function allows the auctioneer to reject the bids
/// @param _id is the bid Id
function rejectBid(uint256 _id) public {
uint256 _publication_Id = bids[_id].publication_Id;
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
require(publications[_publication_Id].isAuction, "Publication not an auction.");
require(publications[_publication_Id].isRunning, "Auction not running.");
bids[_id].status = bidStatus.Rejected;
}
event CancelledBid(
uint256 _id
);
/// @notice This function allows the auctioneer to cancel the bids
/// @param _id is the bid Id
function cancelBid(uint256 _id) public {
uint256 _publication_Id = bids[_id].publication_Id;
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
require(publications[_publication_Id].isAuction, "Publication not an auction.");
require(publications[_publication_Id].isRunning, "Auction not running.");
bids[_id].status = bidStatus.Cancelled;
}
event ChangeToSale(
uint256 _publication_Id,
uint256 _sell_price
);
/// @notice This function allows the auctioneer to change from an auction to a sale
/// @param _publication_Id publication id number
/// @param _sell_price for the research
function changeToSale(uint256 _publication_Id, uint256 _sell_price) public {
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
require(publications[_publication_Id].isAuction, "Publication is not an auction");
publications[_publication_Id].sell_price = _sell_price;
publications[_publication_Id].isAuction = false;
}
event ChangeToAuction(
uint256 _publication_Id,
);
/// @notice This function allows the auctioneer to change from a sale to an auction
/// @param _publication_Id publication id number
function changeToAuction(uint256 _publication_Id) public {
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
require(!publications[_publication_Id].isAuction, "Publication is already on auction");
publications[_publication_Id].sell_price = 0;
publications[_publication_Id].isAuction = true;
}
event ChangeSellPrice(
uint256 _publication_Id,
uint256 _sell_price
);
/// @notice This function allows the auctioneer to change the sell price
/// @param _publication_Id publication id number
/// @param _sell_price for the research
function changeSellPrice(uint256 _publication_Id, uint256 _sell_price) public {
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
require(!publications[_publication_Id].isAuction, "Publication is on auction.");
publications[_publication_Id].sell_price = _sell_price;
}
event ChangeRunningStatus(
uint256 _publication_Id
);
/// @notice This function allows the auctioneer to change the running status
/// @param _publication_Id publication id number
function changeRunningStatus(uint256 _publication_Id) public {
require(userAddresses[msg.sender] == publications[_publication_Id].author_Id, "User not the author of this publication");
publications[_publication_Id].isRunning = !publications[_publication_Id].isRunning;
}
/// @return This function allows anyone to get the list of publications based on the address of the publisher
/// @param _address eth address for the user
function getPublications(address _address) public view returns(uint256[] memory) {
uint256 _author_Id = userAddresses[_address];
return publicationOwners[_author_Id];
}
/// @return This function allows anyone to get the list of bids based on address of the user
function getBids(address _address) public view returns(uint256[] memory) {
uint256 _userAddress = userAddresses[_address];
return bidOwners[_userAddress];
}
// From the user ID, get a list of all publications owned by the user ()
// function getPublications(uint256 _user_Id) public view returns(uint256[] memory) {
// return publicationOwners[_user_Id];
// }
// function getBids(uint256 _user_Id) public view returns(uint256[] memory) {
// return bidOwners[_user_Id];
// }
/// @return This function allows anyone to get list of bids per publication
/// @param _publication_Id publication id number
function getPublicationBids(uint256 _publication_Id) public view returns(uint256[] memory) {
return publications[_publication_Id].publication_bids;
}
/// @return This function allows the return of the total number of publications
function getPublicationLength() public view returns(uint count) {
return publications.length;
}
/// @return Returns information about a spesific publication ID
/// @param _publication_Id publication id number
function getPublication(uint256 _publication_Id) public view returns(
uint256,
string memory,
uint256[] memory,
bool,
bool,
uint256,
uint256[] memory,
uint256[] memory){
Publication memory _publication = publications[_publication_Id];
return (
_publication.author_Id,
_publication.publication_uri,
_publication.publication_bids,
_publication.isAuction,
_publication.isRunning,
_publication.sell_price,
_publication.contributors,
_publication.contributors_weightings);
}
}