pragma solidity ^0.4.20;
library SafeMath {
function mul(uint256 a, uint256 b) internal pure returns (uint256) {
if (a == 0) {
return 0;
}
uint256 c = a * b;
assert(c / a == b);
return c;
}
function div(uint256 a, uint256 b) internal pure returns (uint256) {
// assert(b > 0); // Solidity automatically throws when dividing by 0
uint256 c = a / b;
// assert(a == b * c + a % b); // There is no case in which this doesn't hold
return c;
}
function sub(uint256 a, uint256 b) internal pure returns (uint256) {
assert(b <= a);
return a - b;
}
function add(uint256 a, uint256 b) internal pure returns (uint256) {
uint256 c = a + b;
assert(c >= a);
return c;
}
}
contract Ownable {
address public coOwner;
function Ownable() public {
coOwner = msg.sender;
}
modifier onlyCoOwner() {
require(msg.sender == coOwner);
_;
}
function transferCoOwnership(address _newOwner) public onlyCoOwner {
require(_newOwner != address(0));
coOwner = _newOwner;
CoOwnershipTransferred(coOwner, _newOwner);
}
function CoWithdraw() public onlyCoOwner {
coOwner.transfer(this.balance);
}
event CoOwnershipTransferred(address indexed previousOwner, address indexed newOwner);
}
/// @title Interface for contracts conforming to ERC-721: Non-Fungible Tokens
/// @author Dieter Shirley <dete@axiomzen.co> (https://github.com/dete)
contract ERC721 {
// Required methods
function approve(address _to, uint256 _tokenId) public;
function balanceOf(address _owner) public view returns (uint256 balance);
function implementsERC721() public pure returns (bool);
function ownerOf(uint256 _tokenId) public view returns (address addr);
function takeOwnership(uint256 _tokenId) public;
function totalSupply() public view returns (uint256 total);
function transferFrom(address _from, address _to, uint256 _tokenId) public;
function transfer(address _to, uint256 _tokenId) public;
event Transfer(address indexed from, address indexed to, uint256 tokenId);
event Approval(address indexed owner, address indexed approved, uint256 tokenId);
// Optional
// function name() public view returns (string name);
// function symbol() public view returns (string symbol);
// function tokenOfOwnerByIndex(address _owner, uint256 _index) external view returns (uint256 tokenId);
// function tokenMetadata(uint256 _tokenId) public view returns (string infoUrl);
}
contract CryptoCarsRent is ERC721, Ownable {
event CarCreated(uint256 tokenId, string name, address owner);
event TokenSold(uint256 tokenId, uint256 oldPrice, uint256 newPrice, address prevOwner, address winner, string name);
event Transfer(address from, address to, uint256 tokenId);
string public constant NAME = "CryptoCars";
string public constant SYMBOL = "CarsToken";
uint256 private startingSellPrice = 0.012 ether;
mapping (uint256 => address) public carIdToOwner;
mapping (uint256 => address) public carIdToRenter;
mapping (uint256 => uint256) public carIdRentStartTime;
mapping (address => uint256) private ownershipTokenCount;
mapping (uint256 => address) public carIdToApproved;
mapping (uint256 => uint256) private carIdToPrice;
mapping (uint256 => uint256) private carIdToProfit;
/*** DATATYPES ***/
struct Car {
string name;
}
Car[] private cars;
function approve(address _to, uint256 _tokenId) public { //ERC721
// Caller must own token.
require(_owns(msg.sender, _tokenId));
carIdToApproved[_tokenId] = _to;
Approval(msg.sender, _to, _tokenId);
}
function balanceOf(address _owner) public view returns (uint256 balance) { //ERC721
return ownershipTokenCount[_owner];
}
function createCarToken(string _name) public onlyCoOwner {
_createCar(_name, address(this), startingSellPrice);
}
function createCarsTokens() public onlyCoOwner {
for (uint8 car=0; car<21; car++) {
_createCar("Crypto Car", address(this), startingSellPrice);
}
}
function getCar(uint256 _tokenId) public view returns (string carName, uint256 sellingPrice, address owner) {
Car storage car = cars[_tokenId];
carName = car.name;
sellingPrice = carIdToPrice[_tokenId];
owner = carIdToOwner[_tokenId];
}
function implementsERC721() public pure returns (bool) {
return true;
}
function name() public pure returns (string) { //ERC721
return NAME;
}
function ownerOf(uint256 _tokenId) public view returns (address owner) { //ERC721
owner = carIdToOwner[_tokenId];
require(owner != address(0));
}
// Allows someone to send ether and obtain the token
function purchase(uint256 _tokenId) public payable {
address oldOwner = carIdToOwner[_tokenId];
address newOwner = msg.sender;
uint256 renter_payment;
uint256 payment;
if (now - carIdRentStartTime[_tokenId] > 7200) // 2 hours of rent finished
carIdToRenter[_tokenId] = address(0);
address renter = carIdToRenter[_tokenId];
uint256 sellingPrice = carIdToPrice[_tokenId];
uint256 profit = carIdToProfit[_tokenId];
require(oldOwner != newOwner);
require(_addressNotNull(newOwner));
require(msg.value >= sellingPrice);
if (renter != address(0)) {
renter_payment = uint256(SafeMath.div(SafeMath.mul(profit, 45), 100)); //45% from profit to car's renter
payment = uint256(SafeMath.sub(SafeMath.div(SafeMath.mul(sellingPrice, 97), 100), renter_payment)); //'97% - renter_payment' to previous owner
} else {
renter_payment = 0;
payment = uint256(SafeMath.div(SafeMath.mul(sellingPrice, 94), 100)); //94% to previous owner
}
// Next price will in 2 times more.
if (sellingPrice < 500 finney) {
carIdToPrice[_tokenId] = SafeMath.mul(sellingPrice, 2); //rice by 100%
}
else {
carIdToPrice[_tokenId] = uint256(SafeMath.div(SafeMath.mul(sellingPrice, 15), 10)); //rice by 50%
}
//plannig profit from next selling
carIdToProfit[_tokenId] = uint256(SafeMath.sub(carIdToPrice[_tokenId], sellingPrice));
carIdToRenter[_tokenId] = address(0);
carIdRentStartTime[_tokenId] = 0;
_transfer(oldOwner, newOwner, _tokenId);
// Pay previous tokenOwner if owner is not contract
if (oldOwner != address(this)) {
oldOwner.transfer(payment); //
}
// Pay to token renter
if (renter != address(0)) {
renter.transfer(renter_payment); //
}
TokenSold(_tokenId, sellingPrice, carIdToPrice[_tokenId], oldOwner, newOwner, cars[_tokenId].name);
if (msg.value > sellingPrice) { //if excess pay
uint256 purchaseExcess = SafeMath.sub(msg.value, sellingPrice);
msg.sender.transfer(purchaseExcess);
}
}
function rent(uint256 _tokenId) public payable {
require(now - carIdRentStartTime[_tokenId] > 7200); // 2 hours of previous rent finished
require(msg.sender != carIdToOwner[_tokenId]);
uint256 profit = carIdToProfit[_tokenId]; //plannig profit from selling
uint256 rentPrice = uint256(SafeMath.div(SafeMath.mul(profit, 10), 100)); //10% from profit is a rent price
require(_addressNotNull(msg.sender));
require(msg.value >= rentPrice);
carIdRentStartTime[_tokenId] = now;
carIdToRenter[_tokenId] = msg.sender;
address carOwner = carIdToOwner[_tokenId];
require(carOwner != address(this));
if (carOwner != address(this)) {
carOwner.transfer(rentPrice); //
}
if (msg.value > rentPrice) { //if excess pay
uint256 purchaseExcess = SafeMath.sub(msg.value, rentPrice);
msg.sender.transfer(purchaseExcess);
}
}
function symbol() public pure returns (string) { //ERC721
return SYMBOL;
}
function takeOwnership(uint256 _tokenId) public { //ERC721
address newOwner = msg.sender;
address oldOwner = carIdToOwner[_tokenId];
require(_addressNotNull(newOwner));
require(_approved(newOwner, _tokenId));
_transfer(oldOwner, newOwner, _tokenId);
}
function allCarsInfo() public view returns (address[] owners, address[] renters, uint256[] prices, uint256[] profits) { //for web site view
uint256 totalResultCars = totalSupply();
if (totalResultCars == 0) {
// Return an empty array
return (new address[](0),new address[](0),new uint256[](0),new uint256[](0));
}
address[] memory owners_res = new address[](totalResultCars);
address[] memory renters_res = new address[](totalResultCars);
uint256[] memory prices_res = new uint256[](totalResultCars);
uint256[] memory profits_res = new uint256[](totalResultCars);
for (uint256 carId = 0; carId < totalResultCars; carId++) {
owners_res[carId] = carIdToOwner[carId];
if (now - carIdRentStartTime[carId] <= 7200) // 2 hours of rent finished
renters_res[carId] = carIdToRenter[carId];
else
renters_res[carId] = address(0);
prices_res[carId] = carIdToPrice[carId];
profits_res[carId] = carIdToProfit[carId];
}
return (owners_res, renters_res, prices_res, profits_res);
}
function priceOf(uint256 _tokenId) public view returns (uint256 price) { //for web site view
return carIdToPrice[_tokenId];
}
function tokensOfOwner(address _owner) public view returns(uint256[] ownerTokens) { //for web site view
uint256 tokenCount = balanceOf(_owner);
if (tokenCount == 0) {
// Return an empty array
return new uint256[](0);
} else {
uint256[] memory result = new uint256[](tokenCount);
uint256 totalCars = totalSupply();
uint256 resultIndex = 0;
uint256 carId;
for (carId = 0; carId <= totalCars; carId++) {
if (carIdToOwner[carId] == _owner) {
result[resultIndex] = carId;
resultIndex++;
}
}
return result;
}
}
function totalSupply() public view returns (uint256 total) { //ERC721
return cars.length;
}
function transfer(address _to, uint256 _tokenId) public { //ERC721
require(_owns(msg.sender, _tokenId));
require(_addressNotNull(_to));
_transfer(msg.sender, _to, _tokenId);
}
function transferFrom(address _from, address _to, uint256 _tokenId) public { //ERC721
require(_owns(_from, _tokenId));
require(_approved(_to, _tokenId));
require(_addressNotNull(_to));
_transfer(_from, _to, _tokenId);
}
/* PRIVATE FUNCTIONS */
function _addressNotNull(address _to) private pure returns (bool) {
return _to != address(0);
}
function _approved(address _to, uint256 _tokenId) private view returns (bool) {
return carIdToApproved[_tokenId] == _to;
}
function _createCar(string _name, address _owner, uint256 _price) private {
Car memory _car = Car({
name: _name
});
uint256 newCarId = cars.push(_car) - 1;
require(newCarId == uint256(uint32(newCarId))); //check maximum limit of tokens
CarCreated(newCarId, _name, _owner);
carIdToPrice[newCarId] = _price;
_transfer(address(0), _owner, newCarId);
}
function _owns(address _checkedAddr, uint256 _tokenId) private view returns (bool) {
return _checkedAddr == carIdToOwner[_tokenId];
}
function _transfer(address _from, address _to, uint256 _tokenId) private {
ownershipTokenCount[_to]++;
carIdToOwner[_tokenId] = _to;
// When creating new cars _from is 0x0, but we can't account that address.
if (_from != address(0)) {
ownershipTokenCount[_from]--;
// clear any previously approved ownership exchange
delete carIdToApproved[_tokenId];
}
// Emit the transfer event.
Transfer(_from, _to, _tokenId);
}
}
{
"compilationTarget": {
"CryptoCarsRent.sol": "CryptoCarsRent"
},
"libraries": {},
"optimizer": {
"enabled": false,
"runs": 200
},
"remappings": []
}
[{"constant":true,"inputs":[],"name":"name","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":false,"inputs":[{"name":"_name","type":"string"}],"name":"createCarToken","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_to","type":"address"},{"name":"_tokenId","type":"uint256"}],"name":"approve","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"implementsERC721","outputs":[{"name":"","type":"bool"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":true,"inputs":[{"name":"","type":"uint256"}],"name":"carIdToOwner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"totalSupply","outputs":[{"name":"total","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_from","type":"address"},{"name":"_to","type":"address"},{"name":"_tokenId","type":"uint256"}],"name":"transferFrom","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[],"name":"allCarsInfo","outputs":[{"name":"owners","type":"address[]"},{"name":"renters","type":"address[]"},{"name":"prices","type":"uint256[]"},{"name":"profits","type":"uint256[]"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"uint256"}],"name":"carIdToRenter","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"uint256"}],"name":"carIdRentStartTime","outputs":[{"name":"","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"_tokenId","type":"uint256"}],"name":"getCar","outputs":[{"name":"carName","type":"string"},{"name":"sellingPrice","type":"uint256"},{"name":"owner","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"_tokenId","type":"uint256"}],"name":"ownerOf","outputs":[{"name":"owner","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[],"name":"CoWithdraw","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_owner","type":"address"}],"name":"balanceOf","outputs":[{"name":"balance","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_tokenId","type":"uint256"}],"name":"rent","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":true,"inputs":[{"name":"_owner","type":"address"}],"name":"tokensOfOwner","outputs":[{"name":"ownerTokens","type":"uint256[]"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"symbol","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"pure","type":"function"},{"constant":true,"inputs":[],"name":"NAME","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_to","type":"address"},{"name":"_tokenId","type":"uint256"}],"name":"transfer","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[],"name":"createCarsTokens","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_tokenId","type":"uint256"}],"name":"takeOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":true,"inputs":[{"name":"_tokenId","type":"uint256"}],"name":"priceOf","outputs":[{"name":"price","type":"uint256"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[{"name":"","type":"uint256"}],"name":"carIdToApproved","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":true,"inputs":[],"name":"coOwner","outputs":[{"name":"","type":"address"}],"payable":false,"stateMutability":"view","type":"function"},{"constant":false,"inputs":[{"name":"_newOwner","type":"address"}],"name":"transferCoOwnership","outputs":[],"payable":false,"stateMutability":"nonpayable","type":"function"},{"constant":false,"inputs":[{"name":"_tokenId","type":"uint256"}],"name":"purchase","outputs":[],"payable":true,"stateMutability":"payable","type":"function"},{"constant":true,"inputs":[],"name":"SYMBOL","outputs":[{"name":"","type":"string"}],"payable":false,"stateMutability":"view","type":"function"},{"anonymous":false,"inputs":[{"indexed":false,"name":"tokenId","type":"uint256"},{"indexed":false,"name":"name","type":"string"},{"indexed":false,"name":"owner","type":"address"}],"name":"CarCreated","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"tokenId","type":"uint256"},{"indexed":false,"name":"oldPrice","type":"uint256"},{"indexed":false,"name":"newPrice","type":"uint256"},{"indexed":false,"name":"prevOwner","type":"address"},{"indexed":false,"name":"winner","type":"address"},{"indexed":false,"name":"name","type":"string"}],"name":"TokenSold","type":"event"},{"anonymous":false,"inputs":[{"indexed":false,"name":"from","type":"address"},{"indexed":false,"name":"to","type":"address"},{"indexed":false,"name":"tokenId","type":"uint256"}],"name":"Transfer","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"previousOwner","type":"address"},{"indexed":true,"name":"newOwner","type":"address"}],"name":"CoOwnershipTransferred","type":"event"},{"anonymous":false,"inputs":[{"indexed":true,"name":"owner","type":"address"},{"indexed":true,"name":"approved","type":"address"},{"indexed":false,"name":"tokenId","type":"uint256"}],"name":"Approval","type":"event"}]