monerowp/monero/include/monero_payments.php

415 lines
18 KiB
PHP
Raw Normal View History

2017-06-18 20:53:44 +02:00
<?php
2017-06-27 14:55:02 +02:00
/* Main Gateway of Monero using a daemon online */
2017-07-13 15:04:55 +02:00
class Monero_Gateway extends WC_Payment_Gateway
{
private $reloadTime = 30000;
private $discount;
private $monero_daemon;
2017-07-13 15:04:55 +02:00
function __construct()
{
$this->id = "monero_gateway";
$this->method_title = __("Monero GateWay", 'monero_gateway');
2017-08-13 15:25:39 +02:00
$this->method_description = __("Monero Payment Gateway Plug-in for WooCommerce. You can find more information about this payment gateway in our website. You'll need a daemon online for your address.", 'monero_gateway');
2017-07-13 15:04:55 +02:00
$this->title = __("Monero Gateway", 'monero_gateway');
2017-07-27 13:59:38 +02:00
$this->version = "0.2";
2017-07-13 15:04:55 +02:00
//
$this->icon = apply_filters('woocommerce_offline_icon', '');
$this->has_fields = false;
2017-07-26 21:52:48 +02:00
$this->log = new WC_Logger();
2017-07-13 15:04:55 +02:00
$this->init_form_fields();
2017-07-15 10:09:48 +02:00
$this->host = $this->get_option('daemon_host');
$this->port = $this->get_option('daemon_port');
2017-07-13 15:04:55 +02:00
$this->address = $this->get_option('monero_address');
$this->username = $this->get_option('username');
$this->password = $this->get_option('password');
$this->discount = $this->get_option('discount');
2017-07-13 15:04:55 +02:00
// After init_settings() is called, you can get the settings and load them into variables, e.g:
// $this->title = $this->get_option('title' );
$this->init_settings();
// Turn these settings into variables we can use
foreach ($this->settings as $setting_key => $value) {
$this->$setting_key = $value;
}
2017-08-02 21:14:20 +02:00
add_action('admin_notices', array($this,'do_ssl_check'));
add_action('admin_notices', array($this,'validate_fields'));
add_action('woocommerce_thankyou_' . $this->id, array($this,'instruction'));
2017-07-13 15:04:55 +02:00
if (is_admin()) {
/* Save Settings */
2017-08-02 21:14:20 +02:00
add_action('woocommerce_update_options_payment_gateways_' . $this->id, array($this,'process_admin_options'
2017-07-13 15:04:55 +02:00
));
2017-08-13 15:25:39 +02:00
add_filter( 'woocommerce_currencies', array($this,'add_my_currency') );
add_filter('woocommerce_currency_symbol', 'add_my_currency_symbol', 10, 2);
2017-08-13 14:37:46 +02:00
add_action('woocommerce_email_before_order_table', array($this, 'email_instructions'), 10, 2);
2017-07-13 15:04:55 +02:00
}
$this->monero_daemon = new Monero_Library($this->host . ':' . $this->port . '/json_rpc', $this->username, $this->password);
2017-07-13 15:04:55 +02:00
}
2017-08-13 15:25:39 +02:00
public function add_my_currency( $currencies ) {
$currencies['XMR'] = __('Monero','woocommerce');
2017-08-13 14:37:46 +02:00
return $currencies;
}
2017-08-13 15:25:39 +02:00
public function add_my_currency_symbol( $currency_symbol, $currency ) {
2017-08-13 14:37:46 +02:00
switch( $currency ) {
2017-08-13 15:25:39 +02:00
case 'XMR': $currency_symbol = 'XMR'; break;
2017-08-13 14:37:46 +02:00
}
return $currency_symbol;
}
2017-07-13 15:04:55 +02:00
public function admin_options()
{
2017-07-27 14:17:23 +02:00
$this->log->add('Monero_gateway', '[SUCCESS] Monero Settings OK');
2017-08-13 14:37:46 +02:00
2017-07-13 15:04:55 +02:00
echo "<h1>Monero Payment Gateway</h1>";
2017-08-13 15:25:39 +02:00
2017-07-15 10:09:48 +02:00
echo "<p>Welcome to Monero Extension for WooCommerce. Getting started: Make a connection with daemon <a href='https://reddit.com/u/serhack'>Contact Me</a>";
2017-08-13 15:25:39 +02:00
echo "<div style='border:1px solid #DDD;padding:5px 10px;font-weight:bold;color:#223079;background-color:#9ddff3;'>";
$this->getamountinfo();
echo "</div>";
2017-07-13 15:04:55 +02:00
echo "<table class='form-table'>";
$this->generate_settings_html();
echo "</table>";
echo "<h4>Learn more about using a password with the monero wallet-rpc <a href=\"https://github.com/cryptochangements34/monerowp/blob/master/README.md\">here</a></h4>";
2017-07-13 15:04:55 +02:00
}
public function init_form_fields()
{
$this->form_fields = array(
'enabled' => array(
'title' => __('Enable / Disable', 'monero_gateway'),
'label' => __('Enable this payment gateway', 'monero_gateway'),
'type' => 'checkbox',
'default' => 'no'
),
'title' => array(
'title' => __('Title', 'monero_gateway'),
'type' => 'text',
'desc_tip' => __('Payment title the customer will see during the checkout process.', 'monero_gateway'),
'default' => __('Monero XMR Payment', 'monero_gateway')
),
'description' => array(
'title' => __('Description', 'monero_gateway'),
'type' => 'textarea',
'desc_tip' => __('Payment description the customer will see during the checkout process.', 'monero_gateway'),
'default' => __('Pay securely using XMR.', 'monero_gateway')
),
'monero_address' => array(
'title' => __('Monero Address', 'monero_gateway'),
'label' => __('Useful for people that have not a daemon online'),
'type' => 'text',
'desc_tip' => __('Monero Wallet Address', 'monero_gateway')
),
'daemon_host' => array(
2017-07-13 15:04:55 +02:00
'title' => __('Daemon Host/ IP', 'monero_gateway'),
'type' => 'text',
'desc_tip' => __('This is the Daemon Host/IP to authorize the payment with port', 'monero_gateway'),
'default' => 'localhost',
),
'daemon_port' => array(
'title' => __('Daemon PORT', 'monero_gateway'),
'type' => 'text',
'desc_tip' => __('This is the Daemon Host/IP to authorize the payment with port', 'monero_gateway'),
2017-07-15 10:09:48 +02:00
'default' => '18080',
),
'username' => array(
'title' => __('Daemon username', 'monero_gateway'),
'desc_tip' => __('This is the username that you used with your monero wallet-rpc', 'monero_gateway'),
'type' => __('text'),
2017-07-24 17:20:46 +02:00
'default' => __('username','monero_gateway'),
),
'password' => array(
'title' => __('Daemon password', 'monero_gateway'),
'desc_tip' => __('This is the password that you used with your monero wallet-rpc', 'monero_gateway'),
'description' => __('you can leave these fields empty if you did not set', 'monero_gateway'),
'type' => __('text'),
2017-07-24 17:20:46 +02:00
'default' => ''
),
'discount' => array(
'title' => __('% discount for using XMR', 'monero_gateway'),
2017-08-13 20:26:51 +02:00
'desc_tip' => __('Provide a descount to your customers for paying privatly with XMR!', 'monero_gateway'),
2017-08-12 07:56:00 +02:00
'description' => __(' Want to spread the word about Monero? Offer a little discount! Leave this empty if you do not wish to provide a discount', 'monero_gateway'),
2017-08-13 20:26:51 +02:00
'type' => __('text'),
'default' => '5%'
),
2017-07-13 15:04:55 +02:00
'environment' => array(
'title' => __(' Test Mode', 'monero_gateway'),
'label' => __('Enable Test Mode', 'monero_gateway'),
'type' => 'checkbox',
'description' => __('Check this box if you are using testnet', 'monero_gateway'),
2017-07-13 15:04:55 +02:00
'default' => 'no'
),
2017-07-13 15:04:55 +02:00
);
}
public function retriveprice($currency)
{
2017-07-27 13:59:38 +02:00
$xmr_price = file_get_contents('https://min-api.cryptocompare.com/data/price?fsym=XMR&tsyms=BTC,USD,EUR,CAD,INR,GBP&extraParams=monero_woocommerce');
$price = json_decode($xmr_price, TRUE);
if(!isset($price)){
2017-07-27 14:17:23 +02:00
$this->log->add('Monero_Gateway', '[ERROR] Unable to get the price of Monero');
2017-07-26 21:52:48 +02:00
}
2017-07-13 15:04:55 +02:00
if ($currency == 'USD') {
2017-07-27 13:59:38 +02:00
return $price['USD'];
2017-07-13 15:04:55 +02:00
}
if ($currency == 'EUR') {
2017-07-27 13:59:38 +02:00
return $price['EUR'];
}
if ($currency == 'CAD'){
return $price['CAD'];
}
if ($currency == 'GBP'){
return $price['GBP'];
}
if ($currency == 'INR'){
return $price['INR'];
2017-07-13 15:04:55 +02:00
}
2017-08-13 14:37:46 +02:00
if($currency == 'XMR'){
$price = '1';
return $price;
}
2017-07-13 15:04:55 +02:00
}
public function changeto($amount, $currency, $payment_id)
2017-07-13 15:04:55 +02:00
{
global $wpdb;
// This will create a table named whatever the payment id is inside the database "WordPress"
$create_table = "CREATE TABLE IF NOT EXISTS $payment_id (
rate INT
)";
$wpdb->query($create_table);
$rows_num = $wpdb->get_results("SELECT count(*) as count FROM $payment_id");
if($rows_num[0]->count > 0) // Checks if the row has already been created or not
{
$stored_rate = $wpdb->get_results("SELECT rate FROM $payment_id");
$stored_rate_transformed = $stored_rate[0]->rate / 100; //this will turn the stored rate back into a decimaled number
if(isset($this->discount))
{
$discount_decimal = $this->discount / 100;
$new_amount = $amount / $stored_rate_transformed;
$discount = $new_amount * $discount_decimal;
$final_amount = $new_amount - $discount;
$rounded_amount = round($final_amount, 12);
}
else{
$new_amount = $amount / $stored_rate_transformed;
$rounded_amount = round($new_amount, 12); //the moneo wallet can't handle decimals smaller than 0.000000000001
}
}
else // If the row has not been created then the live exchange rate will be grabbed and stored
{
$xmr_live_price = $this->retriveprice($currency);
$live_for_storing = $xmr_live_price * 100; //This will remove the decimal so that it can easily be stored as an integer
$new_amount = $amount / $xmr_live_price;
$rounded_amount = round($new_amount, 12);
$wpdb->query("INSERT INTO $payment_id (rate)
VALUES ($live_for_storing)");
}
return $rounded_amount;
2017-07-13 15:04:55 +02:00
}
// Submit payment and handle response
public function process_payment($order_id)
{
$order = wc_get_order($order_id);
$order->update_status('on-hold', __('Awaiting offline payment', 'monero_gateway'));
// Reduce stock levels
$order->reduce_order_stock();
// Remove cart
WC()->cart->empty_cart();
// Return thankyou redirect
return array(
'result' => 'success',
'redirect' => $this->get_return_url($order)
);
}
// Validate fields
public function validate_fields()
{
if ($this->check_monero() != TRUE) {
echo "<div class=\"error\"><p>Your Monero Address Seems not valid. Have you checked it?</p></div>";
}
}
public function check_monero()
{
$monero_address = $this->settings['monero_address'];
if (strlen($monero_address) == 95 && substr($monero_address, 1)) {
return true;
}
return false;
}
2017-07-13 15:04:55 +02:00
private function set_paymentid_cookie()
{
if(!isset($_COOKIE['payment_id']))
{
$payment_id = bin2hex(openssl_random_pseudo_bytes(8));
setcookie('payment_id', $payment_id, time()+2700);
}
else
$payment_id = $_COOKIE['payment_id'];
return $payment_id;
}
2017-07-13 15:04:55 +02:00
public function instruction($order_id)
{
$order = wc_get_order($order_id);
2017-08-13 14:37:46 +02:00
$amount = floatval(preg_replace('#[^\d.]#', '', $order->get_total()));
$payment_id = $this->set_paymentid_cookie();
2017-08-13 14:37:46 +02:00
$currency = $order->get_currency();
$amount_xmr2 = $this->changeto($amount, $currency, $payment_id);
2017-07-13 15:04:55 +02:00
$address = $this->address;
if(!isset($address)){
2017-08-13 15:25:39 +02:00
// If there isn't address (merchant missed that field!), $address will be the Monero address for donating :)
$address = "44AFFq5kSiGBoZ4NMDwYtN18obc8AemS33DBLWs3H7otXft3XjrpDtQGv7SqSsaBYBb98uNbr2VBBEt7f2wfn3RVGQBEP3A";
2017-08-13 14:37:46 +02:00
}
2017-07-15 13:32:34 +02:00
$uri = "monero:$address?amount=$amount?payment_id=$payment_id";
2017-07-26 18:47:25 +02:00
$array_integrated_address = $this->monero_daemon->make_integrated_address($payment_id);
if(!isset($array_integrated_address)){
2017-08-13 15:25:39 +02:00
$this->log->add('Monero_Gateway', '[ERROR] Unable to getting integrated address');
// Seems that we can't connect with daemon, then set array_integrated_address, little hack
$array_integrated_address["integrated_address"] = $address;
2017-07-26 21:52:48 +02:00
}
$message = $this->verify_payment($payment_id, $amount_xmr2, $order);
echo "<h4>".$message."</h4>";
2017-08-03 15:47:06 +02:00
echo "<link rel='stylesheet' href='https://maxcdn.bootstrapcdn.com/bootstrap/3.3.7/css/bootstrap.min.css'>";
2017-07-13 15:04:55 +02:00
echo "<div class='row'>
<div class='col-sm-12 col-md-12 col-lg-12'>
2017-08-02 21:14:20 +02:00
<div class='panel panel-default' id='PaymentBox_de3a227fb470475'>
<div class='panel-body'>
<div class='row'>
<div class='col-sm-12 col-md-12 col-lg-12'>
2017-08-13 15:25:39 +02:00
<h3> Monero Payment Box</h3>
2017-08-02 21:14:20 +02:00
</div>
<div class='col-sm-3 col-md-3 col-lg-3'>
2017-08-13 14:37:46 +02:00
<img src='https://chart.googleapis.com/chart?cht=qr&chs=250x250&chl=" . $uri . "' class='img-responsive'>
2017-08-02 21:14:20 +02:00
</div>
<div class='col-sm-9 col-md-9 col-lg-9' style='padding:10px;'>
Send <b>" . $amount_xmr2 . " XMR</b> to<br/><input type='text' class='form-control' value='" . $array_integrated_address["integrated_address"]."'>
2017-08-02 21:14:20 +02:00
or scan QR Code with your mobile device<br/><br/>
2017-08-13 15:25:39 +02:00
<small>If you don't know how to pay with monero or you don't know what monero is, please go <a href='#'>here</a>. </small>
2017-08-02 21:14:20 +02:00
</div>
<div class='col-sm-12 col-md-12 col-lg-12'>
2017-06-29 16:40:47 +02:00
2017-08-13 15:25:39 +02:00
2017-08-02 21:14:20 +02:00
</div>
</div>
</div>
2017-08-13 15:25:39 +02:00
2017-08-02 21:14:20 +02:00
</div>
2017-06-29 16:40:47 +02:00
</div>
2017-08-02 21:14:20 +02:00
</div>
2017-06-29 16:40:47 +02:00
<script type='text/javascript'>
setTimeout(function () { location.reload(true); }, $this->reloadTime);
2017-08-03 15:47:06 +02:00
</script>";
2017-07-13 15:04:55 +02:00
}
// Check if we are forcing SSL on checkout pages
// Custom function not required by the Gateway
public function do_ssl_check()
{
if ($this->enabled == "yes") {
if (get_option('woocommerce_force_ssl_checkout') == "no") {
echo "<div class=\"error\"><p>" . sprintf(__("<strong>%s</strong> is enabled and WooCommerce is not forcing the SSL certificate on your checkout page. Please ensure that you have a valid SSL certificate and that you are <a href=\"%s\">forcing the checkout pages to be secured.</a>"), $this->method_title, admin_url('admin.php?page=wc-settings&tab=checkout')) . "</p></div>";
}
}
}
2017-07-15 10:09:48 +02:00
public function connect_daemon(){
$host = $this->settings['daemon_host'];
$port = $this->settings['daemon_port'];
$monero_library = new Monero($host, $port);
if( $monero_library->works() == true){
echo "<div class=\"notice notice-success is-dismissible\"><p>Everything works! Congratulations and Welcome aboard Monero. <button type=\"button\" class=\"notice-dismiss\">
<span class=\"screen-reader-text\">Dismiss this notice.</span>
</button></p></div>";
}
else{
2017-07-27 14:17:23 +02:00
$this->log->add('Monero_gateway','[ERROR] Plugin can not reach wallet rpc.');
2017-07-15 10:09:48 +02:00
echo "<div class=\" notice notice-error\"><p>Error with connection of daemon, see documentation!</p></div>";
2017-07-16 11:15:43 +02:00
} }
2017-07-15 22:02:33 +02:00
public function verify_payment($payment_id, $amount, $order_id){
2017-07-15 22:02:33 +02:00
/*
2017-08-03 15:47:06 +02:00
* function for verifying payments
* Check if a payment has been made with this payment id then notify the merchant
2017-08-03 15:47:06 +02:00
*/
2017-07-15 10:09:48 +02:00
$amount_atomic_units = $amount * 1000000000000;
$get_payments_method = $this->monero_daemon->get_payments($payment_id);
if(isset($get_payments_method["payments"][0]["amount"]))
{
if($get_payments_method["payments"][0]["amount"] >= $amount_atomic_units)
{
$message = "Payment has been received and confirmed. Thanks!";
$this->log->add('Monero_gateway','[SUCCESS] Payment has been recorded. Congrats!');
$paid = true;
$order = wc_get_order($order_id);
$order->update_status('completed', __('Payment has been received', 'monero_gateway'));
global $wpdb;
$wpdb->query("DROP TABLE $payment_id"); // Drop the table from database after payment has been confirmed as it is no longer needed
$this->reloadTime = 3000000000000; // Greatly increase the reload time as it is no longer needed
}
}
else
{
$message = "We are waiting for your payment to be confirmed";
}
return $message;
}
public function getamountinfo(){
$wallet_amount = $this->monero_daemon->getbalance();
if(!isset($wallet_amount)){
2017-08-14 10:52:50 +02:00
$this->log->add('Monero_gateway','[ERROR] Connection with daemon absend');
$wallet_amount['balance'] = "0";
$wallet_amount['unlocked_balance'] = "0";
}
$real_wallet_amount = $wallet_amount['balance'] / 1000000000000;
$real_amount_rounded = round($real_wallet_amount, 6);
2017-08-13 15:25:39 +02:00
$unlocked_wallet_amount = $wallet_amount['unlocked_balance'] / 1000000000000;
$unlocked_amount_rounded = round($unlocked_wallet_amount, 6);
echo "Your balance is: ".$real_amount_rounded. " XMR </br>";
echo "Unlocked balance: ".$unlocked_amount_rounded." XMR </br>";
}
}