File: /home/retile.ru/public_html/system/library/openbay/etsy.php
<?php
namespace openbay;
final class Etsy {
	private $token;
    private $encryption_key;
    private $encryption_iv;
	private $url = 'https://api.openbaypro.io/';
	private $registry;
	private $logger;
	private $max_log_size = 50; //max log size in Mb
	public function __construct($registry) {
		$this->registry = $registry;
		$this->token = $this->config->get('etsy_token');
		$this->logging = $this->config->get('etsy_logging');
		if ($this->logging == 1) {
			$this->setLogger();
		}
		$this->setEncryptionKey($this->config->get('etsy_encryption_key'));
		$this->setEncryptionIv($this->config->get('etsy_encryption_iv'));
	}
	public function __get($name) {
		return $this->registry->get($name);
	}
    public function getEncryptionKey() {
        return $this->encryption_key;
    }
	public function setEncryptionKey($key) {
	    $this->encryption_key = $key;
    }
    public function getEncryptionIv() {
        return $this->encryption_iv;
    }
    public function setEncryptionIv($encryption_iv) {
        $this->encryption_iv = $encryption_iv;
    }
	public function resetConfig($token, $encryption_key) {
		$this->token = $token;
		$this->setEncryptionKey($encryption_key);
	}
	public function call($uri, $method, $data = array()) {
		if($this->config->get('etsy_status') == 1) {
			$headers = array ();
			$headers[] = 'X-Auth-Token: ' . $this->token;
			$headers[] = 'X-Endpoint-Version: 2';
			$headers[] = 'Content-Type: application/json';
			//$headers[] = 'Content-Length: '.strlen(json_encode($data));
			$defaults = array(
				CURLOPT_HEADER      	=> 0,
				CURLOPT_HTTPHEADER      => $headers,
				CURLOPT_URL             => $this->url . $uri,
				CURLOPT_USERAGENT       => "OpenBay Pro for Etsy/OpenCart",
				CURLOPT_FRESH_CONNECT   => 1,
				CURLOPT_RETURNTRANSFER  => 1,
				CURLOPT_FORBID_REUSE    => 1,
				CURLOPT_TIMEOUT         => 180,
				CURLOPT_SSL_VERIFYPEER  => 0,
				CURLOPT_SSL_VERIFYHOST  => 0,
				//CURLOPT_VERBOSE 		=> true,
				//CURLOPT_STDERR 			=> fopen(DIR_LOGS . 'curl_verbose.log', "w+")
			);
			if ($method == 'POST') {
				$defaults[CURLOPT_POST] = 1;
				$defaults[CURLOPT_POSTFIELDS] = json_encode($data);
			}
			$curl = curl_init();
			curl_setopt_array($curl, $defaults);
			$response = array();
			if (! $result = curl_exec($curl)) {
				$this->log('call() - Curl Failed ' . curl_error($curl) . ' ' . curl_errno($curl));
				return false;
			} else {
				$this->log('call() - Result of : "' . print_r($result, true) . '"');
				$encoding = mb_detect_encoding($result);
				if($encoding == 'UTF-8') {
					$result = preg_replace('/[^(\x20-\x7F)]*/', '', $result);
				}
				$result = json_decode($result, 1);
				$response['header_code'] = curl_getinfo($curl, CURLINFO_HTTP_CODE);
				if(!empty($result)) {
					$response['data'] = $result;
				} else {
					$response['data'] = '';
				}
			}
			curl_close($curl);
			return $response;
		} else {
			$this->log('call() - OpenBay Pro / Etsy not active');
			return false;
		}
	}
	private function setLogger() {
		if (file_exists(DIR_LOGS . 'etsylog.log')) {
			if (filesize(DIR_LOGS . 'etsylog.log') > ($this->max_log_size * 1000000)) {
				rename(DIR_LOGS . 'etsylog.log', DIR_LOGS . '_etsylog_' . date('Y-m-d_H-i-s') . '.log');
			}
		}
		$this->logger = new \Log('etsylog.log');
	}
	public function log($data, $write = true) {
		if ($this->logging == 1) {
			if (function_exists('getmypid')) {
				$process_id = getmypid();
				$data = $process_id . ' - ' . print_r($data, true);
			}
            $this->logger->write($data);
		}
	}
	public function getServer() {
		return $this->url;
	}
	public function settingsUpdate() {
		$this->log("settingsUpdate() - start");
		$response = $this->call('v1/etsy/data/type/getSetup/', 'GET');
		if (isset($response['data']) && is_array($response['data'])) {
			foreach ($response['data'] as $key => $options) {
				$this->db->query("DELETE FROM `" . DB_PREFIX . "etsy_setting_option` WHERE  `key` = '" . $this->db->escape($key) . "' LIMIT 1");
				$this->db->query("INSERT INTO `" . DB_PREFIX . "etsy_setting_option` SET `data` = '" . $this->db->escape(json_encode($options)) . "', `key` = '" . $this->db->escape($key) . "', `last_updated`  = now()");
				$this->log("settingsUpdate() - updated option: " . $key);
			}
			$this->log("settingsUpdate() - complete");
		} else {
			$this->log("settingsUpdate() - failed - no data response");
		}
	}
	public function getSetting($key) {
		$this->log("getSetting() - " . $key);
		$qry = $this->db->query("SELECT `data` FROM `" . DB_PREFIX . "etsy_setting_option` WHERE `key` = '" . $this->db->escape($key) . "' LIMIT 1");
		if($qry->num_rows > 0) {
			$this->log("getSetting() - Found setting");
			return json_decode($qry->row['data']);
		} else {
			return false;
		}
	}
	public function getLinks($product_id, $status = 0, $limit = null) {
		$this->log("getLinks() - Product_id: " . $product_id . " status: " . $status . " limit:" . $limit);
		if ($limit != null) {
			$sql_limit = ' LIMIT 1';
		} else {
			$sql_limit = '';
		}
		$qry = $this->db->query("SELECT `el`.*, `p`.`quantity` FROM `" . DB_PREFIX . "etsy_listing` `el` LEFT JOIN `" . DB_PREFIX . "product` `p` ON `el`.`product_id` = `p`.`product_id` WHERE `el`.`product_id` = '" . (int)$product_id . "' AND `el`.`status` = '" . (int)$status . "' ORDER BY `el`.`created` DESC" . $sql_limit);
		if ($qry->num_rows) {
			$this->log("getLinks() - " . $qry->num_rows . " found");
			$links = array();
			foreach ($qry->rows as $row) {
				$links[] = $row;
			}
			return $links;
		} else {
			$this->log("getLinks() - no links found");
			return false;
		}
	}
	public function getLinkedProduct($etsy_item_id) {
		$this->log("getLinkedProduct() - etsy_item_id: " . $etsy_item_id);
		$qry = $this->db->query("SELECT `p`.`quantity`, `p`.`product_id`, `p`.`model`, `el`.`etsy_listing_id`, `el`.`status` AS `link_status` FROM `" . DB_PREFIX . "etsy_listing` `el` LEFT JOIN `" . DB_PREFIX . "product` `p` ON `p`.`product_id` = `el`.`product_id` WHERE `el`.`etsy_item_id` = '" . (int)$etsy_item_id . "' AND `el`.`status` = 1");
		if($qry->num_rows) {
			$this->log("getLinkedProduct() - " . $qry->num_rows . " found");
			return $qry->row;
		} else {
			$this->log("getLinkedProduct() - no link found");
			return false;
		}
	}
	public function updateListingStock($etsy_item_id, $new_stock, $status) {
		$this->log("updateListingStock() - ItemID: " . $etsy_item_id . ", new stock: " . $new_stock . ", status: " . $status);
		if ($new_stock > 0) {
			$this->log("updateListingStock() - stock > 0 - update stock");
			if ($status == 'edit') {
				$status = 'inactive';
			}
			$response = $this->call('v1/etsy/product/listing/' . (int)$etsy_item_id . '/updateStock/', 'POST', array('quantity' => $new_stock, 'state' => $status));
			if (isset($response['data']['error'])) {
				return $response;
			} else {
				return true;
			}
		} else {
			$this->log("updateListingStock() - stock > 0 - set to inactive");
			$this->deleteLink(null, $etsy_item_id);
			$response = $this->call('v1/etsy/product/listing/' . (int)$etsy_item_id . '/inactive/', 'POST');
			if (isset($response['data']['error'])) {
				$this->log("updateListingStock() - Error: " . json_encode($response));
				return $response;
			} else {
				$this->log("updateListingStock() - Item ended OK");
				return true;
			}
		}
	}
	public function deleteProduct($product_id) {
		$this->log("deleteProduct() - Product ID: " . $product_id);
		$this->db->query("DELETE FROM `" . DB_PREFIX . "etsy_listing` WHERE `product_id` = '" . (int)$product_id . "'");
	}
	public function deleteLink($etsy_listing_id = null, $etsy_item_id = null) {
		$this->log("deleteLink() - Listing ID: " . $etsy_listing_id . ", item ID" . $etsy_item_id);
		if ($etsy_listing_id != null) {
			$this->log("deleteLink() - Listing ID is not null");
			$this->db->query("UPDATE `" . DB_PREFIX . "etsy_listing` SET `status` = 0 WHERE `etsy_listing_id` = '" . (int)$etsy_listing_id . "' LIMIT 1");
		} elseif ($etsy_item_id != null) {
			$this->log("deleteLink() - Item ID is not null");
			$this->db->query("UPDATE `" . DB_PREFIX . "etsy_listing` SET `status` = 0 WHERE `etsy_item_id` = '" . (int)$etsy_item_id . "' LIMIT 1");
		}
	}
	public function productUpdateListen($product_id, $data = array()) {
		$this->log("productUpdateListen() - " . $product_id . ", Data: " . json_encode($data));
		$links = $this->getLinks($product_id, 1);
		if (!empty($links)) {
			foreach ($links as $link) {
				$this->log("productUpdateListen() - Item ID: " . $link['etsy_item_id']);
				$etsy_listing = $this->getEtsyItem($link['etsy_item_id']);
				if ($etsy_listing != false && isset($etsy_listing['state']) && ($etsy_listing['state'] == 'active' || $etsy_listing['state'] == 'private' || $etsy_listing['state'] == 'draft' || $etsy_listing['state'] == 'edit')) {
					$this->log("productUpdateListen() - Listing state seems valid");
					if ($etsy_listing['quantity'] != $link['quantity']) {
						$this->log("productUpdateListen() - Stock is different, do update");
						$this->updateListingStock($link['etsy_item_id'], $link['quantity'], $etsy_listing['state']);
					} else {
						$this->log("productUpdateListen() - Stock is the same: " . $etsy_listing['quantity'] . " " . $link['quantity']);
					}
				} else {
					$this->log("productUpdateListen() - Listing state seems invalid");
					$this->log("productUpdateListen() - " . json_encode($etsy_listing));
					$this->deleteLink($link['etsy_listing_id']);
				}
			}
		} else {
			$this->log("productUpdateListen() - No links");
		}
	}
	public function orderFind($order_id = null, $receipt_id = null) {
		$this->log("orderFind() - OrderID: " . $order_id . ", Receipt ID: " . $receipt_id);
		if ($order_id != null) {
			$this->log("orderFind() - Order ID is not null");
			$query = $this->db->query("SELECT * FROM `" . DB_PREFIX . "etsy_order` WHERE `order_id` = '" . (int)$order_id . "' LIMIT 1");
			if($query->num_rows > 0) {
				$this->log('orderFind() - Found');
				return $query->row;
			} else {
				$this->log('orderFind() - Not found');
				return false;
			}
		} elseif ($receipt_id != null) {
			$this->log("orderFind() - Receipt ID is not null");
			$query = $this->db->query("SELECT * FROM `" . DB_PREFIX . "etsy_order` WHERE `receipt_id` = '" . (int)$receipt_id . "' LIMIT 1");
			if($query->num_rows > 0) {
				$this->log('orderFind() - Found');
				return $query->row;
			} else {
				$this->log('orderFind() - Not found');
				return false;
			}
		}
	}
	public function orderDelete($order_id) {
		$this->log("orderDelete() - ID: " . $order_id);
		if(!$this->orderFind($order_id)) {
			$query = $this->db->query("SELECT `p`.`product_id` FROM `" . DB_PREFIX . "order_product` `op` LEFT JOIN `" . DB_PREFIX . "product` `p` ON `op`.`product_id` = `p`.`product_id` WHERE `op`.`order_id` = '" . (int)$order_id . "'");
			if($query->num_rows > 0) {
				$this->log("orderDelete() - " . $query->num_rows . " products");
				foreach ($query->rows as $product) {
					$this->log("orderDelete() - Processing ID: " . $product['product_id']);
					$this->productUpdateListen((int)$product['product_id']);
				}
			} else {
				$this->log("orderDelete() - No products in order");
			}
		} else {
			$this->log("orderDelete() - Not an Etsy order");
		}
	}
	public function orderUpdatePaid($receipt_id, $status) {
		$this->log("orderUpdatePaid() - Receipt ID: " . $receipt_id . ", Status: " . $status);
		$response = $this->openbay->etsy->call('v1/etsy/order/update/payment/', 'POST', array('receipt_id' => $receipt_id, 'status' => $status));
		if (isset($response['data']['error'])) {
			$this->log("orderUpdatePaid() - Error: " . json_encode($response));
			return $response;
		} else {
			$this->log("orderUpdatePaid() - OK");
			return true;
		}
	}
	public function orderUpdateShipped($receipt_id, $status) {
		$this->log("orderUpdateShipped() - Receipt ID: " . $receipt_id . ", Status: " . $status);
		$response = $this->openbay->etsy->call('v1/etsy/order/update/shipping/', 'POST', array('receipt_id' => $receipt_id, 'status' => $status));
		if (isset($response['data']['error'])) {
			$this->log("orderUpdateShipped() - Error: " . json_encode($response));
			return $response;
		} else {
			$this->log("orderUpdateShipped() - OK");
			return true;
		}
	}
	public function putStockUpdateBulk($product_id_array, $end_inactive) {
		$this->log("putStockUpdateBulk() - ok");
		$this->log("putStockUpdateBulk() - Item count: " . count($product_id_array));
		foreach($product_id_array as $product_id) {
			$this->log("putStockUpdateBulk() - Product ID: " . $product_id);
			$links = $this->getLinks($product_id, 1);
			if (!empty($links)) {
				$this->log("putStockUpdateBulk() - Links found: " . count($links));
				foreach ($links as $link) {
					$etsy_listing = $this->getEtsyItem($link['etsy_item_id']);
					if ($etsy_listing != false && isset($etsy_listing['state']) && ($etsy_listing['state'] == 'active' || $etsy_listing['state'] == 'private' || $etsy_listing['state'] == 'draft' || $etsy_listing['state'] == 'edit')) {
						$this->log("putStockUpdateBulk() - Listing state seems valid");
						if ($etsy_listing['quantity'] != $link['quantity']) {
							$this->log("putStockUpdateBulk() - Stock is different, do update");
							$this->updateListingStock($link['etsy_item_id'], $link['quantity'], $etsy_listing['state']);
						} else {
							$this->log("putStockUpdateBulk() - Stock is the same: " . $etsy_listing['quantity'] . " " . $link['quantity']);
						}
					} else {
						$this->log("putStockUpdateBulk() - Listing state seems invalid");
						$this->log("putStockUpdateBulk() - " . json_encode($etsy_listing));
						$this->deleteLink($link['etsy_listing_id']);
					}
				}
			} else {
				$this->log("putStockUpdateBulk() - No link found");
			}
		}
	}
	public function getEtsyItem($listing_id) {
		$this->log("getEtsyItem(): " . $listing_id);
		$response = $this->openbay->etsy->call('v1/etsy/product/listing/' . $listing_id . '/', 'GET');
		if (isset($response['data']['error'])) {
			$this->log("getEtsyItem() error: ". $response['data']['error']);
			return $response;
		} else {
			$this->log("getEtsyItem() - OK : " . json_encode($response));
			return $response['data']['results'][0];
		}
	}
	public function validate() {
		if ($this->config->get('etsy_token') && $this->config->get('etsy_encryption_key') && $this->config->get('etsy_encryption_iv')) {
			$this->log("Etsy details valid");
			return true;
		} else {
			$this->log("Etsy details are not valid");
			return false;
		}
	}
}