pragma solidity ^0.5.2;
// File: node_modules/openzeppelin-solidity/contracts/access/Roles.sol
/**
* @title Roles
* @dev Library for managing addresses assigned to a Role.
*/
library Roles {
struct Role {
mapping (address => bool) bearer;
}
/**
* @dev give an account access to this role
*/
function add(Role storage role, address account) internal {
require(account != address(0));
require(!has(role, account));
role.bearer[account] = true;
}
/**
* @dev remove an account's access to this role
*/
function remove(Role storage role, address account) internal {
require(account != address(0));
require(has(role, account));
role.bearer[account] = false;
}
/**
* @dev check if an account has this role
* @return bool
*/
function has(Role storage role, address account) internal view returns (bool) {
require(account != address(0));
return role.bearer[account];
}
}
// File: node_modules/openzeppelin-solidity/contracts/access/roles/PauserRole.sol
contract PauserRole {
using Roles for Roles.Role;
event PauserAdded(address indexed account);
event PauserRemoved(address indexed account);
Roles.Role private _pausers;
constructor () internal {
_addPauser(msg.sender);
}
modifier onlyPauser() {
require(isPauser(msg.sender));
_;
}
function isPauser(address account) public view returns (bool) {
return _pausers.has(account);
}
function addPauser(address account) public onlyPauser {
_addPauser(account);
}
function renouncePauser() public {
_removePauser(msg.sender);
}
function _addPauser(address account) internal {
_pausers.add(account);
emit PauserAdded(account);
}
function _removePauser(address account) internal {
_pausers.remove(account);
emit PauserRemoved(account);
}
}
// File: node_modules/openzeppelin-solidity/contracts/lifecycle/Pausable.sol
/**
* @title Pausable
* @dev Base contract which allows children to implement an emergency stop mechanism.
*/
contract Pausable is PauserRole {
event Paused(address account);
event Unpaused(address account);
bool private _paused;
constructor () internal {
_paused = false;
}
/**
* @return true if the contract is paused, false otherwise.
*/
function paused() public view returns (bool) {
return _paused;
}
/**
* @dev Modifier to make a function callable only when the contract is not paused.
*/
modifier whenNotPaused() {
require(!_paused);
_;
}
/**
* @dev Modifier to make a function callable only when the contract is paused.
*/
modifier whenPaused() {
require(_paused);
_;
}
/**
* @dev called by the owner to pause, triggers stopped state
*/
function pause() public onlyPauser whenNotPaused {
_paused = true;
emit Paused(msg.sender);
}
/**
* @dev called by the owner to unpause, returns to normal state
*/
function unpause() public onlyPauser whenPaused {
_paused = false;
emit Unpaused(msg.sender);
}
}
// File: contracts/Adminable.sol
contract Adminable {
using Roles for Roles.Role;
event AdminAdded(address indexed account);
event AdminRemoved(address indexed account);
Roles.Role private _admins;
constructor () internal {
_addAdmin(msg.sender);
}
modifier onlyAdmin() {
require(isAdmin(msg.sender));
_;
}
function isAdmin(address account) public view returns (bool) {
return _admins.has(account);
}
function addAdmin(address account) public onlyAdmin {
_addAdmin(account);
}
function renounceAdmin() public {
_removeAdmin(msg.sender);
}
function _addAdmin(address account) internal {
_admins.add(account);
emit AdminAdded(account);
}
function _removeAdmin(address account) internal {
_admins.remove(account);
emit AdminRemoved(account);
}
}
// File: contracts/Authorizable.sol
contract Authorizable is Adminable {
address public authorizedAddress;
modifier onlyAuthorized() {
require(msg.sender == authorizedAddress);
_;
}
function updateAuthorizedAddress(address _address) onlyAdmin public {
authorizedAddress = _address;
}
}
// File: contracts/SoarStorage.sol
/**
@title Soar Storage
@author Marek Tlacbaba (marek@soar.earth)
@dev This smart contract behave as simple storage and can be
accessed only by authorized caller who is responsible for any
checks and validation. The authorized caller can updated by
admins so it allows to update application logic
and keeping data and events untouched.
*/
//TODO
// use safeMath
contract SoarStorage is Authorizable {
/**
Status:
0 - unknown
1 - created
2 - updated
3 - deleted
*/
struct ListingObject {
address owner;
address sponsor;
bytes12 geohash;
mapping (address => mapping (bytes32 => uint )) sales;
uint256 salesCount;
uint8 status;
}
uint public counter = 0;
mapping (bytes32 => ListingObject) internal listings;
event Listing (
bytes32 filehash,
address indexed owner,
address indexed sponsor,
string previewUrl,
string url,
string pointWKT,
bytes12 geohash,
string metadata
);
event ListingUpdated (
bytes32 filehash,
address indexed owner,
address indexed sponsor,
string previewUrl,
string url,
string pointWKT,
bytes12 geohash,
string metadata
);
event ListingDeleted (
bytes32 filehash,
address indexed owner,
address indexed sponsor
);
event Sale(
address indexed buyer,
bytes32 id,
address indexed owner,
address sponsor,
bytes32 indexed filehash,
uint price
);
function putListing (
bytes32 _filehash,
address _owner,
address _sponsor,
string memory _previewUrl,
string memory _url,
string memory _pointWKT,
bytes12 _geohash,
string memory _metadata
)
public
onlyAuthorized
{
listings[_filehash].owner = _owner;
listings[_filehash].sponsor = _sponsor;
listings[_filehash].geohash = _geohash;
listings[_filehash].status = 1;
counter++;
emit Listing(
_filehash,
_owner,
_sponsor,
_previewUrl,
_url,
_pointWKT,
_geohash,
_metadata
);
}
function updateListing (
bytes32 _filehash,
address _owner,
address _sponsor,
string memory _previewUrl,
string memory _url,
string memory _pointWKT,
bytes12 _geohash,
string memory _metadata
)
public
onlyAuthorized
{
listings[_filehash].geohash = _geohash;
listings[_filehash].status = 2;
emit ListingUpdated(
_filehash,
_owner,
_sponsor,
_previewUrl,
_url,
_pointWKT,
_geohash,
_metadata
);
}
function deleteListing(
bytes32 _filehash
)
public
onlyAuthorized
{
listings[_filehash].status = 3;
counter--;
emit ListingDeleted(_filehash, listings[_filehash].owner, listings[_filehash].sponsor);
}
function putSale (
address _buyer,
bytes32 _id,
bytes32 _filehash,
uint256 _price
)
public
onlyAuthorized
{
listings[_filehash].sales[_buyer][_id] = _price;
listings[_filehash].salesCount++;
emit Sale(_buyer, _id, listings[_filehash].owner, listings[_filehash].sponsor, _filehash, _price);
}
function getListingDetails(bytes32 _filehash, address _user, bytes32 _id)
public view
returns (
address owner_,
address sponsor_,
bytes12 geohash_,
uint8 status_,
uint256 sale_
)
{
owner_ = listings[_filehash].owner;
sponsor_ = listings[_filehash].sponsor;
geohash_ = listings[_filehash].geohash;
status_ = listings[_filehash].status;
sale_ = listings[_filehash].sales[_user][_id];
}
}
// File: contracts/Soar.sol
/**
@title Soar
@author Marek Tlacbaba (marek@soar.earth)
@dev Main Soar smart contract with bussiness logic composing
all other parts together and it is by design upgradable. When all
development is finished then all admins can be removed and no more
upgrade will be allowed.
*/
contract Soar is Pausable, Adminable {
// attributes
address public soarStorageAddress;
// contracts
SoarStorage private soarStorageContract;
bytes32 private emptyUserId = "00000000000000000000000000000000";
address private emptySponsor = address(0);
mapping (address => mapping ( address => bool)) private sponsors;
event SponsorAdminAdded(address indexed sponsor, address admin);
event SponsorAdminRemoved(address indexed sponsor, address admin);
modifier listingExistAndNotDeleted(bytes32 _filehash) {
(,,,uint8 status,) = soarStorageContract.getListingDetails(_filehash, msg.sender, emptyUserId);
require(status == 1 || status == 2, "Listing must exist and not be deleted");
_;
}
modifier listingNotExistOrDeleted(bytes32 _filehash) {
(,,,uint8 status,) = soarStorageContract.getListingDetails(_filehash, msg.sender, emptyUserId);
require(status == 0 || status == 3, "Listing can not exist or must be deleted");
_;
}
modifier onlySponsor(address _sponsor) {
require(sponsors[_sponsor][msg.sender] == true, "Only sponsor");
_;
}
modifier onlySponsorListingOwner(bytes32 _filehash, address _sponsor) {
require(sponsors[_sponsor][msg.sender] == true, "Only sponsor");
(,address sponsor,,uint8 status,) = soarStorageContract.getListingDetails(_filehash, msg.sender, emptyUserId);
require(status == 1 || status == 2, "Listing must exist and not be deleted");
require(sponsor == _sponsor, "Incorrect sponsor");
_;
}
constructor() public {}
/**
@dev Create listing as sponsor and put it in storage.
*/
function sponsorCreateListing(
address _sponsor,
address _owner,
bytes32 _filehash,
string memory _previewUrl,
string memory _url,
string memory _pointWKT,
bytes12 _geohash,
string memory _metadata)
public
whenNotPaused
listingNotExistOrDeleted(_filehash)
onlySponsor(_sponsor)
{
soarStorageContract.putListing(_filehash, _owner, _sponsor, _previewUrl, _url, _pointWKT, _geohash, _metadata);
}
/**
@dev Update listing as sponsor in storage.
*/
function sponsorUpdateListing(
address _sponsor,
address _owner,
bytes32 _filehash,
string memory _previewUrl,
string memory _url,
string memory _pointWKT,
bytes12 _geohash,
string memory _metadata)
public
whenNotPaused
onlySponsorListingOwner(_filehash, _sponsor)
{
soarStorageContract.updateListing(_filehash, _owner, _sponsor, _previewUrl, _url, _pointWKT, _geohash, _metadata);
}
/**
@dev Delete listing as sponsor in storage.
*/
function sponsorDeleteListing(
address _sponsor,
bytes32 _filehash)
public
whenNotPaused
onlySponsorListingOwner(_filehash, _sponsor)
{
soarStorageContract.deleteListing(_filehash);
}
function listingExist(bytes32 _filehash)
public view
whenNotPaused
returns (bool exists_)
{
(,,,uint8 status,) = soarStorageContract.getListingDetails(_filehash, msg.sender, emptyUserId);
exists_ = (status == 1 || status == 2);
}
/**
ADMIN FUNCTIONS
*/
function addSponsorAdmin(address _sponsor, address _admin)
public
whenNotPaused
onlyAdmin
{
sponsors[_sponsor][_admin] = true;
emit SponsorAdminAdded(_sponsor, _admin);
}
function removeSponsorAdmin(address _sponsor, address _admin)
public
whenNotPaused
onlyAdmin
{
sponsors[_sponsor][_admin] = false;
emit SponsorAdminRemoved(_sponsor, _admin);
}
function isSponsorAdmin(address _sponsor)
public view
returns (bool isSponsorAdmin_)
{
isSponsorAdmin_ = sponsors[_sponsor][msg.sender] == true;
}
function setSoarStorageContract(address _address)
public
whenNotPaused
onlyAdmin
{
soarStorageContract = SoarStorage(_address);
soarStorageAddress = _address;
}
}
{
"compilationTarget": {
"Soar.sol": "Soar"
},
"evmVersion": "byzantium",
"libraries": {},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"constant":true,"inputs":[{"name":"_sponsor","type":"address"}],"name":"isSponsorAdmin","outputs":[{"name":"isSponsorAdmin_","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"account","type":"address"}],"name":"isAdmin","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"soarStorageAddress","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_sponsor","type":"address"},{"name":"_admin","type":"address"}],"name":"addSponsorAdmin","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_sponsor","type":"address"},{"name":"_owner","type":"address"},{"name":"_filehash","type":"bytes32"},{"name":"_previewUrl","type":"string"},{"name":"_url","type":"string"},{"name":"_pointWKT","type":"string"},{"name":"_geohash","type":"bytes12"},{"name":"_metadata","type":"string"}],"name":"sponsorUpdateListing","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"unpause","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"account","type":"address"}],"name":"isPauser","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"paused","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"renouncePauser","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"account","type":"address"}],"name":"addAdmin","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"account","type":"address"}],"name":"addPauser","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_sponsor","type":"address"},{"name":"_admin","type":"address"}],"name":"removeSponsorAdmin","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"pause","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"renounceAdmin","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_sponsor","type":"address"},{"name":"_filehash","type":"bytes32"}],"name":"sponsorDeleteListing","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_filehash","type":"bytes32"}],"name":"listingExist","outputs":[{"name":"exists_","type":"bool"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_sponsor","type":"address"},{"name":"_owner","type":"address"},{"name":"_filehash","type":"bytes32"},{"name":"_previewUrl","type":"string"},{"name":"_url","type":"string"},{"name":"_pointWKT","type":"string"},{"name":"_geohash","type":"bytes12"},{"name":"_metadata","type":"string"}],"name":"sponsorCreateListing","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_address","type":"address"}],"name":"setSoarStorageContract","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"inputs":[],"payable":false,"stateMutability":"nonpayable","type":"constructor"},{"anonymous":false,"inputs":[{"indexed":true,"name":"sponsor","type":"address"},{"indexed":false,"name":"admin","type":"address"}],"name":"SponsorAdminAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"sponsor","type":"address"},{"indexed":false,"name":"admin","type":"address"}],"name":"SponsorAdminRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"account","type":"address"}],"name":"AdminAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"account","type":"address"}],"name":"AdminRemoved","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"account","type":"address"}],"name":"Paused","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"account","type":"address"}],"name":"Unpaused","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"account","type":"address"}],"name":"PauserAdded","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"account","type":"address"}],"name":"PauserRemoved","type":"event"}]