Merge pull request #15093 from woocommerce/csv-basic-product-importer
CSV basic product importer
This commit is contained in:
commit
021d1a1255
|
@ -51,10 +51,10 @@ class WC_Admin_Importers {
|
|||
}
|
||||
|
||||
// includes
|
||||
require( dirname( __FILE__ ) . '/importers/class-wc-product-importer.php' );
|
||||
require( dirname( __FILE__ ) . '/importers/class-wc-product-wp-importer.php' );
|
||||
|
||||
// Dispatch
|
||||
$importer = new WC_Product_Importer();
|
||||
$importer = new WC_Product_WP_Importer();
|
||||
$importer->dispatch();
|
||||
}
|
||||
|
||||
|
|
|
@ -15,7 +15,7 @@ if ( ! class_exists( 'WP_Importer' ) ) {
|
|||
* @package WooCommerce/Admin/Importers
|
||||
* @version 3.1.0
|
||||
*/
|
||||
class WC_Product_Importer extends WP_Importer {
|
||||
class WC_Product_WP_Importer extends WP_Importer {
|
||||
|
||||
/**
|
||||
* The current file id.
|
||||
|
@ -59,6 +59,7 @@ class WC_Product_Importer extends WP_Importer {
|
|||
* Manages the three separate stages of the CSV import process.
|
||||
*/
|
||||
public function dispatch() {
|
||||
include_once( WC_ABSPATH . 'includes/import/class-wc-product-csv-importer.php' );
|
||||
|
||||
$this->header();
|
||||
|
||||
|
@ -135,15 +136,30 @@ class WC_Product_Importer extends WP_Importer {
|
|||
$args['mapping'] = wp_unslash( $_POST['map_to'] );
|
||||
}
|
||||
|
||||
$data = $this->read_csv( $file, $args );
|
||||
$importer = $this->get_importer( $file, $args );
|
||||
$data = $importer->import();
|
||||
$imported = count( $data['imported'] );
|
||||
$failed = count( $data['failed'] );
|
||||
|
||||
// Show Result
|
||||
echo '<div class="updated settings-error"><p>';
|
||||
$results = sprintf(
|
||||
/* translators: %d: products count */
|
||||
printf(
|
||||
__( 'Import complete - imported %s products.', 'woocommerce' ),
|
||||
'<strong>' . count( $data ) . '</strong>'
|
||||
_n( 'Imported %s product.', 'Imported %s products.', $imported, 'woocommerce' ),
|
||||
'<strong>' . number_format_i18n( $imported ) . '</strong>'
|
||||
);
|
||||
|
||||
// @todo create a view to display errors or log with WC_Logger.
|
||||
if ( 0 < $failed ) {
|
||||
$results .= ' ' . sprintf(
|
||||
/* translators: %d: products count */
|
||||
_n( 'Failed %s product.', 'Failed %s products.', $failed, 'woocommerce' ),
|
||||
'<strong>' . number_format_i18n( $failed ) . '</strong>'
|
||||
);
|
||||
}
|
||||
|
||||
// Show result.
|
||||
echo '<div class="updated settings-error"><p>';
|
||||
/* translators: %d: import results */
|
||||
printf( __( 'Import complete: %s', 'woocommerce' ), $results );
|
||||
echo '</p></div>';
|
||||
|
||||
$this->import_end();
|
||||
|
@ -184,286 +200,6 @@ class WC_Product_Importer extends WP_Importer {
|
|||
return true;
|
||||
}
|
||||
|
||||
/**
|
||||
* Read a CSV file.
|
||||
*
|
||||
* @param mixed $file
|
||||
* @param array $args See $default_args
|
||||
* @return array
|
||||
*/
|
||||
public function read_csv( $file, $args = array() ) {
|
||||
|
||||
$default_args = array(
|
||||
'start_pos' => 0, // File pointer start.
|
||||
'end_pos' => -1, // File pointer end.
|
||||
'lines' => -1, // Max lines to read.
|
||||
'mapping' => array(), // Column mapping. csv_heading => schema_heading.
|
||||
'parse' => false, // Whether to sanitize and format data.
|
||||
);
|
||||
$args = wp_parse_args( $args, $default_args );
|
||||
|
||||
$data = array(
|
||||
'raw_headers' => array(),
|
||||
'data' => array(),
|
||||
);
|
||||
|
||||
if ( false !== ( $handle = fopen( $file, 'r' ) ) ) {
|
||||
|
||||
$data['raw_headers'] = fgetcsv( $handle, 0, $this->delimiter );
|
||||
|
||||
if ( 0 !== $args['start_pos'] ) {
|
||||
fseek( $handle, (int) $args['start_pos'] );
|
||||
}
|
||||
|
||||
while ( false !== ( $row = fgetcsv( $handle, 0, $this->delimiter ) ) ) {
|
||||
$data['data'][] = $row;
|
||||
|
||||
if ( ( $args['end_pos'] > 0 && ftell( $handle ) >= $args['end_pos'] ) || 0 === --$args['lines'] ) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $args['mapping'] ) ) {
|
||||
$data = $this->map_headers( $data, $args['mapping'] );
|
||||
}
|
||||
|
||||
if ( $args['parse'] ) {
|
||||
$data = $this->parse_data( $data );
|
||||
}
|
||||
|
||||
return apply_filters( 'woocommerce_csv_product_import_data', $data, $file, $args );
|
||||
}
|
||||
|
||||
/**
|
||||
* Map raw headers to known headers.
|
||||
*
|
||||
* @param array $data
|
||||
* @param array $mapping 'raw column name' => 'schema column name'
|
||||
* @return array
|
||||
*/
|
||||
public function map_headers( $data, $mapping ) {
|
||||
$data['headers'] = array();
|
||||
foreach ( $data['raw_headers'] as $heading ) {
|
||||
$data['headers'][] = isset( $mapping[ $heading ] ) ? $mapping[ $heading ] : $heading;
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Map and format raw data to known fields.
|
||||
*
|
||||
* @param array $data
|
||||
* @return array
|
||||
*/
|
||||
public function parse_data( $data ) {
|
||||
|
||||
/**
|
||||
* Columns not mentioned here will get parsed with 'esc_attr'.
|
||||
* column_name => callback.
|
||||
*/
|
||||
$data_formatting = array(
|
||||
'id' => 'absint',
|
||||
'status' => array( $this, 'parse_bool_field' ),
|
||||
'featured' => array( $this, 'parse_bool_field' ),
|
||||
'date_on_sale_from' => 'strtotime',
|
||||
'date_on_sale_to' => 'strtotime',
|
||||
'manage_stock' => array( $this, 'parse_bool_field' ),
|
||||
'backorders' => array( $this, 'parse_bool_field' ),
|
||||
'sold_individually' => array( $this, 'parse_bool_field' ),
|
||||
'width' => array( $this, 'parse_float_field' ),
|
||||
'length' => array( $this, 'parse_float_field' ),
|
||||
'height' => array( $this, 'parse_float_field' ),
|
||||
'weight' => array( $this, 'parse_float_field' ),
|
||||
'reviews_allowed' => array( $this, 'parse_bool_field' ),
|
||||
'purchase_note' => 'wp_kses_post',
|
||||
'price' => 'wc_format_decimal',
|
||||
'regular_price' => 'wc_format_decimal',
|
||||
'stock_quantity' => 'absint',
|
||||
'category_ids' => array( $this, 'parse_categories' ),
|
||||
'tag_ids' => array( $this, 'parse_comma_field' ),
|
||||
'images' => array( $this, 'parse_comma_field' ),
|
||||
'upsell_ids' => array( $this, 'parse_comma_field' ),
|
||||
'cross_sell_ids' => array( $this, 'parse_comma_field' ),
|
||||
'download_limit' => 'absint',
|
||||
'download_expiry' => 'absint',
|
||||
);
|
||||
/**
|
||||
* @todo switch these to some standard, slug format.
|
||||
*/
|
||||
$regex_match_data_formatting = array(
|
||||
'/Attribute * Value\(s\)/' => array( $this, 'parse_comma_field' ),
|
||||
'/Attribute * Visible/' => array( $this, 'parse_bool_field' ),
|
||||
'/Download * URL/' => 'esc_url',
|
||||
);
|
||||
|
||||
$headers = ! empty( $data['headers'] ) ? $data['headers'] : $data['raw_headers'];
|
||||
$parse_functions = array();
|
||||
$parsed_data = array();
|
||||
|
||||
// Figure out the parse function for each column.
|
||||
foreach ( $headers as $index => $heading ) {
|
||||
|
||||
$parse_function = 'esc_attr';
|
||||
if ( isset( $data_formatting[ $heading ] ) ) {
|
||||
$parse_function = $data_formatting[ $heading ];
|
||||
} else {
|
||||
foreach ( $regex_match_data_formatting as $regex => $callback ) {
|
||||
if ( preg_match( $regex, $heading ) ) {
|
||||
$parse_function = $callback;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$parse_functions[] = $parse_function;
|
||||
}
|
||||
|
||||
// Parse the data.
|
||||
foreach ( $data['data'] as $row ) {
|
||||
$item = array();
|
||||
foreach ( $row as $index => $field ) {
|
||||
$item[ $headers[ $index ] ] = call_user_func( $parse_functions[ $index ], $field );
|
||||
}
|
||||
$parsed_data[] = $item;
|
||||
}
|
||||
|
||||
return apply_filters( 'woocommerce_csv_product_parsed_data', $parsed_data, $data );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get default fields.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function get_default_fields() {
|
||||
$fields = array(
|
||||
'id',
|
||||
'type',
|
||||
'sku',
|
||||
'name',
|
||||
'status',
|
||||
'featured',
|
||||
'catalog_visibility',
|
||||
'short_description',
|
||||
'description',
|
||||
'date_on_sale_from',
|
||||
'date_on_sale_to',
|
||||
'tax_status',
|
||||
'tax_class',
|
||||
'stock_status',
|
||||
'backorders',
|
||||
'sold_individually',
|
||||
'weight',
|
||||
'length',
|
||||
'width',
|
||||
'height',
|
||||
'reviews_allowed',
|
||||
'purchase_note',
|
||||
'price',
|
||||
'regular_price',
|
||||
'manage_stock',
|
||||
'stock_quantity',
|
||||
'category_ids',
|
||||
'tag_ids',
|
||||
'shipping_class_id',
|
||||
'images',
|
||||
'downloads',
|
||||
'download_limit',
|
||||
'download_expiry',
|
||||
'parent_id',
|
||||
'upsell_ids',
|
||||
'cross_sell_ids',
|
||||
);
|
||||
|
||||
return apply_filters( 'woocommerce_csv_product_default_fields', $fields );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a comma-delineated field from a CSV.
|
||||
*
|
||||
* @param string $field
|
||||
* @return array
|
||||
*/
|
||||
public function parse_comma_field( $field ) {
|
||||
if ( empty( $field ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
return array_map( 'esc_attr', array_map( 'trim', explode( ',', $field ) ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a field that is generally '1' or '0' but can be something else.
|
||||
*
|
||||
* @param string $field
|
||||
* @return bool|string
|
||||
*/
|
||||
public function parse_bool_field( $field ) {
|
||||
if ( '0' === $field ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( '1' === $field ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// Don't return explicit true or false for empty fields or values like 'notify'.
|
||||
return esc_attr( $field );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a float value field.
|
||||
*
|
||||
* @param string $field
|
||||
* @return float|string
|
||||
*/
|
||||
public function parse_float_field( $field ) {
|
||||
if ( '' === $field ) {
|
||||
return $field;
|
||||
}
|
||||
|
||||
return floatval( $field );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a category field from a CSV.
|
||||
* Categories are separated by commas and subcategories are "parent > subcategory".
|
||||
*
|
||||
* @param string $field
|
||||
* @return array of arrays with "parent" and "name" keys.
|
||||
*/
|
||||
public function parse_categories( $field ) {
|
||||
if ( empty( $field ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$sections = array_map( 'trim', explode( ',', $field ) );
|
||||
$categories = array();
|
||||
|
||||
foreach ( $sections as $section ) {
|
||||
|
||||
// Top level category.
|
||||
if ( false === strpos( $section, '>' ) ) {
|
||||
$categories[] = array(
|
||||
'parent' => false,
|
||||
'name' => esc_attr( $section ),
|
||||
);
|
||||
|
||||
// Subcategory.
|
||||
} else {
|
||||
$chunks = array_map( 'trim', explode( '>', $section ) );
|
||||
$categories[] = array(
|
||||
'parent' => esc_attr( reset( $chunks ) ),
|
||||
'name' => esc_attr( end( $chunks ) ),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return $categories;
|
||||
}
|
||||
|
||||
/**
|
||||
* Output header html.
|
||||
*/
|
||||
|
@ -555,6 +291,54 @@ class WC_Product_Importer extends WP_Importer {
|
|||
die();
|
||||
}
|
||||
|
||||
/**
|
||||
* Get default fields.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function get_default_fields() {
|
||||
$fields = array(
|
||||
'id',
|
||||
'type',
|
||||
'sku',
|
||||
'name',
|
||||
'status',
|
||||
'featured',
|
||||
'catalog_visibility',
|
||||
'short_description',
|
||||
'description',
|
||||
'date_on_sale_from',
|
||||
'date_on_sale_to',
|
||||
'tax_status',
|
||||
'tax_class',
|
||||
'stock_status',
|
||||
'backorders',
|
||||
'sold_individually',
|
||||
'weight',
|
||||
'length',
|
||||
'width',
|
||||
'height',
|
||||
'reviews_allowed',
|
||||
'purchase_note',
|
||||
'price',
|
||||
'regular_price',
|
||||
'manage_stock',
|
||||
'stock_quantity',
|
||||
'category_ids',
|
||||
'tag_ids',
|
||||
'shipping_class_id',
|
||||
'images',
|
||||
'downloads',
|
||||
'download_limit',
|
||||
'download_expiry',
|
||||
'parent_id',
|
||||
'upsell_ids',
|
||||
'cross_sell_ids',
|
||||
);
|
||||
|
||||
return apply_filters( 'woocommerce_csv_product_default_fields', $fields );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get mapping options.
|
||||
*
|
||||
|
@ -616,7 +400,7 @@ class WC_Product_Importer extends WP_Importer {
|
|||
'meta:' . $item => __( 'Import as meta', 'woocommerce' ),
|
||||
);
|
||||
|
||||
return apply_filters( 'woocommerce_csv_product_import_mapping_options', $options. $item );
|
||||
return apply_filters( 'woocommerce_csv_product_import_mapping_options', $options, $item );
|
||||
}
|
||||
|
||||
/**
|
||||
|
@ -625,9 +409,13 @@ class WC_Product_Importer extends WP_Importer {
|
|||
* @param string $file File path.
|
||||
*/
|
||||
protected function importer_mapping( $file ) {
|
||||
$data = $this->read_csv( $file, array( 'lines' => 1 ) );
|
||||
$headers = $data['raw_headers'];
|
||||
$sample = $data['data'][0];
|
||||
$importer = $this->get_importer( $file, array( 'lines' => 1 ) );
|
||||
$headers = $importer->get_raw_keys();
|
||||
$sample = current( $importer->get_raw_data() );
|
||||
|
||||
if ( empty( $sample ) ) {
|
||||
$this->import_error( __( 'The file is empty, please try again with a new file.', 'woocommerce' ) );
|
||||
}
|
||||
|
||||
// Check if all fields matches.
|
||||
if ( 0 === count( array_diff( $headers, $this->get_default_fields() ) ) ) {
|
||||
|
@ -645,4 +433,16 @@ class WC_Product_Importer extends WP_Importer {
|
|||
|
||||
include_once( dirname( __FILE__ ) . '/views/html-csv-mapping.php' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Get importer instance.
|
||||
*
|
||||
* @param string $file File to import.
|
||||
* @param array $args Importer arguments.
|
||||
* @return WC_Product_CSV_Importer
|
||||
*/
|
||||
protected function get_importer( $file, $args = array() ) {
|
||||
$importer_class = apply_filters( 'woocommerce_product_csv_impoter_class', 'WC_Product_CSV_Importer' );
|
||||
return new $importer_class( $file, $args );
|
||||
}
|
||||
}
|
|
@ -0,0 +1,941 @@
|
|||
<?php
|
||||
/**
|
||||
* Abstract Product importer
|
||||
*
|
||||
* @author Automattic
|
||||
* @category Admin
|
||||
* @package WooCommerce/Import
|
||||
* @version 3.1.0
|
||||
*/
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Include dependencies.
|
||||
*/
|
||||
if ( ! class_exists( 'WC_Importer_Interface', false ) ) {
|
||||
include_once( WC_ABSPATH . 'includes/interfaces/class-wc-importer-interface.php' );
|
||||
}
|
||||
|
||||
/**
|
||||
* WC_Product_Importer Class.
|
||||
*/
|
||||
abstract class WC_Product_Importer implements WC_Importer_Interface {
|
||||
|
||||
/**
|
||||
* CSV file.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $file = '';
|
||||
|
||||
/**
|
||||
* Importer parameters.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $params = array();
|
||||
|
||||
/**
|
||||
* Raw keys - CSV raw headers.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $raw_keys = array();
|
||||
|
||||
/**
|
||||
* Mapped keys - CSV headers.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $mapped_keys = array();
|
||||
|
||||
/**
|
||||
* Raw data.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $raw_data = array();
|
||||
|
||||
/**
|
||||
* Parsed data.
|
||||
*
|
||||
* @var array
|
||||
*/
|
||||
protected $parsed_data = array();
|
||||
|
||||
/**
|
||||
* Get file raw headers.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_raw_keys() {
|
||||
return $this->raw_keys;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get file mapped headers.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_mapped_keys() {
|
||||
return $this->mapped_keys;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get raw data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_raw_data() {
|
||||
return $this->raw_data;
|
||||
}
|
||||
|
||||
/**
|
||||
* Get parsed data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_parsed_data() {
|
||||
return apply_filters( 'woocommerce_product_parsed_data', $this->parsed_data, $this->get_raw_data() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Process a single item and save.
|
||||
*
|
||||
* @param array $data Raw CSV data.
|
||||
* @return WC_Product|WC_Error
|
||||
*/
|
||||
protected function process_item( $data ) {
|
||||
// Ignore IDs and create new products.
|
||||
// @todo Mike said that we should have something to force create.
|
||||
$force_create = false;
|
||||
|
||||
try {
|
||||
$object = $this->prepare_product( $data, $force_create );
|
||||
|
||||
if ( is_wp_error( $object ) ) {
|
||||
return $object;
|
||||
}
|
||||
|
||||
$object->save();
|
||||
|
||||
// Clean cache for updated products.
|
||||
$this->clear_cache( $object );
|
||||
|
||||
return $object->get_id();
|
||||
} catch ( WC_Data_Exception $e ) {
|
||||
return new WP_Error( $e->getErrorCode(), $e->getMessage(), $e->getErrorData() );
|
||||
} catch ( Exception $e ) {
|
||||
return new WP_Error( 'woocommerce_product_csv_importer_error', $e->getMessage(), array( 'status' => $e->getCode() ) );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Clear product cache.
|
||||
*
|
||||
* @param WC_Product $object Product instance.
|
||||
*/
|
||||
protected function clear_cache( $object ) {
|
||||
$id = $object->get_id();
|
||||
|
||||
if ( 'variation' === $object->get_type() ) {
|
||||
$id = $object->get_parent_id();
|
||||
}
|
||||
|
||||
wc_delete_product_transients( $id );
|
||||
wp_cache_delete( 'product-' . $id, 'products' );
|
||||
}
|
||||
|
||||
/**
|
||||
* Prepare a single product for create or update.
|
||||
*
|
||||
* @param array $data Row data.
|
||||
* @param bool $creating If should force create a new product.
|
||||
* @return WC_Product|WP_Error
|
||||
*/
|
||||
protected function prepare_product( $data, $force_create = false ) {
|
||||
$id = ! $force_create && isset( $data['id'] ) ? absint( $data['id'] ) : 0;
|
||||
|
||||
// Type is the most important part here because we need to be using the correct class and methods.
|
||||
if ( isset( $data['type'] ) ) {
|
||||
if ( ! in_array( $data['type'], array_keys( wc_get_product_types() ), true ) ) {
|
||||
return new WP_Error( 'woocommerce_product_csv_importer_invalid_type', __( 'Invalid product type.', 'woocommerce' ), array( 'status' => 401 ) );
|
||||
}
|
||||
|
||||
$classname = WC_Product_Factory::get_classname_from_product_type( $data['type'] );
|
||||
|
||||
if ( ! class_exists( $classname ) ) {
|
||||
$classname = 'WC_Product_Simple';
|
||||
}
|
||||
|
||||
$product = new $classname( $id );
|
||||
} elseif ( isset( $data['id'] ) ) {
|
||||
$product = wc_get_product( $id );
|
||||
} else {
|
||||
$product = new WC_Product_Simple();
|
||||
}
|
||||
|
||||
// @todo need to check first how we'll handle variations.
|
||||
if ( 'variation' === $product->get_type() ) {
|
||||
$product = $this->save_variation_data( $product, $data );
|
||||
} else {
|
||||
$product = $this->save_product_data( $product, $data );
|
||||
}
|
||||
|
||||
return apply_filters( 'woocommerce_product_csv_import_pre_insert_product_object', $product, $data );
|
||||
}
|
||||
|
||||
/**
|
||||
* Set product data.
|
||||
*
|
||||
* @param WC_Product $product Product instance.
|
||||
* @param array $data Row data.
|
||||
*
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_product_data( $product, $data ) {
|
||||
|
||||
// Post title.
|
||||
if ( isset( $data['name'] ) ) {
|
||||
$product->set_name( wp_filter_post_kses( $data['name'] ) );
|
||||
}
|
||||
|
||||
// Post content.
|
||||
if ( isset( $data['description'] ) ) {
|
||||
$product->set_description( wp_filter_post_kses( $data['description'] ) );
|
||||
}
|
||||
|
||||
// Post excerpt.
|
||||
if ( isset( $data['short_description'] ) ) {
|
||||
$product->set_short_description( wp_filter_post_kses( $data['short_description'] ) );
|
||||
}
|
||||
|
||||
// Post status.
|
||||
if ( isset( $data['status'] ) ) {
|
||||
$product->set_status( $data['status'] ? 'publish' : 'draft' );
|
||||
}
|
||||
|
||||
// Post slug.
|
||||
if ( isset( $data['slug'] ) ) {
|
||||
$product->set_slug( $data['slug'] );
|
||||
}
|
||||
|
||||
// Menu order.
|
||||
if ( isset( $data['menu_order'] ) ) {
|
||||
$product->set_menu_order( $data['menu_order'] );
|
||||
}
|
||||
|
||||
// Comment status.
|
||||
if ( isset( $data['reviews_allowed'] ) ) {
|
||||
$product->set_reviews_allowed( $data['reviews_allowed'] );
|
||||
}
|
||||
|
||||
// Virtual.
|
||||
if ( isset( $data['virtual'] ) ) {
|
||||
$product->set_virtual( $data['virtual'] );
|
||||
}
|
||||
|
||||
// Tax status.
|
||||
if ( isset( $data['tax_status'] ) ) {
|
||||
$product->set_tax_status( $data['tax_status'] );
|
||||
}
|
||||
|
||||
// Tax Class.
|
||||
if ( isset( $data['tax_class'] ) ) {
|
||||
$product->set_tax_class( $data['tax_class'] );
|
||||
}
|
||||
|
||||
// Catalog Visibility.
|
||||
if ( isset( $data['catalog_visibility'] ) ) {
|
||||
$product->set_catalog_visibility( $data['catalog_visibility'] );
|
||||
}
|
||||
|
||||
// Purchase Note.
|
||||
if ( isset( $data['purchase_note'] ) ) {
|
||||
$product->set_purchase_note( wc_clean( $data['purchase_note'] ) );
|
||||
}
|
||||
|
||||
// Featured Product.
|
||||
if ( isset( $data['featured'] ) ) {
|
||||
$product->set_featured( $data['featured'] );
|
||||
}
|
||||
|
||||
// Shipping data.
|
||||
$product = $this->save_product_shipping_data( $product, $data );
|
||||
|
||||
// SKU.
|
||||
if ( isset( $data['sku'] ) ) {
|
||||
$product->set_sku( wc_clean( $data['sku'] ) );
|
||||
}
|
||||
|
||||
// Attributes.
|
||||
if ( isset( $data['attributes'] ) ) {
|
||||
$attributes = array();
|
||||
|
||||
foreach ( $data['attributes'] as $attribute ) {
|
||||
$attribute_id = 0;
|
||||
$attribute_name = '';
|
||||
|
||||
// Check ID for global attributes or name for product attributes.
|
||||
if ( ! empty( $attribute['id'] ) ) {
|
||||
$attribute_id = absint( $attribute['id'] );
|
||||
$attribute_name = wc_attribute_taxonomy_name_by_id( $attribute_id );
|
||||
} elseif ( ! empty( $attribute['name'] ) ) {
|
||||
$attribute_name = wc_clean( $attribute['name'] );
|
||||
}
|
||||
|
||||
if ( ! $attribute_id && ! $attribute_name ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( $attribute_id ) {
|
||||
|
||||
if ( isset( $attribute['options'] ) ) {
|
||||
$options = $attribute['options'];
|
||||
|
||||
if ( ! is_array( $attribute['options'] ) ) {
|
||||
// Text based attributes - Posted values are term names.
|
||||
$options = explode( WC_DELIMITER, $options );
|
||||
}
|
||||
|
||||
$values = array_map( 'wc_sanitize_term_text_based', $options );
|
||||
$values = array_filter( $values, 'strlen' );
|
||||
} else {
|
||||
$values = array();
|
||||
}
|
||||
|
||||
if ( ! empty( $values ) ) {
|
||||
// Add attribute to array, but don't set values.
|
||||
$attribute_object = new WC_Product_Attribute();
|
||||
$attribute_object->set_id( $attribute_id );
|
||||
$attribute_object->set_name( $attribute_name );
|
||||
$attribute_object->set_options( $values );
|
||||
$attribute_object->set_position( isset( $attribute['position'] ) ? (string) absint( $attribute['position'] ) : '0' );
|
||||
$attribute_object->set_visible( ( isset( $attribute['visible'] ) && $attribute['visible'] ) ? 1 : 0 );
|
||||
$attribute_object->set_variation( ( isset( $attribute['variation'] ) && $attribute['variation'] ) ? 1 : 0 );
|
||||
$attributes[] = $attribute_object;
|
||||
}
|
||||
} elseif ( isset( $attribute['options'] ) ) {
|
||||
// Custom attribute - Add attribute to array and set the values.
|
||||
if ( is_array( $attribute['options'] ) ) {
|
||||
$values = $attribute['options'];
|
||||
} else {
|
||||
$values = explode( WC_DELIMITER, $attribute['options'] );
|
||||
}
|
||||
$attribute_object = new WC_Product_Attribute();
|
||||
$attribute_object->set_name( $attribute_name );
|
||||
$attribute_object->set_options( $values );
|
||||
$attribute_object->set_position( isset( $attribute['position'] ) ? (string) absint( $attribute['position'] ) : '0' );
|
||||
$attribute_object->set_visible( ( isset( $attribute['visible'] ) && $attribute['visible'] ) ? 1 : 0 );
|
||||
$attribute_object->set_variation( ( isset( $attribute['variation'] ) && $attribute['variation'] ) ? 1 : 0 );
|
||||
$attributes[] = $attribute_object;
|
||||
}
|
||||
}
|
||||
$product->set_attributes( $attributes );
|
||||
}
|
||||
|
||||
// Sales and prices.
|
||||
if ( in_array( $product->get_type(), array( 'variable', 'grouped' ), true ) ) {
|
||||
$product->set_regular_price( '' );
|
||||
$product->set_sale_price( '' );
|
||||
$product->set_date_on_sale_to( '' );
|
||||
$product->set_date_on_sale_from( '' );
|
||||
$product->set_price( '' );
|
||||
} else {
|
||||
// Regular Price.
|
||||
if ( isset( $data['regular_price'] ) ) {
|
||||
$product->set_regular_price( $data['regular_price'] );
|
||||
}
|
||||
|
||||
// Sale Price.
|
||||
if ( isset( $data['sale_price'] ) ) {
|
||||
$product->set_sale_price( $data['sale_price'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_from'] ) ) {
|
||||
$product->set_date_on_sale_from( $data['date_on_sale_from'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_from_gmt'] ) ) {
|
||||
$product->set_date_on_sale_from( $data['date_on_sale_from_gmt'] ? strtotime( $data['date_on_sale_from_gmt'] ) : null );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_to'] ) ) {
|
||||
$product->set_date_on_sale_to( $data['date_on_sale_to'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_to_gmt'] ) ) {
|
||||
$product->set_date_on_sale_to( $data['date_on_sale_to_gmt'] ? strtotime( $data['date_on_sale_to_gmt'] ) : null );
|
||||
}
|
||||
}
|
||||
|
||||
// Product parent ID for groups.
|
||||
if ( isset( $data['parent_id'] ) ) {
|
||||
$product->set_parent_id( $data['parent_id'] );
|
||||
}
|
||||
|
||||
// Sold individually.
|
||||
if ( isset( $data['sold_individually'] ) ) {
|
||||
$product->set_sold_individually( $data['sold_individually'] );
|
||||
}
|
||||
|
||||
// Stock status.
|
||||
if ( isset( $data['in_stock'] ) ) {
|
||||
$stock_status = true === $data['in_stock'] ? 'instock' : 'outofstock';
|
||||
} else {
|
||||
$stock_status = $product->get_stock_status();
|
||||
}
|
||||
|
||||
// Stock data.
|
||||
if ( 'yes' === get_option( 'woocommerce_manage_stock' ) ) {
|
||||
// Manage stock.
|
||||
if ( isset( $data['manage_stock'] ) ) {
|
||||
$product->set_manage_stock( $data['manage_stock'] );
|
||||
}
|
||||
|
||||
// Backorders.
|
||||
if ( isset( $data['backorders'] ) ) {
|
||||
$product->set_backorders( $data['backorders'] );
|
||||
}
|
||||
|
||||
if ( $product->is_type( 'grouped' ) ) {
|
||||
$product->set_manage_stock( 'no' );
|
||||
$product->set_backorders( 'no' );
|
||||
$product->set_stock_quantity( '' );
|
||||
$product->set_stock_status( $stock_status );
|
||||
} elseif ( $product->is_type( 'external' ) ) {
|
||||
$product->set_manage_stock( 'no' );
|
||||
$product->set_backorders( 'no' );
|
||||
$product->set_stock_quantity( '' );
|
||||
$product->set_stock_status( 'instock' );
|
||||
} elseif ( $product->get_manage_stock() ) {
|
||||
// Stock status is always determined by children so sync later.
|
||||
if ( ! $product->is_type( 'variable' ) ) {
|
||||
$product->set_stock_status( $stock_status );
|
||||
}
|
||||
|
||||
// Stock quantity.
|
||||
if ( isset( $data['stock_quantity'] ) ) {
|
||||
$product->set_stock_quantity( wc_stock_amount( $data['stock_quantity'] ) );
|
||||
} elseif ( isset( $data['inventory_delta'] ) ) {
|
||||
$stock_quantity = wc_stock_amount( $product->get_stock_quantity() );
|
||||
$stock_quantity += wc_stock_amount( $data['inventory_delta'] );
|
||||
$product->set_stock_quantity( wc_stock_amount( $stock_quantity ) );
|
||||
}
|
||||
} else {
|
||||
// Don't manage stock.
|
||||
$product->set_manage_stock( 'no' );
|
||||
$product->set_stock_quantity( '' );
|
||||
$product->set_stock_status( $stock_status );
|
||||
}
|
||||
} elseif ( ! $product->is_type( 'variable' ) ) {
|
||||
$product->set_stock_status( $stock_status );
|
||||
}
|
||||
|
||||
// Upsells.
|
||||
if ( isset( $data['upsell_ids'] ) ) {
|
||||
$upsells = array();
|
||||
$ids = $data['upsell_ids'];
|
||||
|
||||
if ( ! empty( $ids ) ) {
|
||||
foreach ( $ids as $id ) {
|
||||
if ( $id && $id > 0 ) {
|
||||
$upsells[] = $id;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$product->set_upsell_ids( $upsells );
|
||||
}
|
||||
|
||||
// Cross sells.
|
||||
if ( isset( $data['cross_sell_ids'] ) ) {
|
||||
$crosssells = array();
|
||||
$ids = $data['cross_sell_ids'];
|
||||
|
||||
if ( ! empty( $ids ) ) {
|
||||
foreach ( $ids as $id ) {
|
||||
if ( $id && $id > 0 ) {
|
||||
$crosssells[] = $id;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$product->set_cross_sell_ids( $crosssells );
|
||||
}
|
||||
|
||||
// Product categories.
|
||||
if ( isset( $data['categories'] ) && is_array( $data['categories'] ) ) {
|
||||
$product = $this->save_taxonomy_terms( $product, $data['categories'] );
|
||||
}
|
||||
|
||||
// Product tags.
|
||||
if ( isset( $data['tags'] ) && is_array( $data['tags'] ) ) {
|
||||
$product = $this->save_taxonomy_terms( $product, $data['tags'], 'tag' );
|
||||
}
|
||||
|
||||
// Downloadable.
|
||||
if ( isset( $data['downloadable'] ) ) {
|
||||
$product->set_downloadable( $data['downloadable'] );
|
||||
}
|
||||
|
||||
// Downloadable options.
|
||||
if ( $product->get_downloadable() ) {
|
||||
|
||||
// Downloadable files.
|
||||
if ( isset( $data['downloads'] ) && is_array( $data['downloads'] ) ) {
|
||||
$product = $this->save_downloadable_files( $product, $data['downloads'] );
|
||||
}
|
||||
|
||||
// Download limit.
|
||||
if ( isset( $data['download_limit'] ) ) {
|
||||
$product->set_download_limit( $data['download_limit'] );
|
||||
}
|
||||
|
||||
// Download expiry.
|
||||
if ( isset( $data['download_expiry'] ) ) {
|
||||
$product->set_download_expiry( $data['download_expiry'] );
|
||||
}
|
||||
}
|
||||
|
||||
// Product url and button text for external products.
|
||||
if ( $product->is_type( 'external' ) ) {
|
||||
if ( isset( $data['external_url'] ) ) {
|
||||
$product->set_product_url( $data['external_url'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['button_text'] ) ) {
|
||||
$product->set_button_text( $data['button_text'] );
|
||||
}
|
||||
}
|
||||
|
||||
// Save default attributes for variable products.
|
||||
if ( $product->is_type( 'variable' ) ) {
|
||||
$product = $this->save_default_attributes( $product, $data );
|
||||
}
|
||||
|
||||
// Check for featured/gallery images, upload it and set it.
|
||||
if ( isset( $data['images'] ) ) {
|
||||
$product = $this->save_product_images( $product, $data['images'] );
|
||||
}
|
||||
|
||||
// Allow set meta_data.
|
||||
if ( isset( $data['meta_data'] ) && is_array( $data['meta_data'] ) ) {
|
||||
foreach ( $data['meta_data'] as $meta ) {
|
||||
$product->update_meta_data( $meta['key'], $meta['value'], isset( $meta['id'] ) ? $meta['id'] : '' );
|
||||
}
|
||||
}
|
||||
|
||||
return $product;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set variation data.
|
||||
*
|
||||
* @param WC_Product $variation Product instance.
|
||||
* @param array $data Row data.
|
||||
*
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_variation_data( $variation, $data ) {
|
||||
if ( isset( $data['product_id'] ) ) {
|
||||
$variation->set_parent_id( absint( $data['product_id'] ) );
|
||||
} else {
|
||||
return new WP_Error( 'woocommerce_product_importer_missing_variation_parent_id', __( 'Missing variation product parent ID', 'woocommerce' ), array( 'status' => 401 ) );
|
||||
}
|
||||
|
||||
// Status.
|
||||
if ( isset( $data['status'] ) ) {
|
||||
$variation->set_status( false === $data['status'] ? 'private' : 'publish' );
|
||||
}
|
||||
|
||||
// SKU.
|
||||
if ( isset( $data['sku'] ) ) {
|
||||
$variation->set_sku( wc_clean( $data['sku'] ) );
|
||||
}
|
||||
|
||||
// Thumbnail.
|
||||
if ( isset( $data['image'] ) ) {
|
||||
if ( is_array( $data['image'] ) ) {
|
||||
$image = $data['image'];
|
||||
if ( is_array( $image ) ) {
|
||||
$image['position'] = 0;
|
||||
}
|
||||
|
||||
$variation = $this->save_product_images( $variation, array( $image ) );
|
||||
} else {
|
||||
$variation->set_image_id( '' );
|
||||
}
|
||||
}
|
||||
|
||||
// Virtual variation.
|
||||
if ( isset( $data['virtual'] ) ) {
|
||||
$variation->set_virtual( $data['virtual'] );
|
||||
}
|
||||
|
||||
// Downloadable variation.
|
||||
if ( isset( $data['downloadable'] ) ) {
|
||||
$variation->set_downloadable( $data['downloadable'] );
|
||||
}
|
||||
|
||||
// Downloads.
|
||||
if ( $variation->get_downloadable() ) {
|
||||
// Downloadable files.
|
||||
if ( isset( $data['downloads'] ) && is_array( $data['downloads'] ) ) {
|
||||
$variation = $this->save_downloadable_files( $variation, $data['downloads'] );
|
||||
}
|
||||
|
||||
// Download limit.
|
||||
if ( isset( $data['download_limit'] ) ) {
|
||||
$variation->set_download_limit( $data['download_limit'] );
|
||||
}
|
||||
|
||||
// Download expiry.
|
||||
if ( isset( $data['download_expiry'] ) ) {
|
||||
$variation->set_download_expiry( $data['download_expiry'] );
|
||||
}
|
||||
}
|
||||
|
||||
// Shipping data.
|
||||
$variation = $this->save_product_shipping_data( $variation, $data );
|
||||
|
||||
// Stock handling.
|
||||
if ( isset( $data['manage_stock'] ) ) {
|
||||
$variation->set_manage_stock( $data['manage_stock'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['in_stock'] ) ) {
|
||||
$variation->set_stock_status( true === $data['in_stock'] ? 'instock' : 'outofstock' );
|
||||
}
|
||||
|
||||
if ( isset( $data['backorders'] ) ) {
|
||||
$variation->set_backorders( $data['backorders'] );
|
||||
}
|
||||
|
||||
if ( $variation->get_manage_stock() ) {
|
||||
if ( isset( $data['stock_quantity'] ) ) {
|
||||
$variation->set_stock_quantity( $data['stock_quantity'] );
|
||||
} elseif ( isset( $data['inventory_delta'] ) ) {
|
||||
$stock_quantity = wc_stock_amount( $variation->get_stock_quantity() );
|
||||
$stock_quantity += wc_stock_amount( $data['inventory_delta'] );
|
||||
$variation->set_stock_quantity( $stock_quantity );
|
||||
}
|
||||
} else {
|
||||
$variation->set_backorders( 'no' );
|
||||
$variation->set_stock_quantity( '' );
|
||||
}
|
||||
|
||||
// Regular Price.
|
||||
if ( isset( $data['regular_price'] ) ) {
|
||||
$variation->set_regular_price( $data['regular_price'] );
|
||||
}
|
||||
|
||||
// Sale Price.
|
||||
if ( isset( $data['sale_price'] ) ) {
|
||||
$variation->set_sale_price( $data['sale_price'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_from'] ) ) {
|
||||
$variation->set_date_on_sale_from( $data['date_on_sale_from'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_from_gmt'] ) ) {
|
||||
$variation->set_date_on_sale_from( $data['date_on_sale_from_gmt'] ? strtotime( $data['date_on_sale_from_gmt'] ) : null );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_to'] ) ) {
|
||||
$variation->set_date_on_sale_to( $data['date_on_sale_to'] );
|
||||
}
|
||||
|
||||
if ( isset( $data['date_on_sale_to_gmt'] ) ) {
|
||||
$variation->set_date_on_sale_to( $data['date_on_sale_to_gmt'] ? strtotime( $data['date_on_sale_to_gmt'] ) : null );
|
||||
}
|
||||
|
||||
// Tax class.
|
||||
if ( isset( $data['tax_class'] ) ) {
|
||||
$variation->set_tax_class( $data['tax_class'] );
|
||||
}
|
||||
|
||||
// Description.
|
||||
if ( isset( $data['description'] ) ) {
|
||||
$variation->set_description( wp_kses_post( $data['description'] ) );
|
||||
}
|
||||
|
||||
// Update taxonomies.
|
||||
if ( isset( $data['attributes'] ) ) {
|
||||
$attributes = array();
|
||||
$parent = wc_get_product( $variation->get_parent_id() );
|
||||
$parent_attributes = $parent->get_attributes();
|
||||
|
||||
foreach ( $data['attributes'] as $attribute ) {
|
||||
$attribute_id = 0;
|
||||
$attribute_name = '';
|
||||
|
||||
// Check ID for global attributes or name for product attributes.
|
||||
if ( ! empty( $attribute['id'] ) ) {
|
||||
$attribute_id = absint( $attribute['id'] );
|
||||
$attribute_name = wc_attribute_taxonomy_name_by_id( $attribute_id );
|
||||
} elseif ( ! empty( $attribute['name'] ) ) {
|
||||
$attribute_name = sanitize_title( $attribute['name'] );
|
||||
}
|
||||
|
||||
if ( ! $attribute_id && ! $attribute_name ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( ! isset( $parent_attributes[ $attribute_name ] ) || ! $parent_attributes[ $attribute_name ]->get_variation() ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$attribute_key = sanitize_title( $parent_attributes[ $attribute_name ]->get_name() );
|
||||
$attribute_value = isset( $attribute['option'] ) ? wc_clean( stripslashes( $attribute['option'] ) ) : '';
|
||||
|
||||
if ( $parent_attributes[ $attribute_name ]->is_taxonomy() ) {
|
||||
// If dealing with a taxonomy, we need to get the slug from the name posted to the API.
|
||||
$term = get_term_by( 'name', $attribute_value, $attribute_name );
|
||||
|
||||
if ( $term && ! is_wp_error( $term ) ) {
|
||||
$attribute_value = $term->slug;
|
||||
} else {
|
||||
$attribute_value = sanitize_title( $attribute_value );
|
||||
}
|
||||
}
|
||||
|
||||
$attributes[ $attribute_key ] = $attribute_value;
|
||||
}
|
||||
|
||||
$variation->set_attributes( $attributes );
|
||||
}
|
||||
|
||||
// Menu order.
|
||||
if ( isset( $data['menu_order'] ) ) {
|
||||
$variation->set_menu_order( $data['menu_order'] );
|
||||
}
|
||||
|
||||
// Meta data.
|
||||
if ( isset( $data['meta_data'] ) && is_array( $data['meta_data'] ) ) {
|
||||
foreach ( $data['meta_data'] as $meta ) {
|
||||
$variation->update_meta_data( $meta['key'], $meta['value'], isset( $meta['id'] ) ? $meta['id'] : '' );
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
return $variation;
|
||||
}
|
||||
|
||||
/**
|
||||
* Set product images.
|
||||
*
|
||||
* @param WC_Product $product Product instance.
|
||||
* @param array $images Images data.
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_product_images( $product, $images ) {
|
||||
if ( is_array( $images ) ) {
|
||||
$gallery = array();
|
||||
|
||||
foreach ( $images as $image ) {
|
||||
$attachment_id = isset( $image['id'] ) ? absint( $image['id'] ) : 0;
|
||||
|
||||
if ( 0 === $attachment_id && isset( $image['src'] ) ) {
|
||||
$upload = wc_rest_upload_image_from_url( esc_url_raw( $image['src'] ) );
|
||||
|
||||
if ( is_wp_error( $upload ) ) {
|
||||
if ( ! apply_filters( 'woocommerce_rest_suppress_image_upload_error', false, $upload, $product->get_id(), $images ) ) {
|
||||
throw new Exception( $upload->get_error_message(), 400 );
|
||||
} else {
|
||||
continue;
|
||||
}
|
||||
}
|
||||
|
||||
$attachment_id = wc_rest_set_uploaded_image_as_attachment( $upload, $product->get_id() );
|
||||
}
|
||||
|
||||
if ( ! wp_attachment_is_image( $attachment_id ) ) {
|
||||
throw new Exception( sprintf( __( '#%s is an invalid image ID.', 'woocommerce' ), $attachment_id ), 400 );
|
||||
}
|
||||
|
||||
if ( isset( $image['position'] ) && 0 === absint( $image['position'] ) ) {
|
||||
$product->set_image_id( $attachment_id );
|
||||
} else {
|
||||
$gallery[] = $attachment_id;
|
||||
}
|
||||
|
||||
// Set the image alt if present.
|
||||
if ( ! empty( $image['alt'] ) ) {
|
||||
update_post_meta( $attachment_id, '_wp_attachment_image_alt', wc_clean( $image['alt'] ) );
|
||||
}
|
||||
|
||||
// Set the image name if present.
|
||||
if ( ! empty( $image['name'] ) ) {
|
||||
wp_update_post( array( 'ID' => $attachment_id, 'post_title' => $image['name'] ) );
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $gallery ) ) {
|
||||
$product->set_gallery_image_ids( $gallery );
|
||||
}
|
||||
} else {
|
||||
$product->set_image_id( '' );
|
||||
$product->set_gallery_image_ids( array() );
|
||||
}
|
||||
|
||||
return $product;
|
||||
}
|
||||
|
||||
/**
|
||||
* Save product shipping data.
|
||||
*
|
||||
* @param WC_Product $product Product instance.
|
||||
* @param array $data Shipping data.
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_product_shipping_data( $product, $data ) {
|
||||
// Virtual.
|
||||
if ( isset( $data['virtual'] ) && true === $data['virtual'] ) {
|
||||
$product->set_weight( '' );
|
||||
$product->set_height( '' );
|
||||
$product->set_length( '' );
|
||||
$product->set_width( '' );
|
||||
} else {
|
||||
if ( isset( $data['weight'] ) ) {
|
||||
$product->set_weight( $data['weight'] );
|
||||
}
|
||||
|
||||
// Height.
|
||||
if ( isset( $data['height'] ) ) {
|
||||
$product->set_height( $data['height'] );
|
||||
}
|
||||
|
||||
// Width.
|
||||
if ( isset( $data['width'] ) ) {
|
||||
$product->set_width( $data['width'] );
|
||||
}
|
||||
|
||||
// Length.
|
||||
if ( isset( $data['length'] ) ) {
|
||||
$product->set_length( $data['length'] );
|
||||
}
|
||||
}
|
||||
|
||||
// Shipping class.
|
||||
if ( isset( $data['shipping_class_id'] ) ) {
|
||||
$shipping_class_term = get_term_by( 'id', wc_clean( $data['shipping_class_id'] ), 'product_shipping_class' );
|
||||
|
||||
if ( $shipping_class_term ) {
|
||||
$product->set_shipping_class_id( $shipping_class_term->term_id );
|
||||
}
|
||||
}
|
||||
|
||||
return $product;
|
||||
}
|
||||
|
||||
/**
|
||||
* Save downloadable files.
|
||||
*
|
||||
* @param WC_Product $product Product instance.
|
||||
* @param array $downloads Downloads data.
|
||||
* @param int $deprecated Deprecated since 3.0.
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_downloadable_files( $product, $downloads ) {
|
||||
$files = array();
|
||||
foreach ( $downloads as $key => $file ) {
|
||||
if ( empty( $file['file'] ) ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
$download = new WC_Product_Download();
|
||||
$download->set_id( $key );
|
||||
$download->set_name( $file['name'] ? $file['name'] : wc_get_filename_from_url( $file['file'] ) );
|
||||
$download->set_file( apply_filters( 'woocommerce_file_download_path', $file['file'], $product, $key ) );
|
||||
$files[] = $download;
|
||||
}
|
||||
$product->set_downloads( $files );
|
||||
|
||||
return $product;
|
||||
}
|
||||
|
||||
/**
|
||||
* Save taxonomy terms.
|
||||
*
|
||||
* @param WC_Product $product Product instance.
|
||||
* @param array $terms Terms data.
|
||||
* @param string $taxonomy Taxonomy name.
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_taxonomy_terms( $product, $terms, $taxonomy = 'cat' ) {
|
||||
$term_ids = wp_list_pluck( $terms, 'id' );
|
||||
|
||||
if ( 'cat' === $taxonomy ) {
|
||||
$product->set_category_ids( $term_ids );
|
||||
} elseif ( 'tag' === $taxonomy ) {
|
||||
$product->set_tag_ids( $term_ids );
|
||||
}
|
||||
|
||||
return $product;
|
||||
}
|
||||
|
||||
/**
|
||||
* Save default attributes.
|
||||
*
|
||||
* @since 3.0.0
|
||||
*
|
||||
* @param WC_Product $product Product instance.
|
||||
* @param array $data Row data.
|
||||
* @return WC_Product
|
||||
*/
|
||||
protected function save_default_attributes( $product, $data ) {
|
||||
if ( isset( $data['default_attributes'] ) && is_array( $data['default_attributes'] ) ) {
|
||||
|
||||
$attributes = $product->get_attributes();
|
||||
$default_attributes = array();
|
||||
|
||||
foreach ( $data['default_attributes'] as $attribute ) {
|
||||
$attribute_id = 0;
|
||||
$attribute_name = '';
|
||||
|
||||
// Check ID for global attributes or name for product attributes.
|
||||
if ( ! empty( $attribute['id'] ) ) {
|
||||
$attribute_id = absint( $attribute['id'] );
|
||||
$attribute_name = wc_attribute_taxonomy_name_by_id( $attribute_id );
|
||||
} elseif ( ! empty( $attribute['name'] ) ) {
|
||||
$attribute_name = sanitize_title( $attribute['name'] );
|
||||
}
|
||||
|
||||
if ( ! $attribute_id && ! $attribute_name ) {
|
||||
continue;
|
||||
}
|
||||
|
||||
if ( isset( $attributes[ $attribute_name ] ) ) {
|
||||
$_attribute = $attributes[ $attribute_name ];
|
||||
|
||||
if ( $_attribute['is_variation'] ) {
|
||||
$value = isset( $attribute['option'] ) ? wc_clean( stripslashes( $attribute['option'] ) ) : '';
|
||||
|
||||
if ( ! empty( $_attribute['is_taxonomy'] ) ) {
|
||||
// If dealing with a taxonomy, we need to get the slug from the name posted to the API.
|
||||
$term = get_term_by( 'name', $value, $attribute_name );
|
||||
|
||||
if ( $term && ! is_wp_error( $term ) ) {
|
||||
$value = $term->slug;
|
||||
} else {
|
||||
$value = sanitize_title( $value );
|
||||
}
|
||||
}
|
||||
|
||||
if ( $value ) {
|
||||
$default_attributes[ $attribute_name ] = $value;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$product->set_default_attributes( $default_attributes );
|
||||
}
|
||||
|
||||
return $product;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,277 @@
|
|||
<?php
|
||||
/**
|
||||
* WooCommerce Product CSV importer
|
||||
*
|
||||
* @author Automattic
|
||||
* @category Admin
|
||||
* @package WooCommerce/Import
|
||||
* @version 3.1.0
|
||||
*/
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* Include dependencies.
|
||||
*/
|
||||
if ( ! class_exists( 'WC_Product_Importer', false ) ) {
|
||||
include_once( dirname( __FILE__ ) . '/abstract-wc-product-importer.php' );
|
||||
}
|
||||
|
||||
/**
|
||||
* WC_Product_CSV_Importer Class.
|
||||
*/
|
||||
class WC_Product_CSV_Importer extends WC_Product_Importer {
|
||||
|
||||
/**
|
||||
* Initialize importer.
|
||||
*
|
||||
* @param string $file File to read.
|
||||
* @param array $args Arguments for the parser.
|
||||
*/
|
||||
public function __construct( $file, $params = array() ) {
|
||||
$default_args = array(
|
||||
'start_pos' => 0, // File pointer start.
|
||||
'end_pos' => -1, // File pointer end.
|
||||
'lines' => -1, // Max lines to read.
|
||||
'mapping' => array(), // Column mapping. csv_heading => schema_heading.
|
||||
'parse' => false, // Whether to sanitize and format data.
|
||||
'delimiter' => ',', // CSV delimiter.
|
||||
);
|
||||
|
||||
$this->params = wp_parse_args( $params, $default_args );
|
||||
$this->file = $file;
|
||||
|
||||
$this->read_file();
|
||||
}
|
||||
|
||||
/**
|
||||
* Read file.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function read_file() {
|
||||
if ( false !== ( $handle = fopen( $this->file, 'r' ) ) ) {
|
||||
$this->raw_keys = fgetcsv( $handle, 0, $this->params['delimiter'] );
|
||||
|
||||
if ( 0 !== $this->params['start_pos'] ) {
|
||||
fseek( $handle, (int) $this->params['start_pos'] );
|
||||
}
|
||||
|
||||
while ( false !== ( $row = fgetcsv( $handle, 0, $this->params['delimiter'] ) ) ) {
|
||||
$this->raw_data[] = $row;
|
||||
|
||||
if ( ( $this->params['end_pos'] > 0 && ftell( $handle ) >= $this->params['end_pos'] ) || 0 === --$this->params['lines'] ) {
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
if ( ! empty( $this->params['mapping'] ) ) {
|
||||
$this->set_mapped_keys();
|
||||
}
|
||||
|
||||
if ( $this->params['parse'] ) {
|
||||
$this->set_parsed_data();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Set file mapped keys.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function set_mapped_keys() {
|
||||
$mapping = $this->params['mapping'];
|
||||
|
||||
foreach ( $this->raw_keys as $key ) {
|
||||
$this->mapped_keys[] = isset( $mapping[ $key ] ) ? $mapping[ $key ] : $key;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a comma-delineated field from a CSV.
|
||||
*
|
||||
* @param string $field
|
||||
* @return array
|
||||
*/
|
||||
protected function parse_comma_field( $field ) {
|
||||
if ( empty( $field ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
return array_map( 'esc_attr', array_map( 'trim', explode( ',', $field ) ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a field that is generally '1' or '0' but can be something else.
|
||||
*
|
||||
* @param string $field
|
||||
* @return bool|string
|
||||
*/
|
||||
protected function parse_bool_field( $field ) {
|
||||
if ( '0' === $field ) {
|
||||
return false;
|
||||
}
|
||||
|
||||
if ( '1' === $field ) {
|
||||
return true;
|
||||
}
|
||||
|
||||
// Don't return explicit true or false for empty fields or values like 'notify'.
|
||||
return esc_attr( $field );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a float value field.
|
||||
*
|
||||
* @param string $field
|
||||
* @return float|string
|
||||
*/
|
||||
protected function parse_float_field( $field ) {
|
||||
if ( '' === $field ) {
|
||||
return $field;
|
||||
}
|
||||
|
||||
return floatval( $field );
|
||||
}
|
||||
|
||||
/**
|
||||
* Parse a category field from a CSV.
|
||||
* Categories are separated by commas and subcategories are "parent > subcategory".
|
||||
*
|
||||
* @param string $field
|
||||
* @return array of arrays with "parent" and "name" keys.
|
||||
*/
|
||||
protected function parse_categories( $field ) {
|
||||
if ( empty( $field ) ) {
|
||||
return array();
|
||||
}
|
||||
|
||||
$sections = array_map( 'trim', explode( ',', $field ) );
|
||||
$categories = array();
|
||||
|
||||
foreach ( $sections as $section ) {
|
||||
|
||||
// Top level category.
|
||||
if ( false === strpos( $section, '>' ) ) {
|
||||
$categories[] = array(
|
||||
'parent' => false,
|
||||
'name' => esc_attr( $section ),
|
||||
);
|
||||
|
||||
// Subcategory.
|
||||
} else {
|
||||
$chunks = array_map( 'trim', explode( '>', $section ) );
|
||||
$categories[] = array(
|
||||
'parent' => esc_attr( reset( $chunks ) ),
|
||||
'name' => esc_attr( end( $chunks ) ),
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
return $categories;
|
||||
}
|
||||
|
||||
/**
|
||||
* Map and format raw data to known fields.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
protected function set_parsed_data() {
|
||||
|
||||
/**
|
||||
* Columns not mentioned here will get parsed with 'esc_attr'.
|
||||
* column_name => callback.
|
||||
*/
|
||||
$data_formatting = array(
|
||||
'id' => 'absint',
|
||||
'status' => array( $this, 'parse_bool_field' ),
|
||||
'featured' => array( $this, 'parse_bool_field' ),
|
||||
'date_on_sale_from' => 'strtotime',
|
||||
'date_on_sale_to' => 'strtotime',
|
||||
'manage_stock' => array( $this, 'parse_bool_field' ),
|
||||
'backorders' => array( $this, 'parse_bool_field' ),
|
||||
'sold_individually' => array( $this, 'parse_bool_field' ),
|
||||
'width' => array( $this, 'parse_float_field' ),
|
||||
'length' => array( $this, 'parse_float_field' ),
|
||||
'height' => array( $this, 'parse_float_field' ),
|
||||
'weight' => array( $this, 'parse_float_field' ),
|
||||
'reviews_allowed' => array( $this, 'parse_bool_field' ),
|
||||
'purchase_note' => 'wp_kses_post',
|
||||
'price' => 'wc_format_decimal',
|
||||
'regular_price' => 'wc_format_decimal',
|
||||
'stock_quantity' => 'absint',
|
||||
'category_ids' => array( $this, 'parse_categories' ),
|
||||
'tag_ids' => array( $this, 'parse_comma_field' ),
|
||||
'images' => array( $this, 'parse_comma_field' ),
|
||||
'upsell_ids' => array( $this, 'parse_comma_field' ),
|
||||
'cross_sell_ids' => array( $this, 'parse_comma_field' ),
|
||||
'download_limit' => 'absint',
|
||||
'download_expiry' => 'absint',
|
||||
);
|
||||
|
||||
/**
|
||||
* @todo switch these to some standard, slug format.
|
||||
*/
|
||||
$regex_match_data_formatting = array(
|
||||
'/Attribute * Value\(s\)/' => array( $this, 'parse_comma_field' ),
|
||||
'/Attribute * Visible/' => array( $this, 'parse_bool_field' ),
|
||||
'/Download * URL/' => 'esc_url',
|
||||
);
|
||||
|
||||
$headers = ! empty( $this->mapped_keys ) ? $this->mapped_keys : $this->raw_keys;
|
||||
$parse_functions = array();
|
||||
|
||||
// Figure out the parse function for each column.
|
||||
foreach ( $headers as $index => $heading ) {
|
||||
|
||||
$parse_function = 'esc_attr';
|
||||
if ( isset( $data_formatting[ $heading ] ) ) {
|
||||
$parse_function = $data_formatting[ $heading ];
|
||||
} else {
|
||||
foreach ( $regex_match_data_formatting as $regex => $callback ) {
|
||||
if ( preg_match( $regex, $heading ) ) {
|
||||
$parse_function = $callback;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
$parse_functions[] = $parse_function;
|
||||
}
|
||||
|
||||
// Parse the data.
|
||||
foreach ( $this->raw_data as $row ) {
|
||||
$item = array();
|
||||
foreach ( $row as $index => $field ) {
|
||||
$item[ $headers[ $index ] ] = call_user_func( $parse_functions[ $index ], $field );
|
||||
}
|
||||
$this->parsed_data[] = $item;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Process importer.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function import() {
|
||||
$data = array(
|
||||
'imported' => array(),
|
||||
'failed' => array(),
|
||||
);
|
||||
|
||||
foreach ( $this->parsed_data as $parsed_data ) {
|
||||
$result = $this->process_item( $parsed_data );
|
||||
|
||||
if ( is_wp_error( $result ) ) {
|
||||
$data['failed'][] = $result;
|
||||
} else {
|
||||
$data['imported'][] = $result;
|
||||
}
|
||||
}
|
||||
|
||||
return $data;
|
||||
}
|
||||
}
|
|
@ -0,0 +1,63 @@
|
|||
<?php
|
||||
/**
|
||||
* WooCommerce Importer Interface
|
||||
*
|
||||
* @author Automattic
|
||||
* @category Admin
|
||||
* @package WooCommerce/Import
|
||||
* @version 3.1.0
|
||||
*/
|
||||
if ( ! defined( 'ABSPATH' ) ) {
|
||||
exit;
|
||||
}
|
||||
|
||||
/**
|
||||
* WC_Importer_Interface class.
|
||||
*/
|
||||
interface WC_Importer_Interface {
|
||||
|
||||
/**
|
||||
* Process importation.
|
||||
* Returns an array with the imported and failed items.
|
||||
* 'imported' contains a list of IDs.
|
||||
* 'failed' contains a list of WP_Error objects.
|
||||
*
|
||||
* Example:
|
||||
* ['imported' => [], 'failed' => []]
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function import();
|
||||
|
||||
/**
|
||||
* Get file raw keys.
|
||||
*
|
||||
* CSV - Headers.
|
||||
* XML - Element names.
|
||||
* JSON - Keys
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_raw_keys();
|
||||
|
||||
/**
|
||||
* Get file mapped headers.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_mapped_keys();
|
||||
|
||||
/**
|
||||
* Get raw data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_raw_data();
|
||||
|
||||
/**
|
||||
* Get parsed data.
|
||||
*
|
||||
* @return array
|
||||
*/
|
||||
public function get_parsed_data();
|
||||
}
|
|
@ -4,162 +4,171 @@
|
|||
* Meta
|
||||
* @package WooCommerce\Tests\Importer
|
||||
*/
|
||||
class WC_Tests_Product_Importer extends WC_Unit_Test_Case {
|
||||
class WC_Tests_Product_CSV_Importer extends WC_Unit_Test_Case {
|
||||
|
||||
/**
|
||||
* Test CSV file path.
|
||||
*
|
||||
* @var string
|
||||
*/
|
||||
protected $csv_file = string;
|
||||
|
||||
/**
|
||||
* Load up the importer classes since they aren't loaded by default.
|
||||
*/
|
||||
public function setUp() {
|
||||
require_once ABSPATH . 'wp-admin/includes/import.php';
|
||||
if ( ! class_exists( 'WP_Importer' ) ) {
|
||||
$class_wp_importer = ABSPATH . 'wp-admin/includes/class-wp-importer.php';
|
||||
if ( file_exists( $class_wp_importer ) ) {
|
||||
require $class_wp_importer;
|
||||
}
|
||||
}
|
||||
$this->csv_file = dirname( __FILE__ ) . '/sample.csv';
|
||||
|
||||
$bootstrap = WC_Unit_Tests_Bootstrap::instance();
|
||||
require_once $bootstrap->plugin_dir . '/includes/admin/importers/class-wc-product-importer.php';
|
||||
require_once $bootstrap->plugin_dir . '/includes/import/class-wc-product-csv-importer.php';
|
||||
}
|
||||
|
||||
/**
|
||||
* Test parse_comma_field.
|
||||
* Test import.
|
||||
* @todo enable the importer again after conclude the parser.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_parse_comma_field() {
|
||||
$importer = new WC_Product_Importer();
|
||||
|
||||
$field1 = 'thing 1, thing 2, thing 3';
|
||||
$field2 = 'thing 1';
|
||||
$field3 = '';
|
||||
|
||||
$expected1 = array( 'thing 1', 'thing 2', 'thing 3' );
|
||||
$expected2 = array( 'thing 1' );
|
||||
$expected3 = array();
|
||||
|
||||
$this->assertEquals( $expected1, $importer->parse_comma_field( $field1 ) );
|
||||
$this->assertEquals( $expected2, $importer->parse_comma_field( $field2 ) );
|
||||
$this->assertEquals( $expected3, $importer->parse_comma_field( $field3 ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test parse_bool_field.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_parse_bool_field() {
|
||||
$importer = new WC_Product_Importer();
|
||||
|
||||
$field1 = '1';
|
||||
$field2 = '0';
|
||||
$field3 = '';
|
||||
$field4 = 'notify';
|
||||
|
||||
$this->assertEquals( true, $importer->parse_bool_field( $field1 ) );
|
||||
$this->assertEquals( false, $importer->parse_bool_field( $field2 ) );
|
||||
$this->assertEquals( '', $importer->parse_bool_field( $field3 ) );
|
||||
$this->assertEquals( 'notify', $importer->parse_bool_field( $field4 ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test parse_float_field.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_parse_float_field() {
|
||||
$importer = new WC_Product_Importer();
|
||||
|
||||
$field1 = '12.45';
|
||||
$field2 = '5';
|
||||
$field3 = '';
|
||||
|
||||
$this->assertEquals( 12.45, $importer->parse_float_field( $field1 ) );
|
||||
$this->assertEquals( 5, $importer->parse_float_field( $field2 ) );
|
||||
$this->assertEquals( '', $importer->parse_float_field( $field3 ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test parse_categories.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_parse_categories() {
|
||||
$importer = new WC_Product_Importer();
|
||||
|
||||
$field1 = 'category1';
|
||||
$field2 = 'category1, category2, category1 > subcategory1, category1 > subcategory2';
|
||||
$field3 = '';
|
||||
|
||||
$expected1 = array(
|
||||
array(
|
||||
'parent' => false,
|
||||
'name' => 'category1'
|
||||
)
|
||||
);
|
||||
$expected2 = array(
|
||||
array(
|
||||
'parent' => false,
|
||||
'name' => 'category1'
|
||||
),
|
||||
array(
|
||||
'parent' => false,
|
||||
'name' => 'category2'
|
||||
),
|
||||
array(
|
||||
'parent' => 'category1',
|
||||
'name' => 'subcategory1'
|
||||
),
|
||||
array(
|
||||
'parent' => 'category1',
|
||||
'name' => 'subcategory2'
|
||||
)
|
||||
);
|
||||
$expected3 = array();
|
||||
|
||||
$this->assertEquals( $expected1, $importer->parse_categories( $field1 ) );
|
||||
$this->assertEquals( $expected2, $importer->parse_categories( $field2 ) );
|
||||
$this->assertEquals( $expected3, $importer->parse_categories( $field3 ) );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test parse_data.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_parse_data() {
|
||||
$importer = new WC_Product_Importer();
|
||||
|
||||
$data = array(
|
||||
'headers' => array( 'id', 'weight', 'price', 'category_ids', 'tag_ids', 'extra_thing', 'featured', 'Download 1 URL' ),
|
||||
'data' => array(
|
||||
array( '', '12.2', '12.50', 'category1, category1 > subcategory', 'products, things, etc', 'metadata', '1', '' ),
|
||||
array( '12', '', '5', 'category2', '', '', '0', 'http://www.example.com' ),
|
||||
)
|
||||
public function test_import() {
|
||||
$mapped = array(
|
||||
'Type' => 'type',
|
||||
'SKU' => 'sku',
|
||||
'Name' => 'name',
|
||||
'Status' => 'status',
|
||||
'Regular price' => 'regular_price',
|
||||
);
|
||||
|
||||
$expected = array(
|
||||
$args = array(
|
||||
'mapping' => $mapped,
|
||||
'parse' => true,
|
||||
);
|
||||
|
||||
$importer = new WC_Product_CSV_Importer( $this->csv_file, $args );
|
||||
$results = $importer->import();
|
||||
|
||||
$this->assertEquals( 3, count( $results['imported'] ) );
|
||||
$this->assertEquals( 0, count( $results['failed'] ) );
|
||||
|
||||
// Exclude imported products.
|
||||
foreach ( $results['imported'] as $id ) {
|
||||
wp_delete_post( $id );
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* Test get_raw_keys.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_get_raw_keys() {
|
||||
$importer = new WC_Product_CSV_Importer( $this->csv_file );
|
||||
$raw_keys = array(
|
||||
'Type',
|
||||
'SKU',
|
||||
'Name',
|
||||
'Status',
|
||||
'Regular price',
|
||||
);
|
||||
|
||||
$this->assertEquals( $raw_keys, $importer->get_raw_keys() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test get_mapped_keys.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_get_mapped_keys() {
|
||||
$mapped = array(
|
||||
'Type' => 'type',
|
||||
'SKU' => 'sku',
|
||||
'Name' => 'name',
|
||||
'Status' => 'status',
|
||||
'Regular price' => 'regular_price',
|
||||
);
|
||||
|
||||
$args = array(
|
||||
'mapping' => $mapped,
|
||||
);
|
||||
|
||||
$importer = new WC_Product_CSV_Importer( $this->csv_file, $args );
|
||||
|
||||
$this->assertEquals( array_values( $mapped ), $importer->get_mapped_keys() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test get_raw_data.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_get_raw_data() {
|
||||
$importer = new WC_Product_CSV_Importer( $this->csv_file, array( 'parse' => false ) );
|
||||
$items = array(
|
||||
array(
|
||||
'id' => 0,
|
||||
'weight' => 12.2,
|
||||
'price' => '12.50',
|
||||
'category_ids' => array(
|
||||
array( 'parent' => false, 'name' => 'category1' ),
|
||||
array( 'parent' => 'category1', 'name' => 'subcategory' ),
|
||||
),
|
||||
'tag_ids' => array( 'products', 'things', 'etc' ),
|
||||
'extra_thing' => 'metadata',
|
||||
'featured' => true,
|
||||
'Download 1 URL' => '',
|
||||
'simple',
|
||||
'PRODUCT-01',
|
||||
'Imported Product 1',
|
||||
1,
|
||||
40,
|
||||
),
|
||||
array(
|
||||
'id' => 12,
|
||||
'weight' => '',
|
||||
'price' => '5',
|
||||
'category_ids' => array(
|
||||
array( 'parent' => false, 'name' => 'category2' ),
|
||||
'simple',
|
||||
'PRODUCT-02',
|
||||
'Imported Product 2',
|
||||
1,
|
||||
41,
|
||||
),
|
||||
'tag_ids' => array(),
|
||||
'extra_thing' => '',
|
||||
'featured' => false,
|
||||
'Download 1 URL' => 'http://www.example.com',
|
||||
array(
|
||||
'simple',
|
||||
'PRODUCT-03',
|
||||
'Imported Product 3',
|
||||
1,
|
||||
42,
|
||||
),
|
||||
);
|
||||
|
||||
$this->assertEquals( $expected, $importer->parse_data( $data ) );
|
||||
$this->assertEquals( $items, $importer->get_raw_data() );
|
||||
}
|
||||
|
||||
/**
|
||||
* Test get_parsed_data.
|
||||
* @since 3.1.0
|
||||
*/
|
||||
public function test_get_parsed_data() {
|
||||
$mapped = array(
|
||||
'Type' => 'type',
|
||||
'SKU' => 'sku',
|
||||
'Name' => 'name',
|
||||
'Status' => 'status',
|
||||
'Regular price' => 'regular_price',
|
||||
);
|
||||
|
||||
$args = array(
|
||||
'mapping' => $mapped,
|
||||
'parse' => true,
|
||||
);
|
||||
|
||||
$importer = new WC_Product_CSV_Importer( $this->csv_file, $args );
|
||||
$items = array(
|
||||
array(
|
||||
'type' => 'simple',
|
||||
'sku' => 'PRODUCT-01',
|
||||
'name' => 'Imported Product 1',
|
||||
'status' => 1,
|
||||
'regular_price' => 40,
|
||||
),
|
||||
array(
|
||||
'type' => 'simple',
|
||||
'sku' => 'PRODUCT-02',
|
||||
'name' => 'Imported Product 2',
|
||||
'status' => 1,
|
||||
'regular_price' => 41,
|
||||
),
|
||||
array(
|
||||
'type' => 'simple',
|
||||
'sku' => 'PRODUCT-03',
|
||||
'name' => 'Imported Product 3',
|
||||
'status' => 1,
|
||||
'regular_price' => 42,
|
||||
),
|
||||
);
|
||||
|
||||
$this->assertEquals( $items, $importer->get_parsed_data() );
|
||||
}
|
||||
}
|
||||
|
|
|
@ -0,0 +1,4 @@
|
|||
Type,SKU,Name,Status,Regular price
|
||||
simple,PRODUCT-01,Imported Product 1,1,40
|
||||
simple,PRODUCT-02,Imported Product 2,1,41
|
||||
simple,PRODUCT-03,Imported Product 3,1,42
|
|
Loading…
Reference in New Issue