Projekt

Allgemein

Profil

« Zurück | Weiter » 

Revision 53f7b408

Von Tamino vor fast 4 Jahren hinzugefügt

  • ID 53f7b40841432dec27fcbb0772dc4adb11d9cda7
  • Vorgänger 82034151
  • Nachfolger 196da0e7

WebshopApi: ShopConnector WooCommerce erstellt

Funktion get_version implementiert.

Unterschiede anzeigen:

SL/ShopConnector/ALL.pm
use strict;
use SL::ShopConnector::Shopware;
use SL::ShopConnector::WooCommerce;
my %shop_connector_by_name = (
shopware => 'SL::ShopConnector::Shopware',
woocommerce => 'SL::ShopConnector::WooCommerce',
);
my %shop_connector_by_connector = (
shopware => 'SL::ShopConnector::Shopware',
woocommerce => 'SL::ShopConnector::WooCommerce',
);
my @shop_connector_order = qw(
woocommerce
shopware
);
my @shop_connectors = (
{ id => "shopware", description => "Shopware" },
{ id => "woocommerce", description => "WooCommerce" },
);
SL/ShopConnector/Shopware.pm
my @daten = @{$import->{data}};
my %categories = map { ($_->{id} => $_) } @daten;
my @categories_tree;
for(@daten) {
# ignore root with id=1
if( $_->{id} == 1) {
next;
}
my $parent = $categories{$_->{parentId}};
$parent->{children} ||= [];
push @{$parent->{children}},$_;
if($parent && $parent->{id} != 1) {
$parent->{children} ||= [];
push @{$parent->{children}},$_;
} else {
push @categories_tree, $_;
}
}
return \@daten;
return \@categories_tree;
}
sub get_version {
SL/ShopConnector/WooCommerce.pm
package SL::ShopConnector::WooCommerce;
use strict;
use parent qw(SL::ShopConnector::Base);
use SL::JSON;
use LWP::UserAgent;
use LWP::Authen::Digest;
use SL::DB::ShopOrder;
use SL::DB::ShopOrderItem;
use SL::DB::History;
use SL::DB::File;
use Data::Dumper;
use SL::Helper::Flash;
use Encode qw(encode_utf8);
sub get_one_order {
my ($self, $order_id) = @_;
my $dbh = SL::DB::client;
my $number_of_orders = 0;
my @errors;
my $answer = $self->send_request(
"orders/" . $order_id,
undef,
"get"
);
my %fetched_orders;
if($answer->{success}) {
my $shoporder = $answer->{data};
$dbh->with_transaction( sub{
unless ($self->import_data_to_shop_order($shoporder)) { return 0;}
#update status on server
$shoporder->{status} = "processing";
my %new_status = ( status => "processing" );
my $status_json = SL::JSON::to_json( \%new_status);
$answer = $self->send_request("orders/$shoporder->{id}", $status_json, "put");
unless($answer->{success}){
push @errors,($::locale->text('Saving failed. Error message from the server: #1', $answer->message));
return 0
}
1;
})or do {
push @errors,($::locale->text('Saving failed. Error message from the database: #1', $dbh->error));
};
if(@errors){
flash_later('error', $::locale->text('Errors: #1', @errors));
} else {
$number_of_orders++;
}
%fetched_orders = (shop_description => $self->config->description, number_of_orders => $number_of_orders);
} else {
my %error_msg = (
shop_id => $self->config->id,
shop_description => $self->config->description,
message => $answer->{message},
error => 1,
);
%fetched_orders = %error_msg;
}
return \%fetched_orders;
}
sub get_new_orders {
my ($self) = @_;
my $dbh = SL::DB::client;
my $otf = $self->config->orders_to_fetch || 10;
my $number_of_orders = 0;
my @errors;
my $answer = $self->send_request(
"orders",
undef,
"get",
"&per_page=$otf&status=pending"
);
my %fetched_orders;
if($answer->{success}) {
my $orders = $answer->{data};
foreach my $shoporder(@{$orders}){
$dbh->with_transaction( sub{
unless ($self->import_data_to_shop_order($shoporder)) { return 0;}
#update status on server
$shoporder->{status} = "processing";
my %new_status = ( status => "processing" );
my $status_json = SL::JSON::to_json( \%new_status);
$answer = $self->send_request("orders/$shoporder->{id}", $status_json, "put");
unless($answer->{success}){
push @errors,($::locale->text('Saving failed. Error message from the server: #1', $answer->message));
return 0;
}
1;
})or do {
push @errors,($::locale->text('Saving failed. Error message from the database: #1', $dbh->error));
};
if(@errors){
flash_later('error', $::locale->text('Errors: #1', @errors));
} else {
$number_of_orders++;
}
}
%fetched_orders = (shop_description => $self->config->description, number_of_orders => $number_of_orders);
} else {
my %error_msg = (
shop_id => $self->config->id,
shop_description => $self->config->description,
message => $answer->{message},
error => 1,
);
%fetched_orders = %error_msg;
}
return \%fetched_orders;
}
sub import_data_to_shop_order {
my ( $self, $import ) = @_;
my $shop_order = $self->map_data_to_shoporder($import);
$shop_order->save;
my $id = $shop_order->id;
my @positions = sort { Sort::Naturally::ncmp($a->{"sku"}, $b->{"sku"}) } @{ $import->{line_items} };
my $position = 1;
my $answer= $self->send_request("taxes");
unless ($answer->{success}){ return 0;}
my %taxes = map { ($_->{id} => $_) } @{ $answer->{data} };
my $active_price_source = $self->config->price_source;
#Mapping Positions
foreach my $pos(@positions) {
my $price = $::form->round_amount($pos->{total},2);
my $tax_id = $pos->{taxes}[0]->{id};
my $tax_rate = $taxes{ $tax_id }{rate};
my %pos_columns = ( description => $pos->{name},
partnumber => $pos->{sku}, # sku has to be a valid value in WooCommerce
price => $price,
quantity => $pos->{quantity},
position => $position,
tax_rate => $tax_rate,
shop_trans_id => $pos->{product_id},
shop_order_id => $id,
active_price_source => $active_price_source,
);
my $pos_insert = SL::DB::ShopOrderItem->new(%pos_columns);
$pos_insert->save;
$position++;
}
$shop_order->positions($position-1);
my $customer = $shop_order->get_customer;
if(ref($customer)){
$shop_order->kivi_customer_id($customer->id);
}
$shop_order->save;
}
sub map_data_to_shoporder {
my ($self, $import) = @_;
my $parser = DateTime::Format::Strptime->new( pattern => '%Y-%m-%dT%H:%M:%S',
locale => 'de_DE',
time_zone => 'local'
);
my $shop_id = $self->config->id;
# Mapping to table shoporders. See https://woocommerce.github.io/woocommerce-rest-api-docs/?shell#order-properties
my %columns = (
#billing
billing_firstname => $import->{billing}->{first_name},
billing_lastname => $import->{billing}->{last_name},
#address_1 address_2
billing_street => $import->{billing}->{address_1} . ($import->{billing}->{address_2} ? " " . $import->{billing}->{address_2} : ""),
# ???
billing_city => $import->{billing}->{city},
#state
# ???
billing_zipcode => $import->{billing}->{postcode},
billing_country => $import->{billing}->{country},
billing_email => $import->{billing}->{email},
billing_phone => $import->{billing}->{phone},
#billing_greeting => "",
#billing_fax => "",
#billing_vat => "",
#billing_company => "",
#billing_department => "",
#customer
#customer_id
shop_customer_id => $import->{customer_id},
shop_customer_number => $import->{customer_id},
#customer_ip_address
remote_ip => $import->{customer_ip_address},
#customer_user_agent
#customer_note
shop_customer_comment => $import->{customer_note},
#customer_city => "",
#customer_company => "",
#customer_country => "",
#customer_department => "",
#customer_email => "",
#customer_fax => "",
#customer_firstname => "",
#customer_greeting => "",
#customer_lastname => "",
#customer_phone => "",
#customer_street => "",
#customer_vat => "",
#shipping
delivery_firstname => $import->{shipping}->{first_name},
delivery_lastname => $import->{shipping}->{last_name},
delivery_company => $import->{shipping}->{company},
#address_1 address_2
delivery_street => $import->{shipping}->{address_1} . ($import->{shipping}->{address_2} ? " " . $import->{shipping}->{address_2} : ""),
delivery_city => $import->{shipping}->{city},
#state ???
delivery_zipcode => $import->{shipping}->{postcode},
delivery_country => $import->{shipping}->{country},
#delivery_department => "",
#delivery_email => "",
#delivery_fax => "",
#delivery_phone => "",
#delivery_vat => "",
#other
#id
#parent_id
#number
shop_ordernumber => $import->{number},
#order_key
#created_via
#version
#status
#currency
#date_created
order_date => $parser->parse_datetime($import->{date_created}),
#date_created_gmt
#date_modified
#date_modified_gmt
#discount_total
#discount_tax
#shipping_total
shipping_costs => $import->{shipping_costs},
#shipping_tax
shipping_costs_net => $import->{shipping_costs} - $import->{shipping_tax},
#cart_tax
#total
amount => $import->{total},
#total_tax
netamount => $import->{total} - $import->{total_tax},
#prices_include_tax
tax_included => $import->{prices_include_tax} eq "true" ? 1 : 0,
#payment_method
# ??? payment_id => $import->{payment_method},
#payment_method_title
payment_description => $import->{payment}->{payment_method_title},
#transaction_id
shop_trans_id => $import->{id},
#date_paid
#date_paid_gmt
#date_completed
#date_completed_gmt
host => $import->{_links}->{self}[0]->{href},
#sepa_account_holder => "",
#sepa_bic => "",
#sepa_iban => "",
#shop_c_billing_id => "",
#shop_c_billing_number => "",
shop_c_delivery_id => $import->{shipping_lines}[0]->{id}, # ???
# not in Shop
shop_id => $shop_id,
);
my $shop_order = SL::DB::ShopOrder->new(%columns);
return $shop_order;
}
#TODO CVARS, tax and images
sub update_part {
my ($self, $shop_part, $todo) = @_;
#shop_part is passed as a param
die unless ref($shop_part) eq 'SL::DB::ShopPart';
my $part = SL::DB::Part->new(id => $shop_part->part_id)->load;
# CVARS to map
#my $cvars = {
# map {
# ($_->config->name => {
# value => $_->value_as_text,
# is_valid => $_->is_valid
# })
# }
# @{ $part->cvars_by_config }
#};
my @categories = ();
foreach my $row_cat ( @{ $shop_part->shop_category } ) {
my $temp = { ( id => @{$row_cat}[0], ) };
push ( @categories, $temp );
}
#my @upload_img = $shop_part->get_images;
my $partnumber = $::form->escape($part->partnumber);#don't accept / in articlenumber
my $stock_status = ($part->onhand ? "instock" : "outofstock");
my $status = ($shop_part->active ? "publish" : "private");
my $tax_n_price = $shop_part->get_tax_and_price;
my $price = $tax_n_price->{price};
#my $taxrate = $tax_n_price->{tax};
#my $tax_class = ($taxrate >= 16 ? "standard" : "reduzierter-preis");
my %shop_data;
if($todo eq "price"){
%shop_data = (
regular_price => $price,
);
}elsif($todo eq "stock"){
%shop_data = (
stock_status => $stock_status,
);
}elsif($todo eq "price_stock"){
%shop_data = (
stock_status => $stock_status,
regular_price => $price,
);
}elsif($todo eq "active"){
%shop_data = (
status => $status,
);
}elsif($todo eq "all"){
# mapping still missing attributes,metatags
%shop_data = (
sku => $partnumber,
name => $part->description,
stock_status => $stock_status,
regular_price => $price,
status => $status,
description=> $shop_part->shop_description,
short_description=> $shop_part->shop_description,
categories => [ @categories ],
#tax_class => $tax_class,
);
}
my $dataString = SL::JSON::to_json(\%shop_data);
$dataString = encode_utf8($dataString);
# LWP->post = create || LWP->put = update
my $answer = $self->send_request("products/", undef , "get" , "&sku=$partnumber");
if($answer->{success} && scalar @{$answer->{data}}){
#update
my $woo_shop_part_id = $answer->{data}[0]->{id};
$answer = $self->send_request("products/$woo_shop_part_id", $dataString, "put");
}else{
#upload
$answer = $self->send_request("products", $dataString, "post");
}
# don't know if this is needed
#if(@upload_img) {
# my $partnumber = $::form->escape($part->partnumber);#shopware don't accept / in articlenumber
# my $imgup = $self->connector->put($url . "api/generatearticleimages/$partnumber?useNumberAsId=true");
#}
return $answer->{success};
}
sub get_article {
my ($self) = @_;
my $partnumber = $_[1];
$partnumber = $::form->escape($partnumber);#don't accept / in partnumber
my $answer = $self->send_request("products/", undef , "get" , "&sku=$partnumber");
if($answer->{success} && scalar @{$answer->{data}}){
my $article = $answer->{data}[0];
return $article;
} else {
#What shut be here?
return $answer
}
}
sub get_categories {
my ($self) = @_;
my $answer = $self->send_request("products/categories");
unless($answer->{success}) {
return $answer;
}
my @data = @{$answer->{data}};
my %categories = map { ($_->{id} => $_) } @data;
my @categories_tree;
for(@data) {
my $parent = $categories{$_->{parent}};
if($parent) {
$parent->{children} ||= [];
push @{$parent->{children}},$_;
} else {
push @categories_tree, $_;
}
}
return \@categories_tree;
}
sub get_version {
my ($self) = @_;
my $answer = $self->send_request("system_status");
if($answer->{success}) {
my $version = $answer->{data}->{environment}->{version};
my %return = (
success => 1,
data => { version => $version },
);
return \%return;
} else {
return $answer;
}
}
sub create_url {
my ($self) = @_;
my $request = $_[1];
my $parameters = $_[2];
my $consumer_key = $self->config->login;
my $consumer_secret = $self->config->password;
my $protocol = $self->config->protocol;
my $server = $self->config->server;
my $port = $self->config->port;
my $path = $self->config->path;
return $protocol . "://". $server . ":" . $port . $path . $request . "?consumer_key=" . $consumer_key . "&consumer_secret=" . $consumer_secret . $parameters;
}
sub send_request {
my ($self) = @_;
my $request = $_[1];
my $json_data = $_[2];
my $method_type = $_[3];
my $parameters = $_[4];
my $ua = LWP::UserAgent->new;
my $url = $self->create_url( $request, $parameters );
my $answer;
if( $method_type eq "put" ) {
$answer = $ua->put($url, "Content-Type" => "application/json", Content => $json_data);
} elsif ( $method_type eq "post") {
$answer = $ua->post($url, "Content-Type" => "application/json", Content => $json_data);
} else {
$answer = $ua->get($url);
}
my $type = $answer->content_type;
my $status_line = $answer->status_line;
my %return;
if($answer->is_success && $type eq 'application/json'){
my $data_json = $answer->content;
my $json = SL::JSON::decode_json($data_json);
%return = (
success => 1,
data => $json,
);
}else{
%return = (
success => 0,
data => { version => $url . ": " . $status_line, data_type => $type },
message => "Error: $status_line",
);
}
#$main::lxdebug->dump(0, "TST: WooCommerce send_request return ", \%return);
return \%return;
}
1;
templates/webpages/shop_part/categories.html
[%- USE L -%]
[%- USE P -%]
[%- USE LxERP -%]
[%- USE Dumper -%]
[% LxERP.t8("Part") %]: [% HTML.escape(SELF.shop_part.part.displayable_name) %]<br>
[% LxERP.t8("Shop") %]: [% HTML.escape(SELF.shop_part.shop.description) %]<br>
<form action="controller.pl" method="post">
[% BLOCK recurse %]
[% # path = '' %]<!-- TODO: Pfad wg neuer Kategorie im Shop anlegen -->
[% FOREACH obj = data %]
<ul>
<li>
[% checked = '' %]
[% # path = path _ obj.name %]
[% # test = path.split('\|') %]
[% IF SELF.shop_part.shop_category.1.grep(obj.name).size %]
[% checked = 'checked' %]
[% ELSE %]
[% FOREACH categorie = categories_array %]
<ul>
<li>
[% checked = '' %]
[% FOREACH cat_row = SELF.shop_part.shop_category %]
[% IF cat_row.1.grep(obj.name).size %]
[% IF cat_row.0 == categorie.id %]
[% checked = 'checked' %]
[% END %]
[% END %]
[% END %]
[% L.checkbox_tag('categories[]',value=obj.id, checked=checked) %][% HTML.escape(obj.name) %][% L.hidden_tag("cat_id_" _ obj.id, obj.name) %]</li>
[% IF obj.childrenCount >= 1 %]
[% # path = path _ '|' %]
[% INCLUDE recurse data=obj.children %]
[% END %]
</ul>
[% L.checkbox_tag('categories[]',value=categorie.id, checked=checked) %][% HTML.escape(categorie.name) %][% L.hidden_tag("cat_id_" _ categorie.id, categorie.name) %]
</li>
[% IF categorie.children.size %]
[% INCLUDE recurse categories_array=categorie.children %]
[% END %]
</ul>
[% END %]
[% END %]
<div><h2>[% LxERP.t8("Shopcategories") %]</h2>
[% FOREACH row = CATEGORIES %]
<!-- TODO: Is still hardcoded 'Root' is shopware specified -->
[% IF row.name == 'Root' %]
[% IF row.childrenCount >= 1 %]
[% path = '' %]
[% INCLUDE recurse data=row.children path=path %]
[% END %]
[% END %]
[% END %]
[% # Dumper.dump_html( CATEGORIES ) %]
[% INCLUDE recurse categories_array=CATEGORIES %]
</div>
[% L.button_tag("kivi.ShopPart.save_categories(" _ SELF.shop_part.id _", " _ SELF.shop_part.shop.id _")", LxERP.t8("Save")) %]</td>
</form>

Auch abrufbar als: Unified diff