Okay guys, here’s the scenario.
I’m am trying to setup a function that will automatically duplicate a post (when published) over to another post type. So, a regular blog post is published, and when it is published, all of its information is copied over to a custom post type (for WP ECommerce), automatically creating a store entry for that blog post.
I’ve written a function that takes care of picking up all of the information and inserting it into a new post with wp_insert_post(). It all works great, except for one thing. I’m using wp_set_object_terms() to set the store category ids of the new product based on the tags of the blog post. However, for some reason, wp_set_object_terms() never works.
Now, here’s the catch. I’m running all of this on a multi site install and am using threeWP_Broadcast to allow me to cross publish posts. The blog posts (which have to be copied to the store) are published from a sub site and broadcasted to the main site. The store is on the main site.
So I’ve hooked my custom function into the broadcast plugin so that it gets fired when a post is broadcasted from a sub site to the main site.
Everything with my function works great, except for the wp_set_object_terms()
Here’s function:
function copy_post_to_store_product() {
global $blog_id;
global $wpdb;
// only copy the post over if on main site, not subsites
if ($blog_id == 1) {
$args = array('numberposts' => 1);
$latest = get_posts($args);
foreach($latest as $post) : setup_postdata($post);
// if NOT Tip and NOT source avail, create product
// if source, price is ALWAYS $4
// auto create for source files only -- regular post type
$custom_meta = get_post_custom($post->ID);
// what kind of post is this?
$post_type = $custom_meta['cgc_post_type'][0];
// does this post have a source file product associated with it?
$source_avail = $custom_meta['cgc_source_avail'][0];
// source file price
$price = '4';
$product_info = array(
'post_title' => $post->post_title,
'post_type' => 'wpsc-product',
'post_content' => $post->post_content,
'post_author' => $post->post_author,
'post_status' => 'draft',
);
if($post_type == 'Regular' && $source_avail == true) {
// only auto create product for Regular posts that have source files for sale
$product_id = wp_insert_post($product_info);
update_post_meta($product_id, '_wpsc_price', $price);
if (has_tag('blender', $post->ID))
{
$product_cats = array(23,32);
}
elseif (has_tag('max', $post->ID)) {
$product_cats = array(23,34);
}
elseif (has_tag('modo', $post->ID)) {
$product_cats = array(23,19);
}
// set the product categories
wp_set_object_terms($product_id, $product_cats, 'wpsc_product_category' );
}
endforeach;
}
}
The function works by retrieving the latest post from the main site ($blog_id == 1) and copying all of its information into variables for wp_insert_post().
Now, the really interesting thing is that works perfectly if I attach the function to a publish_post hook, but, unfortunately, I can’t do that because that hook doesn’t fire when a post is broadcasted.
Any ideas would be hugely appreciated.
I don’t have a solution, but there’s a ticket #20541 on Make WordPress Core.
Apparently a call to
switch_to_blog()
would not repopulate$wp_taxomies
, which these taxonomies rely on.I just came across the same problem and worked out a solution with help from Patriek’s answer. You just need to make sure that
wp_set_object_terms()
is run after the custom taxonomy is registered. Sinceregister_taxonomy()
is usually run atinit
, you can also run your function atinit
action hook, but with a lower priority so it runs later.This way, the taxonomy is guaranteed to be available when your function is run.
Forget about the previous answer, You say that it works fine with
publish_post
hook then you create a plugin in your mu-plugins (âmust-useâ plugins) which you can hook your function to and just addswitch_to_blog
before you start inserting and updating things in the database and then revert back usingrestore_current_blog()
So
Just throwing this out there: I had a similar problem once where a taxonomy was not available at the time my function was running. This could either be a problem with wp ecommerce or the broadcast plugin.
Check if the wp ecommerce plugin uses this syntax for registering their taxonomy
Check if they appended the ,0. This sets the priority of the taxonomy function to 0. Which would be as early as possible.
Is it possible to set priority for the broadcast hook?
i have got the solution,
i have investigated the
taxonomy.php
file, then i found that the problem is on the functiontaxonomy_exists()
. Since theswitch_to_blog()
function don’t read the theme and plugin code, so it will not detect our registered taxonomy on the other blog, so we need to manualy init that, just before thetaxonomy_exists()
functionPut this code before the
wp_set_object_terms()
function:After trying so much trial and error I have found the real issue is that you need to set your function of setting up a taxonomy for CTP in Init with low priority.
add_action( 'init', 'your_function_product', 20 );
thank you guys, you saved my life 🙂
Low priority (20) for the function that inserts the post(s) will do the trick!
Here is an approach for wp_insert_post() in functions.php: