Version Description
- Fixed issue with the authors form not showing a the list of authors for a blog
- Simplified check for duplicate comments
- Code simplified for get_authors and get_author_form
- Fixed issue with wpdb prepare and integer keys by switching to a sub select query
- Make comment handling more robust
- Simplified functions to reduce messages in the log
Download this release
Release Info
Developer | Otto42 |
Plugin | Blogger Importer |
Version | 0.8 |
Comparing to | |
See all releases |
Code changes from version 0.6 to 0.8
- blogger-entry.php +103 -0
- blogger-importer-blog.php +812 -0
- blogger-importer-blogitem.php +167 -122
- blogger-importer-connector.php +195 -0
- blogger-importer-sanitize.php +134 -152
- blogger-importer-table.php +118 -0
- blogger-importer.css +62 -29
- blogger-importer.js +146 -0
- blogger-importer.php +490 -1034
- comment-entry.php +93 -0
- languages/blogger-importer-el.mo +0 -0
- languages/blogger-importer-el.po +371 -0
- languages/blogger-importer.pot +292 -220
- oauth.php +930 -926
- readme.txt +275 -135
- uninstall.php +21 -0
blogger-entry.php
ADDED
@@ -0,0 +1,103 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* @author Andy Clark
|
5 |
+
* @copyright 2013
|
6 |
+
* A data object representing the data to be added into Wordpress
|
7 |
+
*/
|
8 |
+
|
9 |
+
if (!class_exists('BloggerEntry'))
|
10 |
+
{
|
11 |
+
class BloggerEntry
|
12 |
+
{
|
13 |
+
var $links = array();
|
14 |
+
var $categories = array();
|
15 |
+
var $blogurl = '';
|
16 |
+
|
17 |
+
function parselinks() {
|
18 |
+
foreach ($this->links as $link) {
|
19 |
+
// save the self link as meta
|
20 |
+
if ($link['rel'] == 'self')
|
21 |
+
{
|
22 |
+
$postself = $link['href'];
|
23 |
+
$parts = parse_url($link['href']);
|
24 |
+
$this->old_permalink = $parts['path'];
|
25 |
+
}
|
26 |
+
|
27 |
+
// get the old URI for the page when available
|
28 |
+
if ($link['rel'] == 'alternate')
|
29 |
+
{
|
30 |
+
$parts = parse_url($link['href']);
|
31 |
+
$this->bookmark = $parts['path'];
|
32 |
+
}
|
33 |
+
|
34 |
+
// save the replies feed link as meta (ignore the comment form one)
|
35 |
+
if ($link['rel'] == 'replies' && false === strpos($link['href'], '#comment-form'))
|
36 |
+
{
|
37 |
+
$this->postreplies = $link['href'];
|
38 |
+
}
|
39 |
+
}
|
40 |
+
}
|
41 |
+
|
42 |
+
function import() {
|
43 |
+
|
44 |
+
$post_date = $this->published;
|
45 |
+
$post_content = $this->content;
|
46 |
+
$post_title = $this->title;
|
47 |
+
$post_status = $this->isDraft ? 'draft' : 'publish';
|
48 |
+
//AGC:24/10/2013 Turn off the pingbacks
|
49 |
+
$post_pingback = Blogger_Import::POST_PINGBACK;
|
50 |
+
|
51 |
+
// N.B. Clean up of $post_content is now part of the sanitize class
|
52 |
+
// Check for duplication part of calling function
|
53 |
+
$post = compact('post_date', 'post_content', 'post_title', 'post_status', 'post_pingback');
|
54 |
+
|
55 |
+
$post_id = wp_insert_post($post);
|
56 |
+
if (is_wp_error($post_id))
|
57 |
+
return $post_id;
|
58 |
+
|
59 |
+
wp_create_categories(array_map('addslashes', $this->categories), $post_id);
|
60 |
+
|
61 |
+
add_post_meta($post_id, 'blogger_blog', $this->blogurl, true);
|
62 |
+
add_post_meta($post_id, 'blogger_author', $this->author, true);
|
63 |
+
|
64 |
+
if (!$this->isDraft && isset($this->bookmark))
|
65 |
+
add_post_meta($post_id, 'blogger_permalink', $this->bookmark, true);
|
66 |
+
|
67 |
+
add_post_meta($post_id, 'blogger_internal', $this->old_permalink, true);
|
68 |
+
|
69 |
+
if (isset($this->geotags)) {
|
70 |
+
add_post_meta($post_id,'geo_latitude',$this->geotags['geo_latitude']);
|
71 |
+
add_post_meta($post_id,'geo_longitude',$this->geotags['geo_longitude']);
|
72 |
+
add_post_meta($post_id,'geo_public',1);
|
73 |
+
if (isset($this->geotags['geo_address'])) {
|
74 |
+
add_post_meta($post_id,'geo_address',$this->geotags['geo_address']);
|
75 |
+
}
|
76 |
+
}
|
77 |
+
|
78 |
+
return $post_id;
|
79 |
+
}
|
80 |
+
|
81 |
+
|
82 |
+
function post_exists() {
|
83 |
+
$p = $this->get_post_by_oldID($this->old_permalink);
|
84 |
+
|
85 |
+
if ($p == 0 && isset($this->bookmark)) {
|
86 |
+
$p = $this->get_post_by_oldID($this->bookmark);
|
87 |
+
}
|
88 |
+
return $p;
|
89 |
+
}
|
90 |
+
|
91 |
+
function get_post_by_oldID($oldID) {
|
92 |
+
//Check to see if this post has been loaded already
|
93 |
+
//Can we use get_posts for this?
|
94 |
+
global $wpdb;
|
95 |
+
$query = "SELECT post_id FROM $wpdb->postmeta m inner join $wpdb->posts p on p.ID = m.post_id where meta_key = 'blogger_internal' and meta_value = '%s' and p.post_type = 'post' LIMIT 0 , 1";
|
96 |
+
$p = (int) $wpdb->get_var( $wpdb->prepare($query, $oldID) );
|
97 |
+
return $p;
|
98 |
+
}
|
99 |
+
}
|
100 |
+
}
|
101 |
+
|
102 |
+
|
103 |
+
?>
|
blogger-importer-blog.php
ADDED
@@ -0,0 +1,812 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* @author Andy Clark
|
5 |
+
* @copyright 2013
|
6 |
+
* The overall blog, stats and methods for importing
|
7 |
+
*/
|
8 |
+
|
9 |
+
class Blogger_Importer_Blog
|
10 |
+
{
|
11 |
+
//We have 2 fields with the same name but different case, that should be eliminated, however need to ensure can be upgraded ok
|
12 |
+
var $ID; //Wordpress ID for the blog
|
13 |
+
var $id; //Blogger ID for the blog
|
14 |
+
var $comments_url;
|
15 |
+
|
16 |
+
function init_defaults($totalposts,$totalcomments)
|
17 |
+
{
|
18 |
+
$this->total_posts = $totalposts;
|
19 |
+
$this->total_comments = $totalcomments;
|
20 |
+
|
21 |
+
$this->posts_done = $this->count_imported_posts();
|
22 |
+
$this->comments_done = $this->count_imported_comments();
|
23 |
+
|
24 |
+
//Initialise other vars
|
25 |
+
$this->posts_skipped = 0;
|
26 |
+
$this->comments_skipped = 0;
|
27 |
+
|
28 |
+
$this->images_done = $this->count_imported_images();
|
29 |
+
$this->images_progress = 0;
|
30 |
+
$this->images_skipped = 0;
|
31 |
+
$this->links_done = 0; //Can't count these as we don't store them in separately in the DB
|
32 |
+
$this->links_progress = 0;
|
33 |
+
|
34 |
+
$this->mode = 'init';
|
35 |
+
}
|
36 |
+
|
37 |
+
//Return the blog from the options or false for failure
|
38 |
+
static function read_option($blogID)
|
39 |
+
{
|
40 |
+
$blog = false;
|
41 |
+
$blogopt = get_option('blogger_importer_blog_'.$blogID);
|
42 |
+
if (is_array($blogopt)) {
|
43 |
+
$blog = new Blogger_Importer_Blog();
|
44 |
+
foreach ($blogopt as $key => $value)
|
45 |
+
$blog->$key = $value;
|
46 |
+
}
|
47 |
+
return $blog;
|
48 |
+
}
|
49 |
+
|
50 |
+
function import_blog($connector)
|
51 |
+
{
|
52 |
+
//AGC: Moved the parameter parsing out to start and the importing into separate functions.
|
53 |
+
//25/1/2013 Suppress post revisions whilst we are importing
|
54 |
+
//10/3/2014 Added start action as proposed by katazina
|
55 |
+
remove_post_type_support('post', 'revisions');
|
56 |
+
|
57 |
+
if (!$this->posts_done && !$this->comments_done)
|
58 |
+
do_action('import_start', 'blogger');
|
59 |
+
|
60 |
+
$this->importer_started = time();
|
61 |
+
|
62 |
+
if (!$this->import_posts($connector))
|
63 |
+
return ('continue');
|
64 |
+
if (!$this->import_comments($connector))
|
65 |
+
return ('continue');
|
66 |
+
|
67 |
+
if (Blogger_Import::IMPORT_IMG)
|
68 |
+
{
|
69 |
+
if (!$this->process_images())
|
70 |
+
return('continue');
|
71 |
+
}
|
72 |
+
|
73 |
+
if (!$this->process_links())
|
74 |
+
return('continue');
|
75 |
+
|
76 |
+
$this->mode = 'authors';
|
77 |
+
$this->save_vars();
|
78 |
+
|
79 |
+
if (!$this->posts_done && !$this->comments_done)
|
80 |
+
return('nothing');
|
81 |
+
|
82 |
+
//Thought: Should this be passing an ID so we know which blog just got imported.
|
83 |
+
do_action('import_done', 'blogger');
|
84 |
+
|
85 |
+
return('done');
|
86 |
+
}
|
87 |
+
|
88 |
+
|
89 |
+
function import_posts($connector)
|
90 |
+
{
|
91 |
+
//Simpler counting for posts as we load them forwards
|
92 |
+
if (isset($this->posts_start_index))
|
93 |
+
$start_index = (int)$this->posts_start_index;
|
94 |
+
else
|
95 |
+
$start_index = 1;
|
96 |
+
|
97 |
+
if ($this->total_posts > $start_index)
|
98 |
+
{
|
99 |
+
// Grab all the posts
|
100 |
+
$this->mode = 'posts';
|
101 |
+
do
|
102 |
+
{
|
103 |
+
$index = $struct = $entries = array();
|
104 |
+
|
105 |
+
$url = $this->posts_url;
|
106 |
+
|
107 |
+
//Thought, how do we decouple this method from the source?
|
108 |
+
//Perhaps the "get data function" can somehow be passed through as a parameter to this.
|
109 |
+
//This decoupling would then allow an easier re-code for say when the blogger data arrives as a file not as a URL
|
110 |
+
$response = $connector->oauth_get($url, array('max-results' => Blogger_Import::MAX_RESULTS, 'start-index' => $start_index));
|
111 |
+
|
112 |
+
if ($response == false)
|
113 |
+
break;
|
114 |
+
|
115 |
+
// parse the feed
|
116 |
+
$feed = new SimplePie();
|
117 |
+
|
118 |
+
//set_xxxx methods depreciated (and not working?) replaced with get_registry as per docs
|
119 |
+
$reg = $feed->get_registry();
|
120 |
+
|
121 |
+
$reg->register('Sanitize', 'Blogger_Importer_Sanitize');
|
122 |
+
$feed->sanitize = $reg->create('Sanitize'); //Should not really need to do this but there seems to be an issue with the SimplePie class?
|
123 |
+
$reg->register('Item', 'WP_SimplePie_Blog_Item');
|
124 |
+
|
125 |
+
$feed->set_raw_data($response);
|
126 |
+
$feed->init();
|
127 |
+
|
128 |
+
foreach ($feed->get_items() as $item)
|
129 |
+
{
|
130 |
+
$blogentry = new BloggerEntry();
|
131 |
+
|
132 |
+
$blogentry->blogurl = $this->host;
|
133 |
+
$blogentry->id = $item->get_id();
|
134 |
+
$blogentry->published = $item->get_published();
|
135 |
+
$blogentry->updated = $item->get_updated();
|
136 |
+
$blogentry->isDraft = $item->get_draft_status();
|
137 |
+
$blogentry->title = $item->get_title();
|
138 |
+
$blogentry->content = $item->get_content();
|
139 |
+
$blogentry->author = $item->get_author()->get_name();
|
140 |
+
$blogentry->geotags = $item->get_geotags();
|
141 |
+
|
142 |
+
$blogentry->links = $item->get_links(array('replies', 'edit', 'self', 'alternate'));
|
143 |
+
$blogentry->parselinks();
|
144 |
+
|
145 |
+
$blogentry->categories = $item->get_categories();
|
146 |
+
|
147 |
+
// Checks for duplicates
|
148 |
+
$post_id = $blogentry->post_exists();
|
149 |
+
|
150 |
+
if ($post_id != 0)
|
151 |
+
{
|
152 |
+
$this->posts_skipped++;
|
153 |
+
} else
|
154 |
+
{
|
155 |
+
//Unique new post so import it
|
156 |
+
$post_id = $blogentry->import();
|
157 |
+
$this->posts_done++;
|
158 |
+
$this->save_vars();
|
159 |
+
}
|
160 |
+
|
161 |
+
//Ref: Not importing properly http://core.trac.wordpress.org/ticket/19096
|
162 |
+
//Simplified this section to count what is loaded rather than parsing the results again
|
163 |
+
$start_index++;
|
164 |
+
}
|
165 |
+
|
166 |
+
$this->posts_start_index = $start_index;
|
167 |
+
|
168 |
+
$this->save_vars();
|
169 |
+
|
170 |
+
} while ($this->total_posts > $start_index && $this->have_time());
|
171 |
+
}
|
172 |
+
return ($this->total_posts <= $start_index);
|
173 |
+
}
|
174 |
+
|
175 |
+
function import_comments($connector)
|
176 |
+
{
|
177 |
+
if (isset($this->comments_start_index))
|
178 |
+
$start_index = (int)$this->comments_start_index;
|
179 |
+
else
|
180 |
+
$start_index = 1;
|
181 |
+
|
182 |
+
if ($start_index > 0 && $this->total_comments > 0)
|
183 |
+
{
|
184 |
+
|
185 |
+
$this->mode = 'comments';
|
186 |
+
do
|
187 |
+
{
|
188 |
+
$index = $struct = $entries = array();
|
189 |
+
|
190 |
+
//So we can link up the comments as we go we need to load them in reverse order
|
191 |
+
//Reverse the start index as the GData Blogger feed can't be sorted
|
192 |
+
$batch = ((floor(($this->total_comments - $start_index) / Blogger_Import::MAX_RESULTS) * Blogger_Import::MAX_RESULTS) + 1);
|
193 |
+
|
194 |
+
$response = $connector->oauth_get($this->comments_url,array('max-results' => Blogger_Import::MAX_RESULTS, 'start-index' => $batch));
|
195 |
+
// parse the feed
|
196 |
+
$feed = new SimplePie();
|
197 |
+
//3/1/2012 Updated syntax for registering the item
|
198 |
+
$reg = $feed->get_registry();
|
199 |
+
$reg->register('Item', 'WP_SimplePie_Blog_Item');
|
200 |
+
// Use the standard "stricter" sanitize class for comments
|
201 |
+
$feed->set_raw_data($response);
|
202 |
+
$feed->init();
|
203 |
+
|
204 |
+
//Reverse the batch so we load the oldest comments first and hence can link up nested comments
|
205 |
+
$comments = array_reverse($feed->get_items());
|
206 |
+
|
207 |
+
if (!is_null($comments))
|
208 |
+
{
|
209 |
+
foreach ($comments as $item)
|
210 |
+
{
|
211 |
+
$commententry = new CommentEntry();
|
212 |
+
|
213 |
+
$commententry->id = $item->get_id();
|
214 |
+
$commententry->updated = $item->get_updated();
|
215 |
+
$commententry->content = $item->get_content();
|
216 |
+
$commententry->author = $item->get_author()->get_name();
|
217 |
+
$commententry->authoruri = $item->get_author()->get_link();
|
218 |
+
$commententry->authoremail = $item->get_author()->get_email();
|
219 |
+
|
220 |
+
$commententry->source = $item->get_source();
|
221 |
+
$parts = parse_url($commententry->source);
|
222 |
+
$commententry->old_post_permalink = $parts['path']; //Will be something like this '/feeds/417730729915399755/posts/default/8397846992898424746'
|
223 |
+
|
224 |
+
$commententry->post_ID = BloggerEntry::get_post_by_oldID($commententry->old_post_permalink);
|
225 |
+
|
226 |
+
//Get the links
|
227 |
+
$commententry->links = $item->get_links(array('edit', 'self', 'alternate', 'related'));
|
228 |
+
$commententry->parselinks();
|
229 |
+
|
230 |
+
// Nested comment?
|
231 |
+
if (isset($commententry->related))
|
232 |
+
{
|
233 |
+
$commententry->parentcommentid = CommentEntry::get_comment_by_oldID($commententry->related);
|
234 |
+
}
|
235 |
+
|
236 |
+
//Perhaps could log errors here?
|
237 |
+
if ($commententry->post_ID != 0)
|
238 |
+
{
|
239 |
+
// Checks for duplicates
|
240 |
+
if ($comment_id = $commententry->exists())
|
241 |
+
{
|
242 |
+
$this->comments_skipped++;
|
243 |
+
} else
|
244 |
+
{
|
245 |
+
$comment_id = $commententry->import();
|
246 |
+
$this->comments_done++;
|
247 |
+
}
|
248 |
+
} else
|
249 |
+
{
|
250 |
+
$this->comments_skipped++;
|
251 |
+
}
|
252 |
+
$this->save_vars();
|
253 |
+
$start_index++;
|
254 |
+
}
|
255 |
+
}
|
256 |
+
|
257 |
+
$this->comments_start_index = $start_index;
|
258 |
+
$this->save_vars();
|
259 |
+
} while ($this->total_comments > $start_index && $this->have_time());
|
260 |
+
}
|
261 |
+
return ($this->total_comments <= $start_index);
|
262 |
+
}
|
263 |
+
|
264 |
+
function save_authors()
|
265 |
+
{
|
266 |
+
global $wpdb;
|
267 |
+
|
268 |
+
//Change to take authors as a parameter
|
269 |
+
$authors = (array )$_POST['authors'];
|
270 |
+
|
271 |
+
//Todo: Handle the case where there are no more authors to remap
|
272 |
+
|
273 |
+
// Get an array of posts => authors
|
274 |
+
$post_ids = (array )$wpdb->get_col($wpdb->prepare("SELECT post_id FROM $wpdb->postmeta WHERE meta_key = 'blogger_blog' AND meta_value = %s", $this->host));
|
275 |
+
$post_ids = join(',', $post_ids);
|
276 |
+
$results = (array )$wpdb->get_results("SELECT post_id, meta_value FROM $wpdb->postmeta WHERE meta_key = 'blogger_author' AND post_id IN ($post_ids)");
|
277 |
+
foreach ($results as $row)
|
278 |
+
$authors_posts[$row->post_id] = $row->meta_value;
|
279 |
+
|
280 |
+
foreach ($authors as $author => $user_id)
|
281 |
+
{
|
282 |
+
$user_id = (int)$user_id;
|
283 |
+
|
284 |
+
// Skip authors that haven't been changed
|
285 |
+
if ($user_id == $this->authors[$author][1])
|
286 |
+
continue;
|
287 |
+
|
288 |
+
// Get a list of the selected author's posts
|
289 |
+
$post_ids = (array )array_keys($authors_posts, $this->authors[$author][0]);
|
290 |
+
$post_ids = join(',', $post_ids);
|
291 |
+
|
292 |
+
$wpdb->query($wpdb->prepare("UPDATE $wpdb->posts SET post_author = %d WHERE id IN ($post_ids)", $user_id));
|
293 |
+
$this->authors[$author][1] = $user_id;
|
294 |
+
|
295 |
+
//Should we be tidying up the post meta data here, rather than in restart?
|
296 |
+
//$wpdb->query("DELETE FROM $wpdb->postmeta WHERE meta_key = 'blogger_author'");
|
297 |
+
}
|
298 |
+
return '';
|
299 |
+
}
|
300 |
+
|
301 |
+
|
302 |
+
/**
|
303 |
+
* Return count of imported posts from WordPress database
|
304 |
+
*
|
305 |
+
* @return int
|
306 |
+
*/
|
307 |
+
function count_imported_posts()
|
308 |
+
{
|
309 |
+
global $wpdb;
|
310 |
+
$count = 0;
|
311 |
+
$meta_key = 'blogger_blog';
|
312 |
+
$sql = $wpdb->prepare("SELECT count(post_id) as cnt FROM $wpdb->postmeta m
|
313 |
+
inner join $wpdb->posts p on p.id = m.post_id and post_type = 'post'
|
314 |
+
where meta_key = '%s' and meta_value = '%s'", $meta_key, $this->host);
|
315 |
+
|
316 |
+
$result = $wpdb->get_results($sql);
|
317 |
+
|
318 |
+
if (!empty($result))
|
319 |
+
$count = intval($result[0]->cnt);
|
320 |
+
|
321 |
+
// unset to save memory
|
322 |
+
unset($results);
|
323 |
+
|
324 |
+
return $count;
|
325 |
+
}
|
326 |
+
|
327 |
+
/**
|
328 |
+
* Return count of imported images from WordPress database
|
329 |
+
*
|
330 |
+
* @param string $bloghost url
|
331 |
+
* @return int
|
332 |
+
*/
|
333 |
+
function count_imported_images()
|
334 |
+
{
|
335 |
+
global $wpdb;
|
336 |
+
$count = 0;
|
337 |
+
$meta_key = 'blogger_blog';
|
338 |
+
$sql = $wpdb->prepare("SELECT count(post_id) as cnt FROM $wpdb->postmeta m
|
339 |
+
inner join $wpdb->posts p on p.id = m.post_id and post_type = 'attachment'
|
340 |
+
where meta_key = '%s' and meta_value = '%s'", $meta_key, $this->host);
|
341 |
+
|
342 |
+
$result = $wpdb->get_results($sql);
|
343 |
+
|
344 |
+
if (!empty($result))
|
345 |
+
$count = intval($result[0]->cnt);
|
346 |
+
|
347 |
+
// unset to save memory
|
348 |
+
unset($results);
|
349 |
+
|
350 |
+
return $count;
|
351 |
+
}
|
352 |
+
|
353 |
+
/**
|
354 |
+
* Return count of imported comments from WordPress database
|
355 |
+
*
|
356 |
+
* @param string $bloghost url
|
357 |
+
* @return int
|
358 |
+
*/
|
359 |
+
function count_imported_comments()
|
360 |
+
{
|
361 |
+
global $wpdb;
|
362 |
+
$count = 0;
|
363 |
+
$meta_key = 'blogger_blog';
|
364 |
+
$sql = $wpdb->prepare("SELECT count(post_id) as cnt FROM $wpdb->postmeta m
|
365 |
+
inner join $wpdb->posts p on p.id = m.post_id and post_type = 'post'
|
366 |
+
inner join $wpdb->comments c on p.id = c.comment_post_ID
|
367 |
+
where meta_key = '%s' and meta_value = '%s'
|
368 |
+
and c.comment_type <> 'pingback'", $meta_key, $this->host);
|
369 |
+
|
370 |
+
$result = $wpdb->get_results($sql);
|
371 |
+
|
372 |
+
if (!empty($result))
|
373 |
+
$count = intval($result[0]->cnt);
|
374 |
+
|
375 |
+
// unset to save memory
|
376 |
+
unset($results);
|
377 |
+
|
378 |
+
return $count;
|
379 |
+
}
|
380 |
+
|
381 |
+
function process_links()
|
382 |
+
{
|
383 |
+
//Update all of the links in the blog
|
384 |
+
global $wpdb;
|
385 |
+
|
386 |
+
$postsprocessed = $this->links_progress;
|
387 |
+
if ($postsprocessed == 0)
|
388 |
+
{
|
389 |
+
$linksprocessed = 0;
|
390 |
+
} else
|
391 |
+
{
|
392 |
+
$linksprocessed = $this->links_done;
|
393 |
+
}
|
394 |
+
$batchsize = 20;
|
395 |
+
|
396 |
+
$oldurlsearch = $this->host;
|
397 |
+
|
398 |
+
if (substr($oldurlsearch, 0, 3) == 'www.')
|
399 |
+
{
|
400 |
+
$oldurlsearch = substr($oldurlsearch, 3, strlen($oldurlsearch - 3));
|
401 |
+
}
|
402 |
+
|
403 |
+
$oldurlsearch = str_replace('.', '\.', $oldurlsearch);
|
404 |
+
|
405 |
+
$blogspot = stripos($oldurlsearch, '\.blogspot\.');
|
406 |
+
if ($blogspot)
|
407 |
+
{ //Blogspot addresses can be international e.g. myblog.blogspot.com, myblog.blogspot.com.au or myblog.blogspot.co.uk or myblog.blogspot.de both resolve to the same blog.
|
408 |
+
//See http://www.searchenginejournal.com/google-blogger-url-censorship/39724/
|
409 |
+
$oldurlsearch = substr($oldurlsearch, 0, $blogspot + 12) . '[\w\.]{2,6}';
|
410 |
+
}
|
411 |
+
|
412 |
+
$loadedposts = get_posts(array('meta_key' => 'blogger_blog', 'meta_value' => $this->host, 'posts_per_page' => $batchsize, 'offset' => $postsprocessed, 'post_status' => array('draft', 'publish','future')));
|
413 |
+
|
414 |
+
//Stop if nothing left
|
415 |
+
if (count($loadedposts) == 0){
|
416 |
+
return true;
|
417 |
+
}
|
418 |
+
|
419 |
+
foreach ($loadedposts as $importedpost) {
|
420 |
+
$importedcontent = $importedpost->post_content;
|
421 |
+
|
422 |
+
$regexp = '<a\s[^>]*href=([\"\'`])(https?:\/\/(?:www\.)*' . $oldurlsearch . '\/)([^\" >]*?)\1[^>]*>(.*)<\/a>';
|
423 |
+
if (preg_match_all("/$regexp/siU", $importedcontent, $matches, PREG_SET_ORDER))
|
424 |
+
{
|
425 |
+
foreach ($matches as $match)
|
426 |
+
{
|
427 |
+
$HostURL = substr($match[2], 0, strlen($match[2]) - 1); //e.g. http://minitemp.blogspot.co.uk
|
428 |
+
$PageURL = '/' . $match[3]; //e.g. '/2011/04/what-happens-if-blog-title-is-really.html'
|
429 |
+
$sql = $wpdb->prepare("SELECT post_id FROM $wpdb->postmeta m
|
430 |
+
inner join $wpdb->posts p on p.id = m.post_id and post_type = 'post'
|
431 |
+
where meta_key = '%s' and meta_value = '%s'", 'blogger_permalink', $PageURL);
|
432 |
+
|
433 |
+
$linkpostid = $wpdb->get_var($sql);
|
434 |
+
|
435 |
+
if ($linkpostid != 0) {
|
436 |
+
$NewURL = get_permalink($linkpostid);
|
437 |
+
} else { // Page not found, update content with just the new domain
|
438 |
+
$NewURL = site_url($PageURL);
|
439 |
+
}
|
440 |
+
|
441 |
+
$importedcontent = str_replace($HostURL . $PageURL, $NewURL, $importedcontent);
|
442 |
+
$linksprocessed++;
|
443 |
+
}
|
444 |
+
|
445 |
+
if ($importedcontent == '')
|
446 |
+
{
|
447 |
+
return new WP_Error('Empty Content', __("Attempting to write back empty content"));
|
448 |
+
}
|
449 |
+
|
450 |
+
$importedpost->post_content = $importedcontent;
|
451 |
+
wp_update_post($importedpost);
|
452 |
+
|
453 |
+
}
|
454 |
+
$postsprocessed++;
|
455 |
+
|
456 |
+
//For some reason the intermediate values are not getting written, is it that the options are cached hence not read back?
|
457 |
+
$this->links_done = $linksprocessed;
|
458 |
+
$this->links_progress = $postsprocessed;
|
459 |
+
$saved = $this->save_vars();
|
460 |
+
}
|
461 |
+
unset($loadedposts);
|
462 |
+
return ($postsprocessed >= $this->total_posts);;
|
463 |
+
}
|
464 |
+
|
465 |
+
function isurlimage($srcurl)
|
466 |
+
{
|
467 |
+
//Process picasaweb links and files that are images
|
468 |
+
if (substr($srcurl, 0, 27) == 'http://picasaweb.google.com')
|
469 |
+
return true;
|
470 |
+
return preg_match('/(?i)\.(jpe?g|png|gif|bmp)$/i', $srcurl);
|
471 |
+
}
|
472 |
+
|
473 |
+
//Get the current status for this blog
|
474 |
+
function get_js_status()
|
475 |
+
{
|
476 |
+
//Trap for division by zero errors
|
477 |
+
if ($this->total_posts == 0) {
|
478 |
+
$p1 = 100;
|
479 |
+
$i2 = 100;
|
480 |
+
$l2 = 100;
|
481 |
+
} else {
|
482 |
+
$p1 = ($this->posts_done / $this->total_posts) * 100;
|
483 |
+
$i2 = ($this->images_progress / $this->total_posts) * 100;
|
484 |
+
$l2 = ($this->links_progress / $this->total_posts) * 100;
|
485 |
+
}
|
486 |
+
|
487 |
+
$p2 = sprintf(__('%d of %d', 'blogger-importer'), $this->posts_done, $this->total_posts);
|
488 |
+
$p3 = sprintf(_n('%d post skipped', '%d posts skipped', $this->posts_skipped, 'blogger-importer'), $this->posts_skipped);
|
489 |
+
$c1 = $this->total_comments == 0 ? 100 : ($this->comments_done / $this->total_comments) * 100;
|
490 |
+
$c2 = sprintf(__('%d of %d', 'blogger-importer'), $this->comments_done, $this->total_comments);
|
491 |
+
$c3 = sprintf(_n('%d comment skipped', '%d comments skipped', $this->comments_skipped, 'blogger-importer'), $this->comments_skipped);
|
492 |
+
$i1 = isset($this->images_done) ? (int)$this->images_done : 0;
|
493 |
+
$i3 = sprintf(_n('%d image skipped', '%d images skipped', $this->images_skipped, 'blogger-importer'), $this->images_skipped);
|
494 |
+
|
495 |
+
$l1 = isset($this->links_done) ? (int)$this->links_done : 0;
|
496 |
+
|
497 |
+
return "{p1:$p1,p2:\"$p2\",p3:\"$p3\",c1:$c1,c2:\"$c2\",c3:\"$c3\",i1:$i1,i2:$i2,i3:\"$i3\",l1:$l1,l2:$l2}";
|
498 |
+
}
|
499 |
+
|
500 |
+
function get_authors()
|
501 |
+
{
|
502 |
+
//Get a list of blogger authors who created the posts
|
503 |
+
//Use a subquery rather than 2 statements to avoid issue with prepare not being able to sanitise an array of integers, also protects us if the post_id type changes
|
504 |
+
global $wpdb;
|
505 |
+
global $current_user;
|
506 |
+
|
507 |
+
$authors = (array )$wpdb->get_col($wpdb->prepare("SELECT DISTINCT meta_value FROM $wpdb->postmeta WHERE meta_key = 'blogger_author' AND post_id IN (SELECT post_id FROM $wpdb->postmeta WHERE meta_key = 'blogger_blog' AND meta_value = %s)",$this->host));
|
508 |
+
|
509 |
+
$this->authors = array_map(null, $authors, array_fill(0, count($authors), $current_user->ID));
|
510 |
+
$this->save_vars();
|
511 |
+
}
|
512 |
+
|
513 |
+
/*
|
514 |
+
* Search for either a linked image or a non linked image within the supplied html
|
515 |
+
* <a href="xxx" yyyy><img src="zzz" ></a> or <img src="zzz" >
|
516 |
+
* Ref: http://www.the-art-of-web.com/php/parse-links/
|
517 |
+
* "<a\s[^>]*href=(\"??)([^\" >]*?)\\1[^>]*>(.*)<\/a>"
|
518 |
+
* http://wordpress.org/extend/plugins/blogger-image-import/
|
519 |
+
* "<a[^>]+href\=([\"'`])(.*)\\1[^<]*?<img[^>]*src\=([\"'`])(.*)\\3[^>]*>"
|
520 |
+
*/
|
521 |
+
function get_images($content)
|
522 |
+
{
|
523 |
+
$highrez = array();
|
524 |
+
$lowrez = array();
|
525 |
+
|
526 |
+
//First images with links
|
527 |
+
//Might be nice to expand this top try and get Alt and/or Title attributes for use as description
|
528 |
+
$regexp = "<a\s[^>]*href\=([\"'`])([^> ]*?)\\1[^<]*?<img[^>]*src\=([\"'`])([^\> ]*?)\\3[^>]*>";
|
529 |
+
|
530 |
+
if (preg_match_all("/$regexp/siU", $content, $matches1, PREG_SET_ORDER))
|
531 |
+
{
|
532 |
+
//http://www.techrepublic.com/article/17-useful-functions-for-manipulating-arrays-in-php/5792851
|
533 |
+
foreach ($matches1 as $match)
|
534 |
+
{
|
535 |
+
if ($this->isurlimage($match[2]))
|
536 |
+
{
|
537 |
+
$highrez[$match[4]] = $match[2];
|
538 |
+
} else
|
539 |
+
{
|
540 |
+
$lowrez[$match[4]] = '';
|
541 |
+
}
|
542 |
+
}
|
543 |
+
}
|
544 |
+
|
545 |
+
//Now any image (n.b. this overlaps the previous set)
|
546 |
+
$regexp = "<img[^>]*src\=([\"'`])([^\> ]*?)\\1[^>]*>";
|
547 |
+
|
548 |
+
if (preg_match_all("/$regexp/siU", $content, $matches2, PREG_SET_ORDER))
|
549 |
+
{
|
550 |
+
foreach ($matches2 as $match)
|
551 |
+
{
|
552 |
+
$lowrez[$match[2]] = '';
|
553 |
+
}
|
554 |
+
}
|
555 |
+
|
556 |
+
//Remove any rows from this second set that are already in the first set and merge two sets of results
|
557 |
+
$images = array_merge($lowrez, $highrez);
|
558 |
+
return $images;
|
559 |
+
}
|
560 |
+
|
561 |
+
/**
|
562 |
+
* Update all of the images in the posts that have already been imported
|
563 |
+
*/
|
564 |
+
function process_images()
|
565 |
+
{
|
566 |
+
$postsprocessed = $this->images_progress;
|
567 |
+
if ($postsprocessed == 0)
|
568 |
+
{
|
569 |
+
$imagesprocessed = 0;
|
570 |
+
$imagesskipped = 0;
|
571 |
+
} else
|
572 |
+
{
|
573 |
+
$imagesprocessed = $this->images_done;
|
574 |
+
$imagesskipped = $this->images_skipped;
|
575 |
+
}
|
576 |
+
|
577 |
+
$batchsize = 20;
|
578 |
+
|
579 |
+
$loadedposts = get_posts(array('meta_key' => 'blogger_blog', 'meta_value' => $this->host, 'posts_per_page' => $batchsize, 'offset' => $postsprocessed, 'post_status' => array('draft', 'publish','future')));
|
580 |
+
|
581 |
+
//Stop if nothing left
|
582 |
+
if (count($loadedposts) == 0){
|
583 |
+
return true;
|
584 |
+
}
|
585 |
+
|
586 |
+
foreach ($loadedposts as $importedpost)
|
587 |
+
{
|
588 |
+
|
589 |
+
$importedcontent = $importedpost->post_content;
|
590 |
+
$author = get_post_meta($importedpost->ID, 'blogger_author', true);
|
591 |
+
|
592 |
+
$img_count = 0; //Count of images for this post
|
593 |
+
foreach ($this->get_images($importedcontent) as $lowrez => $highrez)
|
594 |
+
{
|
595 |
+
if (!$this->image_filter($lowrez))
|
596 |
+
{
|
597 |
+
//Pass null for description so that the default (filename) is used, might be good to use Alt tag instead?
|
598 |
+
$newcontent = $this->import_image($importedpost->ID, $lowrez, $highrez, null, $img_count, $importedcontent, $this->host, $author);
|
599 |
+
if (!is_wp_error($newcontent)) {
|
600 |
+
$importedcontent = $newcontent;
|
601 |
+
$img_count++;
|
602 |
+
} else {
|
603 |
+
Blogger_Import::_log($newcontent);
|
604 |
+
$imagesskipped++;
|
605 |
+
}
|
606 |
+
}
|
607 |
+
}
|
608 |
+
$imagesprocessed += $img_count;
|
609 |
+
|
610 |
+
$importedpost->post_content = $importedcontent;
|
611 |
+
wp_update_post($importedpost);
|
612 |
+
$postsprocessed++;
|
613 |
+
|
614 |
+
//For some reason the intermediate values are not getting written
|
615 |
+
$this->images_done = $imagesprocessed;
|
616 |
+
$this->images_progress = $postsprocessed;
|
617 |
+
$this->images_skipped = $imagesskipped;
|
618 |
+
$saved = $this->save_vars();
|
619 |
+
}
|
620 |
+
unset($loadedposts);
|
621 |
+
return ($postsprocessed >= $this->total_posts);
|
622 |
+
}
|
623 |
+
|
624 |
+
function image_urlremap($url,$large)
|
625 |
+
{ /* Fix problem urls
|
626 |
+
e.g. change https://lh4.googleusercontent.com/-nt66qhxzDyY/TZOD-RhTYMI/AAAAAAAACd4/Elzm1smRFb4/s800-h/Ski%2520Trip.jpg to
|
627 |
+
to https://lh4.googleusercontent.com/-nt66qhxzDyY/TZOD-RhTYMI/AAAAAAAACd4/Elzm1smRFb4/s800/Ski%2520Trip.jpg
|
628 |
+
Could use a apply_filter here to allow users to add their own tweeks
|
629 |
+
*/
|
630 |
+
$pattern = '/(\/)(s\d*)-h(\/)/i';
|
631 |
+
$replacement = '$1$2$3';
|
632 |
+
$img = preg_replace($pattern, $replacement, $url);
|
633 |
+
|
634 |
+
/* Strip out ? and # on the end of files */
|
635 |
+
$pattern = '/(.*)[#\?].*/i';
|
636 |
+
$replacement = '$1';
|
637 |
+
$img = preg_replace($pattern, $replacement, $img);
|
638 |
+
|
639 |
+
if ($large) {
|
640 |
+
// For images on blogger we can swap /sXXX/ with for example /s1600/ to get a larger file.
|
641 |
+
// Use a standardised large size so we can control quality vs filesize.
|
642 |
+
$pattern = '/(\/)(s\d*)(\/)/i';
|
643 |
+
$replacement = '$1s'.Blogger_Import::LARGE_IMAGE_SIZE.'$3';
|
644 |
+
$img = preg_replace($pattern, $replacement, $img);
|
645 |
+
}
|
646 |
+
return $img;
|
647 |
+
}
|
648 |
+
|
649 |
+
function image_filter($url)
|
650 |
+
{
|
651 |
+
// Do we exclude this particular image?
|
652 |
+
// Don't include images that are already loaded onto this site
|
653 |
+
// Could use a apply_filter here to allow users to add their own tweeks
|
654 |
+
return (substr($url, 0, strlen(site_url())) == site_url());
|
655 |
+
}
|
656 |
+
|
657 |
+
function import_image($post_id, $lowrez, $highrez, $description, $imgcount, $postcontent, $blogname, $author)
|
658 |
+
{
|
659 |
+
/* Import a new image unless we specifically filter it out or if it has already been downloaded on another page.
|
660 |
+
Based on http://wordpress.stackexchange.com/questions//media-sideload-image-file-name and the tumblr-importer
|
661 |
+
Simple filename cleaning as characters such as +, % cause issues ref: http://wordpress.org/extend/plugins/uploadplus/
|
662 |
+
|
663 |
+
It's processing links of a form similar to these as provided by the "get_images" function
|
664 |
+
<a href="myhighrezimage.jpg"><img src="mylowrezimage.jpg"></a>
|
665 |
+
or
|
666 |
+
<img src="mylowrezimage.jpg">
|
667 |
+
|
668 |
+
If the high resolution (linked) file is not an image then the low resolution version is downloaded.
|
669 |
+
*/
|
670 |
+
$lowrez_old = $lowrez;
|
671 |
+
$highrez_old = $highrez;
|
672 |
+
$highrezispage = false;
|
673 |
+
|
674 |
+
$lowrez = $this->image_urlremap($lowrez,false);
|
675 |
+
if ($lowrez == '')
|
676 |
+
return new WP_Error('Not an image',$message = __('Lowrez not an image','blogger-importer'), $data = array($lowrez_old, $highrez_old));
|
677 |
+
|
678 |
+
if ($highrez != '')
|
679 |
+
{
|
680 |
+
$highrez = $this->image_urlremap($highrez,true);
|
681 |
+
} else
|
682 |
+
{
|
683 |
+
$highrez = $this->image_urlremap($lowrez,true);
|
684 |
+
}
|
685 |
+
|
686 |
+
if (!$att_id = $this->image_exists($lowrez))
|
687 |
+
{
|
688 |
+
//Option to add a timeout to download_url, but don't use the wp_remote_get timeout as that's much shorter than the default here of 300s
|
689 |
+
$tmp = download_url($highrez);
|
690 |
+
|
691 |
+
if (is_wp_error($tmp))
|
692 |
+
{
|
693 |
+
@unlink($tmp); // clean up, copied this from other examples but how is this supposed to work if $tmp is an error??
|
694 |
+
//Don't exit as can still try the small image
|
695 |
+
}
|
696 |
+
|
697 |
+
// If the highrez was not an image then try the lowrex
|
698 |
+
if (!$this->is_image($tmp, $highrez))
|
699 |
+
{
|
700 |
+
$highrezispage = true; //That image was not valid
|
701 |
+
$tmp = download_url($lowrez); // Option to add a timeout here
|
702 |
+
|
703 |
+
if (is_wp_error($tmp))
|
704 |
+
{
|
705 |
+
@unlink($tmp); // clean up
|
706 |
+
return $tmp; // output wp_error
|
707 |
+
}
|
708 |
+
|
709 |
+
if (!$this->is_image($tmp, $lowrez))
|
710 |
+
{
|
711 |
+
@unlink($tmp); // clean up None of items are actually images, for example might be a single pixel, deliberately filtered out or a 404 error?
|
712 |
+
return new WP_Error('No Images',__('None of the images are valid','blogger-importer'), $data = array($lowrez_old, $highrez_old));
|
713 |
+
}
|
714 |
+
}
|
715 |
+
|
716 |
+
$new_name = preg_replace('/[^A-Za-z0-9._ ]/i', '-', basename($lowrez));
|
717 |
+
|
718 |
+
$file_array = array('name' => $new_name, 'tmp_name' => $tmp);
|
719 |
+
|
720 |
+
$att_id = media_handle_sideload($file_array, $post_id, $description, array('post_excerpt' => $description));
|
721 |
+
if (is_wp_error($att_id))
|
722 |
+
{
|
723 |
+
@unlink($file_array['tmp_name']);
|
724 |
+
return $att_id;
|
725 |
+
}
|
726 |
+
|
727 |
+
// Link attachment upto old url, store the author so we can replace it later
|
728 |
+
add_post_meta($att_id, 'blogger_permalink', $lowrez);
|
729 |
+
add_post_meta($att_id, 'blogger_blog', $blogname, true);
|
730 |
+
add_post_meta($att_id, 'blogger_author', $author, true);
|
731 |
+
|
732 |
+
if ($highrezispage) //Not an image so store so we can link later
|
733 |
+
add_post_meta($att_id, 'blogger_largeimgispage', true);
|
734 |
+
|
735 |
+
} else
|
736 |
+
{
|
737 |
+
//Image already exists, check if the high rez one was valid last time
|
738 |
+
$tmp = get_post_meta($att_id, 'blogger_largeimgispage', true);
|
739 |
+
if ($tmp == true)
|
740 |
+
$highrezispage = true;
|
741 |
+
}
|
742 |
+
|
743 |
+
//Always treat picassa webs as image so they get replaced with the new High rez link
|
744 |
+
if (substr($highrez, 0, 27) == 'http://picasaweb.google.com')
|
745 |
+
$highrezispage = false;
|
746 |
+
|
747 |
+
//Replace the image strings
|
748 |
+
if (!$highrezispage && $highrez_old != '')
|
749 |
+
{
|
750 |
+
$highrez_new = reset(wp_get_attachment_image_src($att_id, 'full'));
|
751 |
+
$postcontent = str_replace($highrez_old, $highrez_new, $postcontent);
|
752 |
+
}
|
753 |
+
$lowrez_new = reset(wp_get_attachment_image_src($att_id, 'medium'));
|
754 |
+
$postcontent = str_replace($lowrez_old, $lowrez_new, $postcontent);
|
755 |
+
|
756 |
+
//Set the first image to be the post thumbnail (zero index)
|
757 |
+
if ($imgcount == 0)
|
758 |
+
{
|
759 |
+
set_post_thumbnail($post_id, $att_id);
|
760 |
+
}
|
761 |
+
|
762 |
+
//media handle sideload moves the file so there should be no temp file left but cleanup just incase.
|
763 |
+
@unlink($tmp);
|
764 |
+
|
765 |
+
// incase something goes wrong
|
766 |
+
if ($postcontent == '')
|
767 |
+
{
|
768 |
+
return new WP_Error('Empty Content', __("Attempting to write back empty content",'blogger-importer'), $data = array($lowrez_old, $highrez_old));
|
769 |
+
}
|
770 |
+
return $postcontent;
|
771 |
+
}
|
772 |
+
|
773 |
+
function is_image($file, $filename)
|
774 |
+
{
|
775 |
+
//Is the downloaded file really an image
|
776 |
+
//e.g. it looked like an image from the URL but when downloaded it was something else perhaps a html page
|
777 |
+
//Also filter out tracking images of 1 pixel square
|
778 |
+
//Found that wp_check_filetype_and_ext and wp_match_mime_types was giving false positives
|
779 |
+
$imgstats = @getimagesize($file);
|
780 |
+
if (!$imgstats)
|
781 |
+
{
|
782 |
+
return false;
|
783 |
+
}
|
784 |
+
|
785 |
+
return (($imgstats[0] > 1) && ($imgstats[1] > 1));
|
786 |
+
}
|
787 |
+
|
788 |
+
function image_exists($lowrez)
|
789 |
+
{
|
790 |
+
global $wpdb;
|
791 |
+
return $wpdb->get_var($wpdb->prepare("SELECT ID FROM $wpdb->posts p INNER JOIN $wpdb->postmeta m ON p.ID = m.post_id AND meta_key = 'blogger_permalink' WHERE post_type = 'attachment' AND meta_value = %s LIMIT 0 , 1",
|
792 |
+
$lowrez));
|
793 |
+
}
|
794 |
+
|
795 |
+
function have_time()
|
796 |
+
{
|
797 |
+
return (time() - $this->importer_started < Blogger_Import::MAX_EXECUTION_TIME);
|
798 |
+
}
|
799 |
+
|
800 |
+
function save_vars()
|
801 |
+
{
|
802 |
+
$vars = get_object_vars($this);
|
803 |
+
//Store each blog it it's own setting so that we retrieve and save less data each time it updates the stats
|
804 |
+
//11/3/2014 Simplified function to avoid logging unneccesary errors
|
805 |
+
update_option('blogger_importer_blog_'.$this->ID, $vars);
|
806 |
+
return !empty($vars);
|
807 |
+
}
|
808 |
+
|
809 |
+
|
810 |
+
}
|
811 |
+
|
812 |
+
?>
|
blogger-importer-blogitem.php
CHANGED
@@ -1,123 +1,168 @@
|
|
1 |
-
<?php
|
2 |
-
|
3 |
-
/**
|
4 |
-
* Based on WP_SimplePieAtomPub_Item
|
5 |
-
* Expect this to become part of core wordpress at some point.
|
6 |
-
* See http://core.trac.wordpress.org/ticket/7652
|
7 |
-
*
|
8 |
-
*
|
9 |
-
*
|
10 |
-
|
11 |
-
|
12 |
-
|
13 |
-
define('
|
14 |
-
define('
|
15 |
-
|
16 |
-
/**
|
17 |
-
* SimplePie Helper for AtomPub
|
18 |
-
*
|
19 |
-
* @package WordPress
|
20 |
-
* @subpackage Publishing
|
21 |
-
* @since 3.1
|
22 |
-
*/
|
23 |
-
if (!class_exists('WP_SimplePie_Blog_Item'))
|
24 |
-
{
|
25 |
-
class WP_SimplePie_Blog_Item extends SimplePie_Item
|
26 |
-
{
|
27 |
-
/**
|
28 |
-
* Constructor
|
29 |
-
*/
|
30 |
-
function
|
31 |
-
{
|
32 |
-
parent::
|
33 |
-
}
|
34 |
-
|
35 |
-
/**
|
36 |
-
* Get the status of the entry
|
37 |
-
*
|
38 |
-
* @return bool True if the item is a draft, false otherwise
|
39 |
-
*/
|
40 |
-
function get_draft_status()
|
41 |
-
{
|
42 |
-
$draft = false;
|
43 |
-
if (($control = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOMPUB, 'control')) && !empty($control[0]['child'][SIMPLEPIE_NAMESPACE_ATOMPUB]['draft'][0]['data']))
|
44 |
-
{
|
45 |
-
$draft = ('yes' == $control[0]['child'][SIMPLEPIE_NAMESPACE_ATOMPUB]['draft'][0]['data']);
|
46 |
-
}
|
47 |
-
return $draft;
|
48 |
-
}
|
49 |
-
|
50 |
-
//Tried using date functions from http://core.trac.wordpress.org/attachment/ticket/7652/7652-separate.diff
|
51 |
-
//but ended up with 1970s dates so returned to Otto's version which is much simplified
|
52 |
-
function get_updated()
|
53 |
-
{
|
54 |
-
$temparray = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'updated');
|
55 |
-
if (isset($temparray[0]['data']))
|
56 |
-
return $this->convert_date($temparray[0]['data']);
|
57 |
-
else
|
58 |
-
return null;
|
59 |
-
}
|
60 |
-
|
61 |
-
function get_published()
|
62 |
-
{
|
63 |
-
$temparray = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'published');
|
64 |
-
if (isset($temparray[0]['data']))
|
65 |
-
return $this->convert_date($temparray[0]['data']);
|
66 |
-
else
|
67 |
-
return null;
|
68 |
-
}
|
69 |
-
|
70 |
-
function get_geotags()
|
71 |
-
{//Return an array of geo tags see http://codex.wordpress.org/Geodata
|
72 |
-
//example source
|
73 |
-
// <georss:featurename>R�dhuspladsen 3, 1550 Copenhagen, Denmark</georss:featurename>
|
74 |
-
// <georss:point>55.6760968 12.5683371</georss:point>
|
75 |
-
|
76 |
-
|
77 |
-
$latlong = $this->get_item_tags(SIMPLEPIE_NAMESPACE_GEOTAG, 'point');
|
78 |
-
|
79 |
-
if (isset($latlong[0]['data'])) {
|
80 |
-
preg_match('/([0-9.-]+).+?([0-9.-]+)/', $latlong[0]['data'], $matches);
|
81 |
-
$lat=(float)$matches[1];
|
82 |
-
$long=(float)$matches[2];
|
83 |
-
}
|
84 |
-
|
85 |
-
if (!isset($lat) |!isset($long)) {
|
86 |
-
return null; //Without lat long we can't have a valid location
|
87 |
-
}
|
88 |
-
|
89 |
-
$address = $this->get_item_tags(SIMPLEPIE_NAMESPACE_GEOTAG, 'featurename');
|
90 |
-
if (isset($address[0]['data']))
|
91 |
-
$geo_address = $address[0]['data'];
|
92 |
-
else
|
93 |
-
$geo_address = null;
|
94 |
-
|
95 |
-
$geo = array('geo_latitude' => $lat, 'geo_longitude' => $long, 'geo_address' => $geo_address );
|
96 |
-
|
97 |
-
return $geo;
|
98 |
-
}
|
99 |
-
|
100 |
-
function convert_date($date)
|
101 |
-
{
|
102 |
-
preg_match('#([0-9]{4})-([0-9]{2})-([0-9]{2})T([0-9]{2}):([0-9]{2}):([0-9]{2})(?:\.[0-9]+)?(Z|[\+|\-][0-9]{2,4}){0,1}#', $date, $date_bits);
|
103 |
-
$offset = iso8601_timezone_to_offset($date_bits[7]);
|
104 |
-
$timestamp = gmmktime($date_bits[4], $date_bits[5], $date_bits[6], $date_bits[2], $date_bits[3], $date_bits[1]);
|
105 |
-
$timestamp -= $offset; // Convert from Blogger local time to GMT
|
106 |
-
$timestamp += get_option('gmt_offset') * 3600; // Convert from GMT to WP local time
|
107 |
-
return gmdate('Y-m-d H:i:s', $timestamp);
|
108 |
-
}
|
109 |
-
|
110 |
-
//Don't Sanitize the ID, the default get_id was cleaning our IDs and that meant that nested comments did not work
|
111 |
-
function get_id()
|
112 |
-
{
|
113 |
-
if ($return = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'id'))
|
114 |
-
{
|
115 |
-
return $return[0]['data'];
|
116 |
-
}
|
117 |
-
}
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
|
122 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
123 |
?>
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* Based on WP_SimplePieAtomPub_Item
|
5 |
+
* Expect this to become part of core wordpress at some point.
|
6 |
+
* See http://core.trac.wordpress.org/ticket/7652
|
7 |
+
*
|
8 |
+
* http://codex.wordpress.org/Geodata
|
9 |
+
*
|
10 |
+
*/
|
11 |
+
|
12 |
+
define('SIMPLEPIE_NAMESPACE_ATOMPUB', 'http://purl.org/atom/app#');
|
13 |
+
define('SIMPLEPIE_NAMESPACE_GEOTAG', 'http://www.georss.org/georss');
|
14 |
+
define('SIMPLEPIE_NAMESPACE_THREAD','http://purl.org/syndication/thread/1.0');
|
15 |
+
|
16 |
+
/**
|
17 |
+
* SimplePie Helper for AtomPub
|
18 |
+
*
|
19 |
+
* @package WordPress
|
20 |
+
* @subpackage Publishing
|
21 |
+
* @since 3.1
|
22 |
+
*/
|
23 |
+
if (!class_exists('WP_SimplePie_Blog_Item'))
|
24 |
+
{
|
25 |
+
class WP_SimplePie_Blog_Item extends SimplePie_Item
|
26 |
+
{
|
27 |
+
/**
|
28 |
+
* Constructor
|
29 |
+
*/
|
30 |
+
function __construct($feed, $data)
|
31 |
+
{
|
32 |
+
parent::__construct($feed, $data);
|
33 |
+
}
|
34 |
+
|
35 |
+
/**
|
36 |
+
* Get the status of the entry
|
37 |
+
*
|
38 |
+
* @return bool True if the item is a draft, false otherwise
|
39 |
+
*/
|
40 |
+
function get_draft_status()
|
41 |
+
{
|
42 |
+
$draft = false;
|
43 |
+
if (($control = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOMPUB, 'control')) && !empty($control[0]['child'][SIMPLEPIE_NAMESPACE_ATOMPUB]['draft'][0]['data']))
|
44 |
+
{
|
45 |
+
$draft = ('yes' == $control[0]['child'][SIMPLEPIE_NAMESPACE_ATOMPUB]['draft'][0]['data']);
|
46 |
+
}
|
47 |
+
return $draft;
|
48 |
+
}
|
49 |
+
|
50 |
+
//Tried using date functions from http://core.trac.wordpress.org/attachment/ticket/7652/7652-separate.diff
|
51 |
+
//but ended up with 1970s dates so returned to Otto's version which is much simplified
|
52 |
+
function get_updated()
|
53 |
+
{
|
54 |
+
$temparray = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'updated');
|
55 |
+
if (isset($temparray[0]['data']))
|
56 |
+
return $this->convert_date($temparray[0]['data']);
|
57 |
+
else
|
58 |
+
return null;
|
59 |
+
}
|
60 |
+
|
61 |
+
function get_published()
|
62 |
+
{
|
63 |
+
$temparray = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'published');
|
64 |
+
if (isset($temparray[0]['data']))
|
65 |
+
return $this->convert_date($temparray[0]['data']);
|
66 |
+
else
|
67 |
+
return null;
|
68 |
+
}
|
69 |
+
|
70 |
+
function get_geotags()
|
71 |
+
{//Return an array of geo tags see http://codex.wordpress.org/Geodata
|
72 |
+
//example source
|
73 |
+
// <georss:featurename>R�dhuspladsen 3, 1550 Copenhagen, Denmark</georss:featurename>
|
74 |
+
// <georss:point>55.6760968 12.5683371</georss:point>
|
75 |
+
|
76 |
+
|
77 |
+
$latlong = $this->get_item_tags(SIMPLEPIE_NAMESPACE_GEOTAG, 'point');
|
78 |
+
|
79 |
+
if (isset($latlong[0]['data'])) {
|
80 |
+
preg_match('/([0-9.-]+).+?([0-9.-]+)/', $latlong[0]['data'], $matches);
|
81 |
+
$lat=(float)$matches[1];
|
82 |
+
$long=(float)$matches[2];
|
83 |
+
}
|
84 |
+
|
85 |
+
if (!isset($lat) |!isset($long)) {
|
86 |
+
return null; //Without lat long we can't have a valid location
|
87 |
+
}
|
88 |
+
|
89 |
+
$address = $this->get_item_tags(SIMPLEPIE_NAMESPACE_GEOTAG, 'featurename');
|
90 |
+
if (isset($address[0]['data']))
|
91 |
+
$geo_address = $address[0]['data'];
|
92 |
+
else
|
93 |
+
$geo_address = null;
|
94 |
+
|
95 |
+
$geo = array('geo_latitude' => $lat, 'geo_longitude' => $long, 'geo_address' => $geo_address );
|
96 |
+
|
97 |
+
return $geo;
|
98 |
+
}
|
99 |
+
|
100 |
+
function convert_date($date)
|
101 |
+
{
|
102 |
+
preg_match('#([0-9]{4})-([0-9]{2})-([0-9]{2})T([0-9]{2}):([0-9]{2}):([0-9]{2})(?:\.[0-9]+)?(Z|[\+|\-][0-9]{2,4}){0,1}#', $date, $date_bits);
|
103 |
+
$offset = iso8601_timezone_to_offset($date_bits[7]);
|
104 |
+
$timestamp = gmmktime($date_bits[4], $date_bits[5], $date_bits[6], $date_bits[2], $date_bits[3], $date_bits[1]);
|
105 |
+
$timestamp -= $offset; // Convert from Blogger local time to GMT
|
106 |
+
$timestamp += get_option('gmt_offset') * 3600; // Convert from GMT to WP local time
|
107 |
+
return gmdate('Y-m-d H:i:s', $timestamp);
|
108 |
+
}
|
109 |
+
|
110 |
+
//Don't Sanitize the ID, the default get_id was cleaning our IDs and that meant that nested comments did not work
|
111 |
+
function get_id($hash = false)
|
112 |
+
{
|
113 |
+
if ($return = $this->get_item_tags(SIMPLEPIE_NAMESPACE_ATOM_10, 'id'))
|
114 |
+
{
|
115 |
+
return $return[0]['data'];
|
116 |
+
}
|
117 |
+
}
|
118 |
+
|
119 |
+
//Prefiltered links
|
120 |
+
function get_links($rel = 'alternate') {
|
121 |
+
|
122 |
+
$mylinks = array();
|
123 |
+
foreach ($rel as $type)
|
124 |
+
{
|
125 |
+
$links =parent::get_links($type);
|
126 |
+
|
127 |
+
if (!is_null($links)) {
|
128 |
+
foreach ($links as $link) {
|
129 |
+
$mylinks[] = array('rel' => $type, 'href' => $link);
|
130 |
+
}
|
131 |
+
}
|
132 |
+
}
|
133 |
+
return $mylinks;
|
134 |
+
}
|
135 |
+
|
136 |
+
//Preprocessed categories
|
137 |
+
function get_categories() {
|
138 |
+
$cats = parent::get_categories();
|
139 |
+
$mycats = array();
|
140 |
+
|
141 |
+
if (!is_null($cats)) {
|
142 |
+
foreach ($cats as $cat) {
|
143 |
+
$mycats[] = $cat->term;
|
144 |
+
}
|
145 |
+
}
|
146 |
+
return $mycats;
|
147 |
+
}
|
148 |
+
|
149 |
+
//What is the source of this item e.g. a comment linked to a post
|
150 |
+
//10/3/2014 Added error handling for where the comment links to a post that no longer exists on blogger.
|
151 |
+
function get_source() {
|
152 |
+
$temp = $this->get_item_tags(SIMPLEPIE_NAMESPACE_THREAD, 'in-reply-to');
|
153 |
+
|
154 |
+
if (!is_null($temp)){
|
155 |
+
foreach ($temp as $t) {
|
156 |
+
if (isset($t['attribs']['']['source'])) {
|
157 |
+
$source = $t['attribs']['']['source'];
|
158 |
+
}
|
159 |
+
}
|
160 |
+
return $source;
|
161 |
+
}
|
162 |
+
}
|
163 |
+
|
164 |
+
}
|
165 |
+
|
166 |
+
}
|
167 |
+
|
168 |
?>
|
blogger-importer-connector.php
ADDED
@@ -0,0 +1,195 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* @author Andy Clark
|
5 |
+
* @copyright 2013
|
6 |
+
* Connector for Google Blogger
|
7 |
+
*
|
8 |
+
*/
|
9 |
+
|
10 |
+
class Blogger_Importer_Connector
|
11 |
+
{
|
12 |
+
var $token;
|
13 |
+
var $secret;
|
14 |
+
var $endpoint;
|
15 |
+
var $oauth_token;
|
16 |
+
var $callback_url;
|
17 |
+
|
18 |
+
/*
|
19 |
+
Make the initial connection to blogger, authorise tokens etc.
|
20 |
+
*/
|
21 |
+
function connect($url)
|
22 |
+
{
|
23 |
+
// Establish an Blogger_OAuth consumer
|
24 |
+
$base_url = admin_url();
|
25 |
+
$request_token_endpoint = 'https://www.google.com/accounts/OAuthGetRequestToken';
|
26 |
+
$authorize_endpoint = 'https://www.google.com/accounts/OAuthAuthorizeToken';
|
27 |
+
|
28 |
+
$test_consumer = new Blogger_OAuthConsumer('anonymous', 'anonymous', null); // anonymous is a google thing to allow non-registered apps to work
|
29 |
+
|
30 |
+
if (!Blogger_OAuthUtil::sslinstalled())
|
31 |
+
{
|
32 |
+
return new WP_Error('OpenSSL', __('OpenSSL is not installed check your PHP.INI or ask your server provider to enable the OpenSSL module.','blogger-importer'));
|
33 |
+
}
|
34 |
+
|
35 |
+
//prepare to get request token
|
36 |
+
//https://developers.google.com/accounts/docs/OAuth_ref#RequestToken
|
37 |
+
$sig_method = new Blogger_OAuthSignatureMethod_HMAC_SHA1();
|
38 |
+
$parsed = parse_url($request_token_endpoint);
|
39 |
+
$params = array('callback' => $base_url, 'scope' => 'http://www.blogger.com/feeds/', 'xoauth_displayname' => 'WordPress');
|
40 |
+
|
41 |
+
$req_req = Blogger_OAuthRequest::from_consumer_and_token($test_consumer, null, "GET", $request_token_endpoint, $params);
|
42 |
+
$req_req->sign_request($sig_method, $test_consumer, null);
|
43 |
+
|
44 |
+
// go get the request tokens from Google
|
45 |
+
$req_response = wp_remote_get($req_req->to_url(), array('sslverify' => false, 'timeout' => Blogger_Import::REMOTE_TIMEOUT));
|
46 |
+
if (is_wp_error($req_response))
|
47 |
+
{
|
48 |
+
return $req_response;
|
49 |
+
}
|
50 |
+
$req_token = wp_remote_retrieve_body($req_response);
|
51 |
+
if (is_wp_error($req_token))
|
52 |
+
{
|
53 |
+
return $req_token;
|
54 |
+
}
|
55 |
+
|
56 |
+
// parse the tokens
|
57 |
+
parse_str($req_token, $tokens);
|
58 |
+
|
59 |
+
$oauth_token = $tokens['oauth_token'];
|
60 |
+
$oauth_token_secret = $tokens['oauth_token_secret'];
|
61 |
+
|
62 |
+
//AGC:10/10/2012 Added error handling here based on http://wordpress.org/support/topic/plugin-blogger-importer-invalid-token/page/2?replies=31#post-3243710
|
63 |
+
if ($oauth_token == '')
|
64 |
+
{
|
65 |
+
return new WP_Error('Invalid Token', __('Invalid Token: Check server date, firewall settings and transports (curl, streams and fsockopen)','blogger-importer'));
|
66 |
+
} else
|
67 |
+
{
|
68 |
+
$this->callback_url = add_query_arg(array( 'token' => $oauth_token, 'secret' => $oauth_token_secret ),$url);
|
69 |
+
$this->endpoint = $authorize_endpoint;
|
70 |
+
$this->oauth_token = $oauth_token;
|
71 |
+
return true;
|
72 |
+
}
|
73 |
+
}
|
74 |
+
|
75 |
+
/*
|
76 |
+
*
|
77 |
+
*/
|
78 |
+
function auth_form(){
|
79 |
+
//$next_url = admin_url('index.php?import=blogger&noheader=true');
|
80 |
+
$auth = esc_attr__('Authorize', 'blogger-importer');
|
81 |
+
|
82 |
+
return(sprintf ("<form action='%s' method='get'>
|
83 |
+
<p class='submit' style='text-align:left;'>
|
84 |
+
<input type='submit' class='button' value='%s' />
|
85 |
+
<input type='hidden' name='oauth_token' value='%s' />
|
86 |
+
<input type='hidden' name='oauth_callback' value='%s' />
|
87 |
+
</p>
|
88 |
+
</form>
|
89 |
+
</div>\n",$this->endpoint,$auth,$this->oauth_token,$this->callback_url));
|
90 |
+
}
|
91 |
+
|
92 |
+
/**
|
93 |
+
* We have a authorized request token now, so upgrade it to an access token
|
94 |
+
*
|
95 |
+
* @link https://developers.google.com/gdata/articles/oauth#UsingAccessToken
|
96 |
+
*/
|
97 |
+
function auth($token,$secret)
|
98 |
+
{
|
99 |
+
$oauth_access_token_endpoint = 'https://www.google.com/accounts/OAuthGetAccessToken';
|
100 |
+
|
101 |
+
// auth the token
|
102 |
+
$test_consumer = new Blogger_OAuthConsumer('anonymous', 'anonymous', null);
|
103 |
+
$auth_token = new Blogger_OAuthConsumer($token, $secret);
|
104 |
+
$access_token_req = new Blogger_OAuthRequest("GET", $oauth_access_token_endpoint);
|
105 |
+
$access_token_req = $access_token_req->from_consumer_and_token($test_consumer, $auth_token, "GET", $oauth_access_token_endpoint);
|
106 |
+
|
107 |
+
$access_token_req->sign_request(new Blogger_OAuthSignatureMethod_HMAC_SHA1(), $test_consumer, $auth_token);
|
108 |
+
|
109 |
+
// Should be doing validation here, what if wp_remote calls failed?
|
110 |
+
$after_access_request = wp_remote_retrieve_body(wp_remote_get($access_token_req->to_url(), array('sslverify' => false, 'timeout' => Blogger_Import::REMOTE_TIMEOUT)));
|
111 |
+
|
112 |
+
parse_str($after_access_request, $access_tokens);
|
113 |
+
|
114 |
+
$this->token = $access_tokens['oauth_token'];
|
115 |
+
$this->secret = $access_tokens['oauth_token_secret'];
|
116 |
+
|
117 |
+
$this->save_vars();
|
118 |
+
}
|
119 |
+
|
120 |
+
|
121 |
+
/**
|
122 |
+
* Get a URL using the oauth token for authentication (returns false on failure)
|
123 |
+
*/
|
124 |
+
function oauth_get($url,$params = null)
|
125 |
+
{
|
126 |
+
$test_consumer = new Blogger_OAuthConsumer('anonymous', 'anonymous', null);
|
127 |
+
$goog = new Blogger_OAuthConsumer($this->token, $this->secret, null);
|
128 |
+
$request = new Blogger_OAuthRequest("GET", $url, $params);
|
129 |
+
|
130 |
+
//Ref: Not importing properly http://core.trac.wordpress.org/ticket/19096
|
131 |
+
$blog_req = $request->from_consumer_and_token($test_consumer, $goog, 'GET', $url, $params);
|
132 |
+
|
133 |
+
$blog_req->sign_request(new Blogger_OAuthSignatureMethod_HMAC_SHA1(), $test_consumer, $goog);
|
134 |
+
|
135 |
+
$data = wp_remote_get($blog_req->to_url(), array('sslverify' => false, 'timeout' => Blogger_Import::REMOTE_TIMEOUT));
|
136 |
+
|
137 |
+
if (wp_remote_retrieve_response_code($data) == 200)
|
138 |
+
{
|
139 |
+
$response = wp_remote_retrieve_body($data);
|
140 |
+
} else
|
141 |
+
{
|
142 |
+
$response = false;
|
143 |
+
}
|
144 |
+
|
145 |
+
return $response;
|
146 |
+
}
|
147 |
+
|
148 |
+
|
149 |
+
//Need to check, why start index 2? Surely 1 would be better
|
150 |
+
function get_total_results($url)
|
151 |
+
{
|
152 |
+
$response = $this->oauth_get($url,array('max-results' => 1, 'start-index' => 2));
|
153 |
+
|
154 |
+
$feed = new SimplePie();
|
155 |
+
$feed->set_raw_data($response);
|
156 |
+
$feed->init();
|
157 |
+
$results = $feed->get_channel_tags('http://a9.com/-/spec/opensearchrss/1.0/', 'totalResults');
|
158 |
+
|
159 |
+
$total_results = $results[0]['data'];
|
160 |
+
unset($feed);
|
161 |
+
return (int)$total_results;
|
162 |
+
}
|
163 |
+
|
164 |
+
//Have we authorised the connection
|
165 |
+
function isconnected()
|
166 |
+
{
|
167 |
+
return (isset($this->secret));
|
168 |
+
}
|
169 |
+
|
170 |
+
function reset()
|
171 |
+
{
|
172 |
+
unset($this->secret);
|
173 |
+
unset($this->token);
|
174 |
+
delete_option('blogger_importer_connector');
|
175 |
+
}
|
176 |
+
|
177 |
+
function save_vars()
|
178 |
+
{
|
179 |
+
if (!update_option('blogger_importer_connector', $this)) {
|
180 |
+
Blogger_Import::_log('Error saving connection');
|
181 |
+
Blogger_Import::_log(var_export(get_object_vars($this),true));
|
182 |
+
};
|
183 |
+
return !empty($vars);
|
184 |
+
}
|
185 |
+
|
186 |
+
//Return the connection
|
187 |
+
static function read_option()
|
188 |
+
{
|
189 |
+
$connection = get_option('blogger_importer_connector',new Blogger_Importer_Connector());
|
190 |
+
return $connection;
|
191 |
+
}
|
192 |
+
|
193 |
+
}
|
194 |
+
|
195 |
+
?>
|
blogger-importer-sanitize.php
CHANGED
@@ -1,152 +1,134 @@
|
|
1 |
-
<?php
|
2 |
-
|
3 |
-
/**
|
4 |
-
* New class to sanitize trusted content from blogger import
|
5 |
-
* Based on the SimplePie_Sanitize class by Ryan Parman and Geoffrey Sneddon
|
6 |
-
*
|
7 |
-
*/
|
8 |
-
|
9 |
-
class Blogger_Importer_Sanitize extends
|
10 |
-
{
|
11 |
-
// Private vars
|
12 |
-
var $base;
|
13 |
-
|
14 |
-
// Options
|
15 |
-
var $image_handler = '';
|
16 |
-
var $strip_htmltags = array('base', 'blink', 'body', 'doctype', 'font', 'form', 'frame', 'frameset', 'html', 'input', 'marquee', 'meta', 'script', 'style');
|
17 |
-
//Allow iframe (new style) and embed, param and object(old style) so that we get youtube videos transferred
|
18 |
-
//Allow object and noscript for Amazon widgets
|
19 |
-
var $encode_instead_of_strip = false;
|
20 |
-
var $strip_attributes = array('bgsound', 'class', 'expr', 'id', 'imageanchor', 'onclick', 'onerror', 'onfinish', 'onmouseover', 'onmouseout', 'onfocus', 'onblur', 'lowsrc', 'dynsrc');
|
21 |
-
//Allow styles so we don't have to redo in
|
22 |
-
//Brett Morgan from Google has confirmed that imageanchor is a made up attribute that is just used in the blogger editor so we can remove that
|
23 |
-
|
24 |
-
var $
|
25 |
-
var $
|
26 |
-
var $
|
27 |
-
var $
|
28 |
-
var $
|
29 |
-
var $
|
30 |
-
var $
|
31 |
-
var $
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
|
45 |
-
|
46 |
-
|
47 |
-
|
48 |
-
|
49 |
-
|
50 |
-
$data =
|
51 |
-
|
52 |
-
//
|
53 |
-
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
60 |
-
|
61 |
-
|
62 |
-
|
63 |
-
|
64 |
-
|
65 |
-
|
66 |
-
|
67 |
-
|
68 |
-
|
69 |
-
|
70 |
-
|
71 |
-
|
72 |
-
|
73 |
-
|
74 |
-
|
75 |
-
|
76 |
-
|
77 |
-
|
78 |
-
|
79 |
-
|
80 |
-
|
81 |
-
|
82 |
-
|
83 |
-
|
84 |
-
|
85 |
-
|
86 |
-
|
87 |
-
|
88 |
-
|
89 |
-
|
90 |
-
|
91 |
-
|
92 |
-
|
93 |
-
|
94 |
-
|
95 |
-
|
96 |
-
|
97 |
-
|
98 |
-
|
99 |
-
|
100 |
-
|
101 |
-
|
102 |
-
|
103 |
-
|
104 |
-
|
105 |
-
|
106 |
-
|
107 |
-
|
108 |
-
|
109 |
-
|
110 |
-
|
111 |
-
|
112 |
-
|
113 |
-
|
114 |
-
|
115 |
-
|
116 |
-
|
117 |
-
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
|
122 |
-
|
123 |
-
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
$data = str_replace($element['full'], $new_element, $data);
|
136 |
-
$element['full'] = $new_element;
|
137 |
-
}
|
138 |
-
}
|
139 |
-
} elseif (isset($element['attribs'][$attributes]['data']))
|
140 |
-
{
|
141 |
-
$element['attribs'][$attributes]['data'] = SimplePie_Misc::absolutize_url($element['attribs'][$attributes]['data'], $this->base);
|
142 |
-
$data = str_replace($element['full'], SimplePie_Misc::element_implode($element), $data);
|
143 |
-
}
|
144 |
-
}
|
145 |
-
}
|
146 |
-
return $data;
|
147 |
-
}
|
148 |
-
|
149 |
-
|
150 |
-
}
|
151 |
-
|
152 |
-
?>
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* New class to sanitize trusted content from blogger import
|
5 |
+
* Based on the SimplePie_Sanitize class by Ryan Parman and Geoffrey Sneddon
|
6 |
+
*
|
7 |
+
*/
|
8 |
+
|
9 |
+
class Blogger_Importer_Sanitize extends SimplePie_Sanitize
|
10 |
+
{
|
11 |
+
// Private vars
|
12 |
+
var $base;
|
13 |
+
|
14 |
+
// Options
|
15 |
+
var $image_handler = '';
|
16 |
+
var $strip_htmltags = array('base', 'blink', 'body', 'doctype', 'font', 'form', 'frame', 'frameset', 'html', 'input', 'marquee', 'meta', 'script', 'style');
|
17 |
+
//Allow iframe (new style) and embed, param and object(old style) so that we get youtube videos transferred
|
18 |
+
//Allow object and noscript for Amazon widgets
|
19 |
+
var $encode_instead_of_strip = false;
|
20 |
+
var $strip_attributes = array('bgsound', 'class', 'expr', 'id', 'imageanchor', 'onclick', 'onerror', 'onfinish', 'onmouseover', 'onmouseout', 'onfocus', 'onblur', 'lowsrc', 'dynsrc', 'trbidi');
|
21 |
+
//Allow styles so we don't have to redo in WordPress
|
22 |
+
//Brett Morgan from Google has confirmed that imageanchor is a made up attribute that is just used in the blogger editor so we can remove that
|
23 |
+
//trbidi another blogger proprietory attribute
|
24 |
+
var $output_encoding = 'UTF-8';
|
25 |
+
var $enable_cache = true;
|
26 |
+
var $cache_location = './cache';
|
27 |
+
var $cache_name_function = 'md5';
|
28 |
+
var $cache_class = 'SimplePie_Cache';
|
29 |
+
var $file_class = 'SimplePie_File';
|
30 |
+
var $timeout = 10;
|
31 |
+
var $useragent = '';
|
32 |
+
var $force_fsockopen = false;
|
33 |
+
|
34 |
+
var $replace_url_attributes = array('a' => 'href', 'area' => 'href', 'blockquote' => 'cite', 'del' => 'cite', 'form' => 'action', 'img' => array('longdesc', 'src'), 'input' => 'src', 'ins' => 'cite',
|
35 |
+
'q' => 'cite');
|
36 |
+
|
37 |
+
public function __construct()
|
38 |
+
{
|
39 |
+
parent::__construct();
|
40 |
+
}
|
41 |
+
|
42 |
+
function _normalize_tag($matches)
|
43 |
+
{
|
44 |
+
return '<' . strtolower($matches[1]);
|
45 |
+
}
|
46 |
+
|
47 |
+
function sanitize($data, $type, $base = '')
|
48 |
+
{
|
49 |
+
//Simplified function
|
50 |
+
$data = trim($data);
|
51 |
+
|
52 |
+
// Normalise tags (string replacement is case sensitive)
|
53 |
+
$data = preg_replace_callback('|<(/?[A-Z]+)|', array(&$this, '_normalize_tag'), $data);
|
54 |
+
|
55 |
+
// Remappings
|
56 |
+
$data = str_replace('<br>', '<br />', $data);
|
57 |
+
$data = str_replace('<hr>', '<hr />', $data);
|
58 |
+
//<span style="font-weight:bold;">Workshopshed:</span> > <b>Workshopshed:</b>
|
59 |
+
$data = preg_replace('|(<span style="font-weight:bold;">)(?<!<span style="font-weight:bold;">).*(.*)(</span>)|', '<strong>$2</strong>', $data);
|
60 |
+
|
61 |
+
//N.B. Don't strip comments as blogger uses <!--more--> which is the same as WordPress
|
62 |
+
// Comments might also contain section targetting e.g. <!-- google_ad_section_start -->
|
63 |
+
|
64 |
+
//Now clean up
|
65 |
+
foreach ($this->strip_htmltags as $tag)
|
66 |
+
{
|
67 |
+
$pcre = "/<($tag)" . SIMPLEPIE_PCRE_HTML_ATTRIBUTE . "(>(.*)<\/$tag" . SIMPLEPIE_PCRE_HTML_ATTRIBUTE . '>|(\/)?>)/siU';
|
68 |
+
while (preg_match($pcre, $data))
|
69 |
+
{
|
70 |
+
$data = preg_replace_callback($pcre, array(&$this, 'do_strip_htmltags'), $data);
|
71 |
+
}
|
72 |
+
}
|
73 |
+
|
74 |
+
foreach ($this->strip_attributes as $attrib)
|
75 |
+
{
|
76 |
+
$data = preg_replace('/(<[A-Za-z][^\x09\x0A\x0B\x0C\x0D\x20\x2F\x3E]*)' . SIMPLEPIE_PCRE_HTML_ATTRIBUTE . trim($attrib) . '(?:\s*=\s*(?:"(?:[^"]*)"|\'(?:[^\']*)\'|(?:[^\x09\x0A\x0B\x0C\x0D\x20\x22\x27\x3E][^\x09\x0A\x0B\x0C\x0D\x20\x3E]*)?))?' .
|
77 |
+
SIMPLEPIE_PCRE_HTML_ATTRIBUTE . '>/', '\1\2\3>', $data);
|
78 |
+
}
|
79 |
+
|
80 |
+
// Replace relative URLs
|
81 |
+
$this->base = $base;
|
82 |
+
|
83 |
+
foreach ($this->replace_url_attributes as $element => $attributes)
|
84 |
+
{
|
85 |
+
$data = $this->replace_urls($data, $element, $attributes);
|
86 |
+
}
|
87 |
+
|
88 |
+
// Images are handled as a separate step as we need to download them
|
89 |
+
|
90 |
+
// Having (possibly) taken stuff out, there may now be whitespace at the beginning/end of the data
|
91 |
+
$data = trim($data);
|
92 |
+
|
93 |
+
return $data;
|
94 |
+
}
|
95 |
+
|
96 |
+
function replace_urls($data, $tag, $attributes)
|
97 |
+
{
|
98 |
+
//This seems to do nothing at all!?
|
99 |
+
if (!is_array($this->strip_htmltags) || !in_array($tag, $this->strip_htmltags))
|
100 |
+
{
|
101 |
+
$elements = SimplePie_Misc::get_element($tag, $data);
|
102 |
+
foreach ($elements as $element)
|
103 |
+
{
|
104 |
+
if (is_array($attributes))
|
105 |
+
{
|
106 |
+
foreach ($attributes as $attribute)
|
107 |
+
{
|
108 |
+
if (isset($element['attribs'][$attribute]['data']))
|
109 |
+
{
|
110 |
+
$element['attribs'][$attribute]['data'] = SimplePie_Misc::absolutize_url($element['attribs'][$attribute]['data'], $this->base);
|
111 |
+
$new_element = SimplePie_Misc::element_implode($element);
|
112 |
+
$data = str_replace($element['full'], $new_element, $data);
|
113 |
+
$element['full'] = $new_element;
|
114 |
+
}
|
115 |
+
}
|
116 |
+
} elseif (isset($element['attribs'][$attributes]['data']))
|
117 |
+
{
|
118 |
+
$element['attribs'][$attributes]['data'] = SimplePie_Misc::absolutize_url($element['attribs'][$attributes]['data'], $this->base);
|
119 |
+
$data = str_replace($element['full'], SimplePie_Misc::element_implode($element), $data);
|
120 |
+
}
|
121 |
+
}
|
122 |
+
}
|
123 |
+
return $data;
|
124 |
+
}
|
125 |
+
|
126 |
+
//Latest SimplePie checks for this function
|
127 |
+
public function set_registry(SimplePie_Registry $registry)
|
128 |
+
{
|
129 |
+
parent::set_registry($registry);
|
130 |
+
}
|
131 |
+
|
132 |
+
}
|
133 |
+
|
134 |
+
?>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
blogger-importer-table.php
ADDED
@@ -0,0 +1,118 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* @author Andy Clark
|
5 |
+
* @copyright 2013
|
6 |
+
* Ref:http://wpengineer.com/2426/wp_list_table-a-step-by-step-guide/
|
7 |
+
* Class to render the blog table
|
8 |
+
*/
|
9 |
+
|
10 |
+
if( ! class_exists( 'WP_List_Table' ) ) {
|
11 |
+
require_once( ABSPATH . 'wp-admin/includes/class-wp-list-table.php' );
|
12 |
+
}
|
13 |
+
|
14 |
+
class Blogger_Import_List_Table extends WP_List_Table {
|
15 |
+
|
16 |
+
//Use unique names here so that WP does not style our comments as pers some other tables.
|
17 |
+
function get_columns(){
|
18 |
+
$columns = array(
|
19 |
+
'BL_name' => __('Blog Name','blogger-importer'),
|
20 |
+
'BL_url' => __('Blog URL','blogger-importer'),
|
21 |
+
'BL_posts' => __('Posts','blogger-importer'),
|
22 |
+
'BL_comments' => __('Comments','blogger-importer'),
|
23 |
+
'BL_images' => __('Images','blogger-importer'),
|
24 |
+
'BL_links' => __('Links','blogger-importer'),
|
25 |
+
'BL_action' => __('Action Button','blogger-importer')
|
26 |
+
);
|
27 |
+
|
28 |
+
return $columns;
|
29 |
+
}
|
30 |
+
|
31 |
+
function prepare_items($blogs,$JSInit) {
|
32 |
+
|
33 |
+
$columns = $this->get_columns();
|
34 |
+
$hidden = array();
|
35 |
+
$sortable = array();
|
36 |
+
$this->_column_headers = array($columns, $hidden, $sortable);
|
37 |
+
$this->JSInit = $JSInit;
|
38 |
+
|
39 |
+
foreach ($blogs as $i => $blog) {
|
40 |
+
$row = array();
|
41 |
+
$row['ID'] = $i;
|
42 |
+
$row['BL_name'] = esc_js($blog->title);
|
43 |
+
$row['BL_url'] = $blog->host;
|
44 |
+
$row['BL_posts'] = array('posts_done'=> ($blog->posts_done) ? (int)$blog->posts_done : 0,'total_posts' => $blog->total_posts);
|
45 |
+
$row['BL_comments'] = array('comments_done'=> ($blog->comments_done) ? (int)$blog->comments_done : 0,'total_comments' => $blog->total_comments);
|
46 |
+
$row['BL_images'] = $blog->images_done ? (int)$blog->images_done : 0;
|
47 |
+
$row['BL_links'] = $blog->links_done ? (int)$blog->links_done : 0;
|
48 |
+
$row['BL_action'] = $blog->mode;
|
49 |
+
$this->items[] = $row;
|
50 |
+
}
|
51 |
+
}
|
52 |
+
|
53 |
+
//Column Rendering
|
54 |
+
function column_BL_posts( $item ) {
|
55 |
+
return sprintf('<div ID="Posts%s"><div ID="PostsLabel%s" class="ui-progressbar-caption">'.__('%d of %d', 'blogger-importer').'</div></div>',$item['ID'],$item['ID'],$item['BL_posts']['posts_done'],$item['BL_posts']['total_posts']);
|
56 |
+
}
|
57 |
+
|
58 |
+
function column_BL_comments( $item ) {
|
59 |
+
return sprintf('<div ID="Comments%s"><div ID="CommentsLabel%s" class="ui-progressbar-caption">'.__('%d of %d', 'blogger-importer').'</div></div>',$item['ID'],$item['ID'],$item['BL_comments']['comments_done'],$item['BL_comments']['total_comments']);
|
60 |
+
}
|
61 |
+
|
62 |
+
function column_BL_images( $item ) {
|
63 |
+
return sprintf('<div ID="Images%s"><div ID="ImagesLabel%s" class="ui-progressbar-caption">%d</div></div>',$item['ID'],$item['ID'],$item['BL_images']);
|
64 |
+
}
|
65 |
+
|
66 |
+
function column_BL_links( $item ) {
|
67 |
+
return sprintf('<div ID="Links%s"><div ID="LinksLabel%s" class="ui-progressbar-caption">%d</div></div>',$item['ID'],$item['ID'],$item['BL_links']);
|
68 |
+
}
|
69 |
+
|
70 |
+
function column_BL_action( $item ) {
|
71 |
+
$start = esc_js(__('Import', 'blogger-importer'));
|
72 |
+
$continue = esc_js(__('Continue', 'blogger-importer'));
|
73 |
+
$stop = esc_js(__('Importing...', 'blogger-importer'));
|
74 |
+
$authors = esc_js(__('Set Authors', 'blogger-importer'));
|
75 |
+
|
76 |
+
if ($item['BL_action'] == 'init')
|
77 |
+
$value = $start;
|
78 |
+
elseif ($item['BL_action'] == 'posts' || $item['BL_action'] == 'comments')
|
79 |
+
$value = $continue;
|
80 |
+
else
|
81 |
+
$value = $authors;
|
82 |
+
$value = esc_attr($value);
|
83 |
+
|
84 |
+
return sprintf("<input type='submit' class='button' id='submit%s' value='%s' /><input type='hidden' name='blog' value='%s' />",$item['ID'],$value,$item['ID']);
|
85 |
+
}
|
86 |
+
|
87 |
+
function column_default( $item, $column_name ) {
|
88 |
+
switch( $column_name ) {
|
89 |
+
case 'BL_name':
|
90 |
+
case 'BL_url':
|
91 |
+
case 'BL_comments':
|
92 |
+
case 'BL_images':
|
93 |
+
case 'BL_links':
|
94 |
+
case 'BL_action':
|
95 |
+
return $item[ $column_name ];
|
96 |
+
default:
|
97 |
+
return print_r( $item, true ) ; //Show the whole array for troubleshooting purposes
|
98 |
+
}
|
99 |
+
}
|
100 |
+
|
101 |
+
//Run Javascript at the bottom of the page to pass table data through to JQuery
|
102 |
+
function _js_vars() {
|
103 |
+
printf( "<!--blogger-importer-table _js_vars -->
|
104 |
+
<script type='text/javascript'>
|
105 |
+
/* <![CDATA[ */
|
106 |
+
var blogs = {};
|
107 |
+
jQuery(function() {
|
108 |
+
%s
|
109 |
+
jQuery.each(blogs, function(i, me){me.init();});
|
110 |
+
jQuery('#Refresh').bind('click', function(){return jQuery.each(blogs, function(i, me){me.stop();})});
|
111 |
+
});
|
112 |
+
/* ]]> */
|
113 |
+
</script>\n",$this->JSInit );
|
114 |
+
}
|
115 |
+
|
116 |
+
|
117 |
+
}
|
118 |
+
?>
|
blogger-importer.css
CHANGED
@@ -1,29 +1,62 @@
|
|
1 |
-
|
2 |
-
td { text-align: center; line-height: 2em;}
|
3 |
-
thead td { font-weight: bold; }
|
4 |
-
.bar {
|
5 |
-
width: 200px;
|
6 |
-
text-align: left;
|
7 |
-
line-height: 2em;
|
8 |
-
padding: 0px;
|
9 |
-
}
|
10 |
-
.ind {
|
11 |
-
position: absolute;
|
12 |
-
background-color: #83B4D8;
|
13 |
-
width: 1px;
|
14 |
-
z-index: 9;
|
15 |
-
}
|
16 |
-
|
17 |
-
|
18 |
-
|
19 |
-
|
20 |
-
|
21 |
-
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
|
26 |
-
|
27 |
-
|
28 |
-
padding-
|
29 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
|
2 |
+
td { text-align: center; line-height: 2em;}
|
3 |
+
thead td { font-weight: bold; }
|
4 |
+
.bar {
|
5 |
+
width: 200px;
|
6 |
+
text-align: left;
|
7 |
+
line-height: 2em;
|
8 |
+
padding: 0px;
|
9 |
+
}
|
10 |
+
.ind {
|
11 |
+
position: absolute;
|
12 |
+
background-color: #83B4D8;
|
13 |
+
width: 1px;
|
14 |
+
z-index: 9;
|
15 |
+
}
|
16 |
+
/* Todo: Review some of these since moving to WP_Tables */
|
17 |
+
.stat {
|
18 |
+
z-index: 10;
|
19 |
+
position: relative;
|
20 |
+
text-align: center;
|
21 |
+
}
|
22 |
+
td.submit {
|
23 |
+
margin:0;
|
24 |
+
padding:0;
|
25 |
+
}
|
26 |
+
|
27 |
+
td {
|
28 |
+
padding-left:10px;
|
29 |
+
padding-right:10px;
|
30 |
+
}
|
31 |
+
|
32 |
+
/* Fix for MP6 plugin aka future styling for Admin UI */
|
33 |
+
table.wp-list-table td { line-height: 2em;}
|
34 |
+
|
35 |
+
.column-BL_name { text-align: left;}
|
36 |
+
.column-BL_url { text-align: left;}
|
37 |
+
.column-BL_posts { text-align: left; width:12%; }
|
38 |
+
.column-BL_comments { text-align: left; width:12%;}
|
39 |
+
.column-BL_images { text-align: left; width:12%;}
|
40 |
+
.column-BL_links { text-align: left; width:12%;}
|
41 |
+
.column-BL_action { text-align: left; width:8%;}
|
42 |
+
|
43 |
+
td.column-BL_posts div { width: 100%; }
|
44 |
+
|
45 |
+
/* JQuery Progress Bars */
|
46 |
+
.ui-progressbar-value {background-color: #83B4D8;
|
47 |
+
position:relative;
|
48 |
+
height:2em;
|
49 |
+
margin-top:-2em;
|
50 |
+
z-index: 9;
|
51 |
+
display: block;}
|
52 |
+
.ui-progressbar {border: thin solid;
|
53 |
+
height:2em;
|
54 |
+
}
|
55 |
+
|
56 |
+
.ui-progressbar-caption {
|
57 |
+
z-index:10;
|
58 |
+
display: block;
|
59 |
+
position: relative;
|
60 |
+
text-align: center;
|
61 |
+
|
62 |
+
}
|
blogger-importer.js
ADDED
@@ -0,0 +1,146 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
/* Javascript for refreshing the blog table. */
|
2 |
+
/* Localised strings passed to this via wp_localize_script and BL_strings object */
|
3 |
+
/* Stop on error or stop if Ajax Admin can't find a suitable handler */
|
4 |
+
function blog(i, title, mode, status){
|
5 |
+
this.blog = i;
|
6 |
+
this.mode = mode;
|
7 |
+
this.title = title;
|
8 |
+
eval('this.status='+status);
|
9 |
+
this.button = '#submit'+this.blog;
|
10 |
+
};
|
11 |
+
blog.prototype = {
|
12 |
+
start: function() {
|
13 |
+
this.cont = true;
|
14 |
+
this.kick();
|
15 |
+
this.check();
|
16 |
+
},
|
17 |
+
kick: function() {
|
18 |
+
++this.kicks;
|
19 |
+
with ({currentblog: this.blog}) {
|
20 |
+
jQuery.post(BL_strings.ajaxURL,{
|
21 |
+
action: 'BL_import',
|
22 |
+
blogID:this.blog
|
23 |
+
},
|
24 |
+
function(text,result){blogs[currentblog].kickd(text,result)});
|
25 |
+
}
|
26 |
+
},
|
27 |
+
check: function() {
|
28 |
+
++this.checks;
|
29 |
+
with ({currentblog: this.blog}) {
|
30 |
+
jQuery.post(BL_strings.ajaxURL,{
|
31 |
+
action: 'BL_status',
|
32 |
+
blogID:this.blog
|
33 |
+
},
|
34 |
+
function(text,result){blogs[currentblog].checkd(text,result)});
|
35 |
+
}
|
36 |
+
},
|
37 |
+
kickd: function(text, result) {
|
38 |
+
--this.kicks;
|
39 |
+
if ( result == 'error' || text == '0' ) {
|
40 |
+
// TODO: exception handling
|
41 |
+
this.stop();
|
42 |
+
} else {
|
43 |
+
if ( text == 'done' ) {
|
44 |
+
this.stop();
|
45 |
+
this.done();
|
46 |
+
this.check();
|
47 |
+
} else if ( text == 'nothing' ) {
|
48 |
+
this.stop();
|
49 |
+
this.nothing();
|
50 |
+
this.done();
|
51 |
+
} else if ( text == 'continue' ) {
|
52 |
+
this.kick();
|
53 |
+
} else if ( this.mode == 'stopped' )
|
54 |
+
jQuery(this.button).attr('value', BL_strings.cont);
|
55 |
+
|
56 |
+
}
|
57 |
+
},
|
58 |
+
checkd: function(text, result) {
|
59 |
+
--this.checks;
|
60 |
+
if ( result == 'error' || text == '0' ) {
|
61 |
+
// TODO: exception handling
|
62 |
+
this.stop();
|
63 |
+
} else {
|
64 |
+
eval('this.status='+text);
|
65 |
+
this.update();
|
66 |
+
if ( this.cont || this.kicks > 0 ) {
|
67 |
+
with ({currentblog: this.blog}) {
|
68 |
+
setTimeout(function() {
|
69 |
+
blogs[currentblog].check();
|
70 |
+
},parseInt(BL_strings.interval));
|
71 |
+
}
|
72 |
+
}
|
73 |
+
}
|
74 |
+
},
|
75 |
+
update: function() {
|
76 |
+
try
|
77 |
+
{
|
78 |
+
jQuery('#Posts'+this.blog).progressbar({ value: this.status.p1});
|
79 |
+
jQuery('#PostsLabel'+this.blog).text(this.status.p2);
|
80 |
+
jQuery('#Posts'+this.blog).attr('title', this.status.p3);
|
81 |
+
jQuery('#Comments'+this.blog).progressbar({ value: this.status.c1 });
|
82 |
+
jQuery('#CommentsLabel'+this.blog).text(this.status.c2);
|
83 |
+
jQuery('#Comments'+this.blog).attr('title',this.status.c3);
|
84 |
+
jQuery('#Images'+this.blog).progressbar({ value: this.status.i2});
|
85 |
+
jQuery('#ImagesLabel'+this.blog).text(this.status.i1);
|
86 |
+
jQuery('#Images'+this.blog).attr('title',this.status.i3);
|
87 |
+
jQuery('#Links'+this.blog).progressbar({ value: this.status.l2 });
|
88 |
+
jQuery('#LinksLabel'+this.blog).text(this.status.l1);
|
89 |
+
}
|
90 |
+
catch(err)
|
91 |
+
{
|
92 |
+
txt="Error displaying updates for "+this.title+", importer cannot continue\n\n";
|
93 |
+
txt+="Error description: " + err.message + "\n\n";
|
94 |
+
txt+="Click OK to continue.\n\n";
|
95 |
+
alert(txt);
|
96 |
+
}
|
97 |
+
},
|
98 |
+
stop: function() {
|
99 |
+
this.cont = false;
|
100 |
+
},
|
101 |
+
done: function() {
|
102 |
+
this.mode = 'authors';
|
103 |
+
jQuery(this.button).attr('value', BL_strings.authors);
|
104 |
+
},
|
105 |
+
nothing: function() {
|
106 |
+
this.mode = 'nothing';
|
107 |
+
jQuery(this.button).remove();
|
108 |
+
alert(BL_strings.nothing);
|
109 |
+
},
|
110 |
+
getauthors: function() {
|
111 |
+
if ( jQuery('div.wrap').length > 1 )
|
112 |
+
jQuery('div.wrap').gt(0).remove();
|
113 |
+
jQuery('div.wrap').empty().append('<h2>'+BL_strings.authhead+'</h2><h3>' + this.title + '</h3>');
|
114 |
+
jQuery('div.wrap').append('<p id=\"auth\">'+BL_strings.loadauth+'</p>');
|
115 |
+
jQuery('p#auth').load('index.php?import=blogger&noheader=true&authors=1',{blog:this.blog});
|
116 |
+
},
|
117 |
+
init: function() {
|
118 |
+
this.update();
|
119 |
+
var i = this.blog;
|
120 |
+
jQuery(this.button).bind('click', function(){return blogs[i].click();});
|
121 |
+
this.kicks = 0;
|
122 |
+
this.checks = 0;
|
123 |
+
},
|
124 |
+
click: function() {
|
125 |
+
if ( this.mode == 'init' || this.mode == 'stopped' || this.mode == 'posts' || this.mode == 'comments' ) {
|
126 |
+
this.mode = 'started';
|
127 |
+
this.start();
|
128 |
+
jQuery(this.button).attr('value', BL_strings.stop);
|
129 |
+
} else if ( this.mode == 'started' ) {
|
130 |
+
//return false; // let it run...
|
131 |
+
this.mode = 'stopped';
|
132 |
+
this.stop();
|
133 |
+
if ( this.checks > 0 || this.kicks > 0 ) {
|
134 |
+
this.mode = 'stopping';
|
135 |
+
jQuery(this.button).attr('value', BL_strings.stopping);
|
136 |
+
} else {
|
137 |
+
jQuery(this.button).attr('value', BL_strings.cont);
|
138 |
+
}
|
139 |
+
} else if ( this.mode == 'authors' ) {
|
140 |
+
document.location = 'index.php?import=blogger&authors=1&blog='+this.blog;
|
141 |
+
//this.mode = 'authors2';
|
142 |
+
//this.getauthors();
|
143 |
+
}
|
144 |
+
return false;
|
145 |
+
}
|
146 |
+
};
|
blogger-importer.php
CHANGED
@@ -1,1034 +1,490 @@
|
|
1 |
-
<?php
|
2 |
-
|
3 |
-
|
4 |
-
Plugin
|
5 |
-
|
6 |
-
|
7 |
-
Author
|
8 |
-
|
9 |
-
|
10 |
-
License
|
11 |
-
|
12 |
-
|
13 |
-
|
14 |
-
|
15 |
-
|
16 |
-
|
17 |
-
|
18 |
-
|
19 |
-
|
20 |
-
require_once
|
21 |
-
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
|
45 |
-
*
|
46 |
-
*
|
47 |
-
|
48 |
-
|
49 |
-
|
50 |
-
|
51 |
-
|
52 |
-
|
53 |
-
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
60 |
-
|
61 |
-
|
62 |
-
|
63 |
-
|
64 |
-
|
65 |
-
|
66 |
-
|
67 |
-
|
68 |
-
|
69 |
-
|
70 |
-
|
71 |
-
|
72 |
-
|
73 |
-
|
74 |
-
|
75 |
-
|
76 |
-
|
77 |
-
|
78 |
-
|
79 |
-
|
80 |
-
|
81 |
-
|
82 |
-
|
83 |
-
|
84 |
-
|
85 |
-
|
86 |
-
|
87 |
-
|
88 |
-
|
89 |
-
|
90 |
-
|
91 |
-
|
92 |
-
|
93 |
-
|
94 |
-
|
95 |
-
|
96 |
-
|
97 |
-
|
98 |
-
|
99 |
-
|
100 |
-
|
101 |
-
|
102 |
-
|
103 |
-
|
104 |
-
|
105 |
-
|
106 |
-
|
107 |
-
|
108 |
-
|
109 |
-
|
110 |
-
|
111 |
-
|
112 |
-
|
113 |
-
|
114 |
-
|
115 |
-
|
116 |
-
|
117 |
-
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
|
122 |
-
|
123 |
-
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
|
136 |
-
|
137 |
-
|
138 |
-
|
139 |
-
|
140 |
-
|
141 |
-
|
142 |
-
|
143 |
-
|
144 |
-
|
145 |
-
|
146 |
-
|
147 |
-
|
148 |
-
|
149 |
-
|
150 |
-
|
151 |
-
|
152 |
-
|
153 |
-
|
154 |
-
|
155 |
-
|
156 |
-
|
157 |
-
|
158 |
-
|
159 |
-
|
160 |
-
|
161 |
-
|
162 |
-
|
163 |
-
$
|
164 |
-
|
165 |
-
|
166 |
-
$
|
167 |
-
|
168 |
-
|
169 |
-
|
170 |
-
|
171 |
-
|
172 |
-
|
173 |
-
|
174 |
-
|
175 |
-
|
176 |
-
|
177 |
-
|
178 |
-
|
179 |
-
|
180 |
-
|
181 |
-
|
182 |
-
|
183 |
-
|
184 |
-
|
185 |
-
|
186 |
-
|
187 |
-
|
188 |
-
|
189 |
-
|
190 |
-
|
191 |
-
|
192 |
-
|
193 |
-
|
194 |
-
|
195 |
-
|
196 |
-
|
197 |
-
|
198 |
-
|
199 |
-
|
200 |
-
|
201 |
-
|
202 |
-
|
203 |
-
|
204 |
-
|
205 |
-
|
206 |
-
|
207 |
-
|
208 |
-
|
209 |
-
|
210 |
-
|
211 |
-
|
212 |
-
|
213 |
-
|
214 |
-
{
|
215 |
-
$
|
216 |
-
|
217 |
-
|
218 |
-
|
219 |
-
|
220 |
-
|
221 |
-
|
222 |
-
|
223 |
-
|
224 |
-
|
225 |
-
|
226 |
-
|
227 |
-
|
228 |
-
|
229 |
-
|
230 |
-
|
231 |
-
|
232 |
-
|
233 |
-
|
234 |
-
|
235 |
-
|
236 |
-
|
237 |
-
|
238 |
-
|
239 |
-
|
240 |
-
|
241 |
-
|
242 |
-
|
243 |
-
|
244 |
-
|
245 |
-
|
246 |
-
|
247 |
-
|
248 |
-
|
249 |
-
|
250 |
-
|
251 |
-
|
252 |
-
|
253 |
-
|
254 |
-
|
255 |
-
|
256 |
-
|
257 |
-
|
258 |
-
|
259 |
-
|
260 |
-
|
261 |
-
|
262 |
-
|
263 |
-
|
264 |
-
|
265 |
-
|
266 |
-
|
267 |
-
|
268 |
-
|
269 |
-
|
270 |
-
|
271 |
-
|
272 |
-
|
273 |
-
|
274 |
-
//
|
275 |
-
$
|
276 |
-
|
277 |
-
$
|
278 |
-
|
279 |
-
$
|
280 |
-
$
|
281 |
-
$
|
282 |
-
$
|
283 |
-
$
|
284 |
-
$
|
285 |
-
|
286 |
-
$
|
287 |
-
|
288 |
-
|
289 |
-
|
290 |
-
'blogger-importer');
|
291 |
-
|
292 |
-
|
293 |
-
|
294 |
-
|
295 |
-
|
296 |
-
|
297 |
-
|
298 |
-
|
299 |
-
|
300 |
-
|
301 |
-
|
302 |
-
|
303 |
-
|
304 |
-
|
305 |
-
|
306 |
-
|
307 |
-
|
308 |
-
|
309 |
-
|
310 |
-
|
311 |
-
|
312 |
-
|
313 |
-
|
314 |
-
|
315 |
-
|
316 |
-
|
317 |
-
|
318 |
-
|
319 |
-
|
320 |
-
|
321 |
-
|
322 |
-
|
323 |
-
|
324 |
-
|
325 |
-
|
326 |
-
|
327 |
-
|
328 |
-
|
329 |
-
|
330 |
-
|
331 |
-
|
332 |
-
|
333 |
-
|
334 |
-
|
335 |
-
|
336 |
-
|
337 |
-
|
338 |
-
|
339 |
-
|
340 |
-
|
341 |
-
|
342 |
-
|
343 |
-
|
344 |
-
|
345 |
-
|
346 |
-
|
347 |
-
|
348 |
-
|
349 |
-
|
350 |
-
|
351 |
-
|
352 |
-
|
353 |
-
|
354 |
-
|
355 |
-
|
356 |
-
|
357 |
-
|
358 |
-
|
359 |
-
|
360 |
-
|
361 |
-
|
362 |
-
|
363 |
-
|
364 |
-
|
365 |
-
|
366 |
-
|
367 |
-
|
368 |
-
|
369 |
-
|
370 |
-
|
371 |
-
|
372 |
-
|
373 |
-
|
374 |
-
|
375 |
-
|
376 |
-
|
377 |
-
|
378 |
-
|
379 |
-
|
380 |
-
|
381 |
-
|
382 |
-
|
383 |
-
|
384 |
-
|
385 |
-
|
386 |
-
|
387 |
-
|
388 |
-
|
389 |
-
|
390 |
-
|
391 |
-
|
392 |
-
|
393 |
-
|
394 |
-
|
395 |
-
|
396 |
-
|
397 |
-
|
398 |
-
|
399 |
-
|
400 |
-
|
401 |
-
|
402 |
-
|
403 |
-
|
404 |
-
|
405 |
-
|
406 |
-
|
407 |
-
|
408 |
-
|
409 |
-
|
410 |
-
|
411 |
-
|
412 |
-
|
413 |
-
|
414 |
-
|
415 |
-
|
416 |
-
|
417 |
-
|
418 |
-
|
419 |
-
|
420 |
-
|
421 |
-
|
422 |
-
|
423 |
-
|
424 |
-
|
425 |
-
|
426 |
-
|
427 |
-
|
428 |
-
|
429 |
-
|
430 |
-
|
431 |
-
|
432 |
-
|
433 |
-
|
434 |
-
|
435 |
-
|
436 |
-
|
437 |
-
|
438 |
-
|
439 |
-
|
440 |
-
|
441 |
-
|
442 |
-
|
443 |
-
|
444 |
-
|
445 |
-
|
446 |
-
|
447 |
-
|
448 |
-
|
449 |
-
|
450 |
-
|
451 |
-
|
452 |
-
|
453 |
-
|
454 |
-
|
455 |
-
|
456 |
-
|
457 |
-
|
458 |
-
|
459 |
-
|
460 |
-
|
461 |
-
|
462 |
-
|
463 |
-
|
464 |
-
|
465 |
-
|
466 |
-
|
467 |
-
|
468 |
-
|
469 |
-
|
470 |
-
|
471 |
-
|
472 |
-
|
473 |
-
|
474 |
-
|
475 |
-
|
476 |
-
|
477 |
-
$
|
478 |
-
|
479 |
-
|
480 |
-
|
481 |
-
|
482 |
-
|
483 |
-
|
484 |
-
|
485 |
-
|
486 |
-
|
487 |
-
|
488 |
-
|
489 |
-
|
490 |
-
|
491 |
-
$url = $this->blogs[$importing_blog]['posts_url'];
|
492 |
-
|
493 |
-
$response = $this->oauth_get($url, array('max-results' => MAX_RESULTS, 'start-index' => $start_index));
|
494 |
-
|
495 |
-
if ($response == false)
|
496 |
-
break;
|
497 |
-
|
498 |
-
// parse the feed
|
499 |
-
$feed = new SimplePie();
|
500 |
-
$feed->set_item_class('WP_SimplePie_Blog_Item');
|
501 |
-
$feed->set_sanitize_class('Blogger_Importer_Sanitize');
|
502 |
-
$feed->set_raw_data($response);
|
503 |
-
$feed->init();
|
504 |
-
|
505 |
-
foreach ($feed->get_items() as $item)
|
506 |
-
{
|
507 |
-
|
508 |
-
$blogentry = new BloggerEntry();
|
509 |
-
|
510 |
-
$blogentry->id = $item->get_id();
|
511 |
-
$blogentry->published = $item->get_published();
|
512 |
-
$blogentry->updated = $item->get_updated();
|
513 |
-
$blogentry->isDraft = $item->get_draft_status($item);
|
514 |
-
$blogentry->title = $item->get_title();
|
515 |
-
$blogentry->content = $item->get_content();
|
516 |
-
$blogentry->author = $item->get_author()->get_name();
|
517 |
-
$blogentry->geotags = $item->get_geotags();
|
518 |
-
|
519 |
-
$linktypes = array('replies', 'edit', 'self', 'alternate');
|
520 |
-
foreach ($linktypes as $type)
|
521 |
-
{
|
522 |
-
$links = $item->get_links($type);
|
523 |
-
|
524 |
-
if (!is_null($links))
|
525 |
-
{
|
526 |
-
foreach ($links as $link)
|
527 |
-
{
|
528 |
-
$blogentry->links[] = array('rel' => $type, 'href' => $link);
|
529 |
-
}
|
530 |
-
}
|
531 |
-
}
|
532 |
-
|
533 |
-
$cats = $item->get_categories();
|
534 |
-
|
535 |
-
if (!is_null($cats))
|
536 |
-
{
|
537 |
-
foreach ($cats as $cat)
|
538 |
-
{
|
539 |
-
$blogentry->categories[] = $cat->term;
|
540 |
-
}
|
541 |
-
}
|
542 |
-
|
543 |
-
$result = $this->import_post($blogentry);
|
544 |
-
|
545 |
-
//Ref: Not importing properly http://core.trac.wordpress.org/ticket/19096
|
546 |
-
//Simplified this section to count what is loaded rather than parsing the results again
|
547 |
-
$start_index++;
|
548 |
-
}
|
549 |
-
|
550 |
-
$this->blogs[$importing_blog]['posts_start_index'] = $start_index;
|
551 |
-
|
552 |
-
$this->save_vars();
|
553 |
-
|
554 |
-
} while ($this->blogs[$importing_blog]['total_posts'] > $start_index && $this->have_time()); //have time function will "die" if it's out of time
|
555 |
-
}
|
556 |
-
|
557 |
-
|
558 |
-
if (isset($this->blogs[$importing_blog]['comments_start_index']))
|
559 |
-
$start_index = (int)$this->blogs[$importing_blog]['comments_start_index'];
|
560 |
-
else
|
561 |
-
$start_index = 1;
|
562 |
-
|
563 |
-
if ($start_index > 0 && $this->blogs[$importing_blog]['total_comments'] > 0)
|
564 |
-
{
|
565 |
-
|
566 |
-
$this->blogs[$importing_blog]['mode'] = 'comments';
|
567 |
-
do
|
568 |
-
{
|
569 |
-
$index = $struct = $entries = array();
|
570 |
-
|
571 |
-
//So we can link up the comments as we go we need to load them in reverse order
|
572 |
-
//Reverse the start index as the GData Blogger feed can't be sorted
|
573 |
-
$batch = ((floor(($this->blogs[$importing_blog]['total_comments'] - $start_index) / MAX_RESULTS) * MAX_RESULTS) + 1);
|
574 |
-
|
575 |
-
$response = $this->oauth_get($this->blogs[$importing_blog]['comments_url'], array('max-results' => MAX_RESULTS, 'start-index' => $batch));
|
576 |
-
|
577 |
-
// parse the feed
|
578 |
-
$feed = new SimplePie();
|
579 |
-
$feed->set_item_class('WP_SimplePie_Blog_Item');
|
580 |
-
// Use the standard "stricter" sanitize class for comments
|
581 |
-
$feed->set_raw_data($response);
|
582 |
-
$feed->init();
|
583 |
-
|
584 |
-
//Reverse the batch so we load the oldest comments first and hence can link up nested comments
|
585 |
-
$comments = array_reverse($feed->get_items());
|
586 |
-
|
587 |
-
if (!is_null($comments))
|
588 |
-
{
|
589 |
-
foreach ($comments as $item)
|
590 |
-
{
|
591 |
-
|
592 |
-
$blogentry = new BloggerEntry();
|
593 |
-
$blogentry->id = $item->get_id();
|
594 |
-
$blogentry->updated = $item->get_updated();
|
595 |
-
$blogentry->content = $item->get_content();
|
596 |
-
$blogentry->author = $item->get_author()->get_name();
|
597 |
-
$blogentry->authoruri = $item->get_author()->get_link();
|
598 |
-
$blogentry->authoremail = $item->get_author()->get_email();
|
599 |
-
|
600 |
-
$temp = $item->get_item_tags('http://purl.org/syndication/thread/1.0', 'in-reply-to');
|
601 |
-
|
602 |
-
foreach ($temp as $t)
|
603 |
-
{
|
604 |
-
if (isset($t['attribs']['']['source']))
|
605 |
-
{
|
606 |
-
$blogentry->source = $t['attribs']['']['source'];
|
607 |
-
}
|
608 |
-
}
|
609 |
-
|
610 |
-
//Get the links
|
611 |
-
$linktypes = array('edit', 'self', 'alternate', 'related');
|
612 |
-
foreach ($linktypes as $type)
|
613 |
-
{
|
614 |
-
$links = $item->get_links($type);
|
615 |
-
if (!is_null($links))
|
616 |
-
{
|
617 |
-
foreach ($links as $link)
|
618 |
-
{
|
619 |
-
$blogentry->links[] = array('rel' => $type, 'href' => $link);
|
620 |
-
}
|
621 |
-
}
|
622 |
-
}
|
623 |
-
|
624 |
-
$this->import_comment($blogentry);
|
625 |
-
$start_index++;
|
626 |
-
}
|
627 |
-
}
|
628 |
-
|
629 |
-
$this->blogs[$importing_blog]['comments_start_index'] = $start_index;
|
630 |
-
$this->save_vars();
|
631 |
-
} while ($this->blogs[$importing_blog]['total_comments'] > $start_index && $this->have_time());
|
632 |
-
}
|
633 |
-
|
634 |
-
$this->blogs[$importing_blog]['mode'] = 'authors';
|
635 |
-
$this->save_vars();
|
636 |
-
|
637 |
-
if (!$this->blogs[$importing_blog]['posts_done'] && !$this->blogs[$importing_blog]['comments_done'])
|
638 |
-
self::ajax_die('nothing');
|
639 |
-
|
640 |
-
do_action('import_done', 'blogger');
|
641 |
-
self::ajax_die('done');
|
642 |
-
}
|
643 |
-
|
644 |
-
function no_apos($string)
|
645 |
-
{
|
646 |
-
return str_replace(''', "'", $string);
|
647 |
-
}
|
648 |
-
|
649 |
-
function min_whitespace($string)
|
650 |
-
{
|
651 |
-
return preg_replace('|\s+|', ' ', $string);
|
652 |
-
}
|
653 |
-
|
654 |
-
function _normalize_tag($matches)
|
655 |
-
{
|
656 |
-
return '<' . strtolower($matches[1]);
|
657 |
-
}
|
658 |
-
|
659 |
-
function import_post($entry)
|
660 |
-
{
|
661 |
-
global $importing_blog;
|
662 |
-
|
663 |
-
foreach ($entry->links as $link)
|
664 |
-
{
|
665 |
-
// save the self link as meta
|
666 |
-
if ($link['rel'] == 'self')
|
667 |
-
{
|
668 |
-
$postself = $link['href'];
|
669 |
-
$parts = parse_url($link['href']);
|
670 |
-
$entry->old_permalink = $parts['path'];
|
671 |
-
}
|
672 |
-
|
673 |
-
// get the old URI for the page when available
|
674 |
-
if ($link['rel'] == 'alternate')
|
675 |
-
{
|
676 |
-
$parts = parse_url($link['href']);
|
677 |
-
$entry->bookmark = $parts['path'];
|
678 |
-
}
|
679 |
-
|
680 |
-
// save the replies feed link as meta (ignore the comment form one)
|
681 |
-
if ($link['rel'] == 'replies' && false === strpos($link['href'], '#comment-form'))
|
682 |
-
{
|
683 |
-
$postreplies = $link['href'];
|
684 |
-
}
|
685 |
-
}
|
686 |
-
|
687 |
-
//Check if we are double cleaning here? Does the Simplepie already do all this?
|
688 |
-
$post_date = $entry->published;
|
689 |
-
$post_content = trim(addslashes($this->no_apos(@html_entity_decode($entry->content, ENT_COMPAT, get_option('blog_charset')))));
|
690 |
-
$post_title = trim(addslashes($this->no_apos($this->min_whitespace($entry->title))));
|
691 |
-
|
692 |
-
$post_status = $entry->isDraft ? 'draft' : 'publish';
|
693 |
-
|
694 |
-
// N.B. Clean up of $post_content is now part of the sanitize class
|
695 |
-
|
696 |
-
// Checks for duplicates
|
697 |
-
if (isset($this->blogs[$importing_blog]['posts'][$entry->old_permalink]))
|
698 |
-
{
|
699 |
-
$this->blogs[$importing_blog]['posts_skipped']++;
|
700 |
-
} elseif ($post_id = post_exists($post_title, $post_content, $post_date))
|
701 |
-
{
|
702 |
-
$this->blogs[$importing_blog]['posts'][$entry->old_permalink] = $post_id;
|
703 |
-
$this->blogs[$importing_blog]['posts_skipped']++;
|
704 |
-
} else
|
705 |
-
{
|
706 |
-
$post = compact('post_date', 'post_content', 'post_title', 'post_status');
|
707 |
-
|
708 |
-
$post_id = wp_insert_post($post);
|
709 |
-
if (is_wp_error($post_id))
|
710 |
-
return $post_id;
|
711 |
-
|
712 |
-
wp_create_categories(array_map('addslashes', $entry->categories), $post_id);
|
713 |
-
|
714 |
-
$author = $this->no_apos(strip_tags($entry->author));
|
715 |
-
|
716 |
-
add_post_meta($post_id, 'blogger_blog', $this->blogs[$importing_blog]['host'], true);
|
717 |
-
add_post_meta($post_id, 'blogger_author', $author, true);
|
718 |
-
|
719 |
-
//Use the page id if available or the blogger internal id if it's a draft
|
720 |
-
if ($entry->isDraft | !isset($entry->bookmark))
|
721 |
-
add_post_meta($post_id, 'blogger_permalink', $entry->old_permalink, true);
|
722 |
-
else
|
723 |
-
add_post_meta($post_id, 'blogger_permalink', $entry->bookmark, true);
|
724 |
-
|
725 |
-
add_post_meta($post_id, '_blogger_self', $postself, true);
|
726 |
-
|
727 |
-
if (isset($entry->geotags)) {
|
728 |
-
add_post_meta($post_id,'geo_latitude',$entry->geotags['geo_latitude']);
|
729 |
-
add_post_meta($post_id,'geo_longitude',$entry->geotags['geo_longitude']);
|
730 |
-
if (isset($entry->geotags['geo_address'])) {
|
731 |
-
add_post_meta($post_id,'geo_address',$entry->geotags['geo_address']);
|
732 |
-
}
|
733 |
-
}
|
734 |
-
|
735 |
-
$this->blogs[$importing_blog]['posts'][$entry->old_permalink] = $post_id;
|
736 |
-
|
737 |
-
$this->blogs[$importing_blog]['posts_done']++;
|
738 |
-
}
|
739 |
-
$this->save_vars();
|
740 |
-
return;
|
741 |
-
}
|
742 |
-
|
743 |
-
function import_comment($entry)
|
744 |
-
{
|
745 |
-
global $importing_blog;
|
746 |
-
|
747 |
-
$parts = parse_url($entry->source);
|
748 |
-
$entry->old_post_permalink = $parts['path']; //Will be something like this '/feeds/417730729915399755/posts/default/8397846992898424746'
|
749 |
-
|
750 |
-
// Drop the #fragment and we have the comment's old post permalink.
|
751 |
-
foreach ($entry->links as $link)
|
752 |
-
{
|
753 |
-
if ($link['rel'] == 'alternate')
|
754 |
-
{
|
755 |
-
$parts = parse_url($link['href']);
|
756 |
-
$entry->old_permalink = $parts['fragment'];
|
757 |
-
}
|
758 |
-
//Parent post for nested links
|
759 |
-
if ($link['rel'] == 'related')
|
760 |
-
{
|
761 |
-
$parts = parse_url($link['href']);
|
762 |
-
$entry->related = $parts['path'];
|
763 |
-
}
|
764 |
-
if ($link['rel'] == 'self')
|
765 |
-
{
|
766 |
-
$parts = parse_url($link['href']);
|
767 |
-
$entry->self = $parts['path'];
|
768 |
-
}
|
769 |
-
}
|
770 |
-
|
771 |
-
//Check for duplicated cleanup here
|
772 |
-
$comment_post_ID = (int)$this->blogs[$importing_blog]['posts'][$entry->old_post_permalink];
|
773 |
-
$comment_author = addslashes($this->no_apos(strip_tags($entry->author)));
|
774 |
-
$comment_author_url = addslashes($this->no_apos(strip_tags($entry->authoruri)));
|
775 |
-
$comment_author_email = addslashes($this->no_apos(strip_tags($entry->authoremail)));
|
776 |
-
$comment_date = $entry->updated;
|
777 |
-
|
778 |
-
// Clean up content
|
779 |
-
// Again, check if the Simplepie is already handling all of the cleaning
|
780 |
-
$comment_content = addslashes($this->no_apos(@html_entity_decode($entry->content, ENT_COMPAT, get_option('blog_charset'))));
|
781 |
-
$comment_content = preg_replace_callback('|<(/?[A-Z]+)|', array(&$this, '_normalize_tag'), $comment_content);
|
782 |
-
$comment_content = str_replace('<br>', '<br />', $comment_content);
|
783 |
-
$comment_content = str_replace('<hr>', '<hr />', $comment_content);
|
784 |
-
|
785 |
-
// Nested comment?
|
786 |
-
if (!is_null($entry->related))
|
787 |
-
{
|
788 |
-
$comment_parent = $this->blogs[$importing_blog]['comments'][$entry->related];
|
789 |
-
}
|
790 |
-
|
791 |
-
// if the post does not exist then we need stop and not add the comment
|
792 |
-
if ($comment_post_ID != 0)
|
793 |
-
{
|
794 |
-
// Checks for duplicates
|
795 |
-
if (isset($this->blogs[$importing_blog][$entry->id]) || $this->comment_exists($comment_post_ID, $comment_author, $comment_date))
|
796 |
-
{
|
797 |
-
$this->blogs[$importing_blog]['comments_skipped']++;
|
798 |
-
} else
|
799 |
-
{
|
800 |
-
$comment = compact('comment_post_ID', 'comment_author', 'comment_author_url', 'comment_author_email', 'comment_date', 'comment_content', 'comment_parent');
|
801 |
-
|
802 |
-
$comment = wp_filter_comment($comment);
|
803 |
-
$comment_id = wp_insert_comment($comment);
|
804 |
-
|
805 |
-
$this->blogs[$importing_blog]['comments'][$entry->id] = $comment_id;
|
806 |
-
$this->blogs[$importing_blog]['comments'][$entry->self] = $comment_id; //For nested comments
|
807 |
-
|
808 |
-
$this->blogs[$importing_blog]['comments_done']++;
|
809 |
-
}
|
810 |
-
} else
|
811 |
-
{
|
812 |
-
$this->blogs[$importing_blog]['comments_skipped']++;
|
813 |
-
}
|
814 |
-
$this->save_vars();
|
815 |
-
}
|
816 |
-
|
817 |
-
function ajax_die($data)
|
818 |
-
{
|
819 |
-
ob_clean(); //Discard any debug messages or other fluff already sent
|
820 |
-
header('Content-Type: text/plain');
|
821 |
-
die($data);
|
822 |
-
}
|
823 |
-
|
824 |
-
|
825 |
-
function get_js_status($blog = false)
|
826 |
-
{
|
827 |
-
global $importing_blog;
|
828 |
-
if ($blog === false)
|
829 |
-
$blog = $this->blogs[$importing_blog];
|
830 |
-
else
|
831 |
-
$blog = $this->blogs[$blog];
|
832 |
-
|
833 |
-
$p1 = isset($blog['posts_done']) ? (int)$blog['posts_done'] : 0;
|
834 |
-
$p2 = isset($blog['total_posts']) ? (int)$blog['total_posts'] : 0;
|
835 |
-
$p3 = isset($blog['posts_skipped']) ? (int)$blog['posts_skipped'] : 0;
|
836 |
-
$c1 = isset($blog['comments_done']) ? (int)$blog['comments_done'] : 0;
|
837 |
-
$c2 = isset($blog['total_comments']) ? (int)$blog['total_comments'] : 0;
|
838 |
-
$c3 = isset($blog['comments_skipped']) ? (int)$blog['comments_skipped'] : 0;
|
839 |
-
return "{p1:$p1,p2:$p2,p3:$p3,c1:$c1,c2:$c2,c3:$c3}";
|
840 |
-
}
|
841 |
-
|
842 |
-
function get_author_form($blog = false)
|
843 |
-
{
|
844 |
-
global $importing_blog, $wpdb, $current_user;
|
845 |
-
if ($blog === false)
|
846 |
-
$blog = &$this->blogs[$importing_blog];
|
847 |
-
else
|
848 |
-
$blog = &$this->blogs[$blog];
|
849 |
-
|
850 |
-
if (!isset($blog['authors']))
|
851 |
-
{
|
852 |
-
$post_ids = array_values($blog['posts']);
|
853 |
-
$authors = (array )$wpdb->get_col("SELECT DISTINCT meta_value FROM $wpdb->postmeta WHERE meta_key = 'blogger_author' AND post_id IN (" . join(',', $post_ids) . ")");
|
854 |
-
$blog['authors'] = array_map(null, $authors, array_fill(0, count($authors), $current_user->ID));
|
855 |
-
$this->save_vars();
|
856 |
-
}
|
857 |
-
|
858 |
-
$directions = sprintf(__('All posts were imported with the current user as author. Use this form to move each Blogger user’s posts to a different WordPress user. You may <a href="%s">add users</a> and then return to this page and complete the user mapping. This form may be used as many times as you like until you activate the “Restart” function below.',
|
859 |
-
'blogger-importer'), 'users.php');
|
860 |
-
$heading = __('Author mapping', 'blogger-importer');
|
861 |
-
$blogtitle = "{$blog['title']} ({$blog['host']})";
|
862 |
-
$mapthis = __('Blogger username', 'blogger-importer');
|
863 |
-
$tothis = __('WordPress login', 'blogger-importer');
|
864 |
-
$submit = esc_js(__('Save Changes', 'blogger-importer'));
|
865 |
-
$rows = '';
|
866 |
-
|
867 |
-
foreach ($blog['authors'] as $i => $author)
|
868 |
-
$rows .= "<tr><td><label for='authors[$i]'>{$author[0]}</label></td><td><select name='authors[$i]' id='authors[$i]'>" . $this->get_user_options($author[1]) . "</select></td></tr>";
|
869 |
-
|
870 |
-
return "<div class='wrap'>".screen_icon()."<h2>$heading</h2><h3>$blogtitle</h3><p>$directions</p><form action='index.php?import=blogger&noheader=true&saveauthors=1' method='post'><input type='hidden' name='blog' value='" .
|
871 |
-
esc_attr($importing_blog) . "' /><table cellpadding='5'><thead><td>$mapthis</td><td>$tothis</td></thead>$rows<tr><td></td><td class='submit'><input type='submit' class='button authorsubmit' value='$submit' /></td></tr></table></form></div>";
|
872 |
-
}
|
873 |
-
|
874 |
-
function get_user_options($current)
|
875 |
-
{
|
876 |
-
global $importer_users;
|
877 |
-
if (!isset($importer_users))
|
878 |
-
$importer_users = (array )get_users(); //Function: get_users_of_blog() Deprecated in version 3.1. Use get_users() instead.
|
879 |
-
|
880 |
-
$options = '';
|
881 |
-
|
882 |
-
foreach ($importer_users as $user)
|
883 |
-
{
|
884 |
-
$sel = ($user->ID == $current) ? " selected='selected'" : '';
|
885 |
-
$options .= "<option value='$user->ID'$sel>$user->display_name</option>";
|
886 |
-
}
|
887 |
-
|
888 |
-
return $options;
|
889 |
-
}
|
890 |
-
|
891 |
-
function save_authors()
|
892 |
-
{
|
893 |
-
global $importing_blog, $wpdb;
|
894 |
-
$blog = &$this->blogs[$importing_blog]; //Get a reference to blogs so we don't have to write it longhand
|
895 |
-
|
896 |
-
$authors = (array )$_POST['authors'];
|
897 |
-
|
898 |
-
$host = $blog['host'];
|
899 |
-
|
900 |
-
// Get an array of posts => authors
|
901 |
-
$post_ids = (array )$wpdb->get_col($wpdb->prepare("SELECT post_id FROM $wpdb->postmeta WHERE meta_key = 'blogger_blog' AND meta_value = %s", $host));
|
902 |
-
$post_ids = join(',', $post_ids);
|
903 |
-
$results = (array )$wpdb->get_results("SELECT post_id, meta_value FROM $wpdb->postmeta WHERE meta_key = 'blogger_author' AND post_id IN ($post_ids)");
|
904 |
-
foreach ($results as $row)
|
905 |
-
$authors_posts[$row->post_id] = $row->meta_value;
|
906 |
-
|
907 |
-
foreach ($authors as $author => $user_id)
|
908 |
-
{
|
909 |
-
$user_id = (int)$user_id;
|
910 |
-
|
911 |
-
// Skip authors that haven't been changed
|
912 |
-
if ($user_id == $blog['authors'][$author][1])
|
913 |
-
continue;
|
914 |
-
|
915 |
-
// Get a list of the selected author's posts
|
916 |
-
$post_ids = (array )array_keys($authors_posts, $blog['authors'][$author][0]);
|
917 |
-
$post_ids = join(',', $post_ids);
|
918 |
-
|
919 |
-
$wpdb->query($wpdb->prepare("UPDATE $wpdb->posts SET post_author = %d WHERE id IN ($post_ids)", $user_id));
|
920 |
-
$blog['authors'][$author][1] = $user_id;
|
921 |
-
}
|
922 |
-
$this->save_vars();
|
923 |
-
|
924 |
-
wp_redirect('edit.php');
|
925 |
-
}
|
926 |
-
|
927 |
-
function restart()
|
928 |
-
{
|
929 |
-
global $wpdb;
|
930 |
-
$options = get_option('blogger_importer');
|
931 |
-
|
932 |
-
if ( check_admin_referer( 'clear-blogger-importer', 'clear-blogger-importer-nonce' ) ) {
|
933 |
-
delete_option('blogger_importer');
|
934 |
-
$wpdb->query("DELETE FROM $wpdb->postmeta WHERE meta_key = 'blogger_author'");
|
935 |
-
}
|
936 |
-
wp_redirect('?import=blogger');
|
937 |
-
exit;
|
938 |
-
}
|
939 |
-
|
940 |
-
// Step 9: Congratulate the user
|
941 |
-
function congrats()
|
942 |
-
{
|
943 |
-
$blog = (int)$_GET['blog'];
|
944 |
-
echo '<h1>' . __('Congratulations!', 'blogger-importer') . '</h1><p>' . __('Now that you have imported your Blogger blog into WordPress, what are you going to do? Here are some suggestions:',
|
945 |
-
'blogger-importer') . '</p><ul><li>' . __('That was hard work! Take a break.', 'blogger-importer') . '</li>';
|
946 |
-
if (count($this->import['blogs']) > 1)
|
947 |
-
echo '<li>' . __('In case you haven’t done it already, you can import the posts from your other blogs:', 'blogger-importer') . $this->show_blogs() . '</li>';
|
948 |
-
if ($n = count($this->import['blogs'][$blog]['newusers']))
|
949 |
-
echo '<li>' . sprintf(__('Go to <a href="%s" target="%s">Authors & Users</a>, where you can modify the new user(s) or delete them. If you want to make all of the imported posts yours, you will be given that option when you delete the new authors.',
|
950 |
-
'blogger-importer'), 'users.php', '_parent') . '</li>';
|
951 |
-
echo '<li>' . __('For security, click the link below to reset this importer.', 'blogger-importer') . '</li>';
|
952 |
-
echo '</ul>';
|
953 |
-
}
|
954 |
-
|
955 |
-
// Figures out what to do, then does it.
|
956 |
-
function start()
|
957 |
-
{
|
958 |
-
if (isset($_POST['restart']))
|
959 |
-
$this->restart();
|
960 |
-
|
961 |
-
$options = get_option('blogger_importer');
|
962 |
-
|
963 |
-
if (is_array($options))
|
964 |
-
foreach ($options as $key => $value)
|
965 |
-
$this->$key = $value;
|
966 |
-
|
967 |
-
if (isset($_REQUEST['blog']))
|
968 |
-
{
|
969 |
-
$blog = is_array($_REQUEST['blog']) ? array_shift($keys = array_keys($_REQUEST['blog'])) : $_REQUEST['blog'];
|
970 |
-
$blog = (int)$blog;
|
971 |
-
$result = $this->import_blog($blog);
|
972 |
-
if (is_wp_error($result))
|
973 |
-
echo $result->get_error_message();
|
974 |
-
} elseif (isset($_GET['token']) && isset($_GET['token_secret']))
|
975 |
-
$this->auth();
|
976 |
-
elseif (isset($this->token) && isset($this->token_secret))
|
977 |
-
$this->show_blogs();
|
978 |
-
else
|
979 |
-
$this->greet();
|
980 |
-
|
981 |
-
$saved = $this->save_vars();
|
982 |
-
|
983 |
-
if ($saved && !isset($_GET['noheader']))
|
984 |
-
{
|
985 |
-
$restart = __('Restart', 'blogger-importer');
|
986 |
-
$message = __('We have saved some information about your Blogger account in your WordPress database. Clearing this information will allow you to start over. Restarting will not affect any posts you have already imported. If you attempt to re-import a blog, duplicate posts and comments will be skipped.',
|
987 |
-
'blogger-importer');
|
988 |
-
$submit = esc_attr__('Clear account information', 'blogger-importer');
|
989 |
-
echo "<div class='wrap'><h2>$restart</h2><p>$message</p><form method='post' action='?import=blogger&noheader=true'>";
|
990 |
-
wp_nonce_field( 'clear-blogger-importer', 'clear-blogger-importer-nonce' );
|
991 |
-
echo "<p class='submit' style='text-align:left;'><input type='submit' class='button' value='$submit' name='restart' /></p></form></div>";
|
992 |
-
}
|
993 |
-
}
|
994 |
-
|
995 |
-
function save_vars()
|
996 |
-
{
|
997 |
-
$vars = get_object_vars($this);
|
998 |
-
update_option('blogger_importer', $vars);
|
999 |
-
|
1000 |
-
return !empty($vars);
|
1001 |
-
}
|
1002 |
-
|
1003 |
-
function comment_exists($post_id, $comment_author, $comment_date)
|
1004 |
-
{
|
1005 |
-
//Do we have 2 comments for the same author at the same time, on the same post?
|
1006 |
-
//returns comment id
|
1007 |
-
global $wpdb;
|
1008 |
-
|
1009 |
-
$comment_author = stripslashes($comment_author);
|
1010 |
-
$comment_date = stripslashes($comment_date);
|
1011 |
-
|
1012 |
-
return $wpdb->get_var($wpdb->prepare("SELECT comment_ID FROM $wpdb->comments
|
1013 |
-
WHERE comment_post_ID = %s and comment_author = %s AND comment_date = %s", $post_id, $comment_author, $comment_date));
|
1014 |
-
}
|
1015 |
-
|
1016 |
-
}
|
1017 |
-
|
1018 |
-
class BloggerEntry
|
1019 |
-
{
|
1020 |
-
var $links = array();
|
1021 |
-
var $categories = array();
|
1022 |
-
}
|
1023 |
-
|
1024 |
-
} // class_exists( 'WP_Importer' )
|
1025 |
-
|
1026 |
-
function blogger_importer_init()
|
1027 |
-
{
|
1028 |
-
load_plugin_textdomain('blogger-importer', false, dirname(plugin_basename(__file__)) . '/languages');
|
1029 |
-
|
1030 |
-
$blogger_import = new Blogger_Import();
|
1031 |
-
register_importer('blogger', __('Blogger', 'blogger-importer'), __('Import categories, posts and comments then maps users from a Blogger blog.', 'blogger-importer'), array($blogger_import, 'start'));
|
1032 |
-
|
1033 |
-
}
|
1034 |
-
add_action('admin_init', 'blogger_importer_init');
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/*
|
4 |
+
Plugin Name: Blogger Importer
|
5 |
+
Plugin URI: http://wordpress.org/extend/plugins/blogger-importer/
|
6 |
+
Description: Imports posts, comments, images and tags from a Blogger blog then migrates authors to WordPress users.
|
7 |
+
Author: wordpressdotorg
|
8 |
+
Author URI: http://wordpress.org/
|
9 |
+
Version: 0.8
|
10 |
+
License: GPLv2
|
11 |
+
License URI: http://www.gnu.org/licenses/old-licenses/gpl-2.0.html
|
12 |
+
*/
|
13 |
+
|
14 |
+
// Wordpress Classes needed by importer
|
15 |
+
require_once ABSPATH . 'wp-admin/includes/import.php';
|
16 |
+
if (!class_exists('WP_Importer'))
|
17 |
+
{
|
18 |
+
$class_wp_importer = ABSPATH . 'wp-admin/includes/class-wp-importer.php';
|
19 |
+
if (file_exists($class_wp_importer))
|
20 |
+
require_once $class_wp_importer;
|
21 |
+
}
|
22 |
+
if (!class_exists('WP_Error'))
|
23 |
+
{
|
24 |
+
$class_wp_error = ABSPATH . 'wp-includes/class-wp-error.php';
|
25 |
+
if (file_exists($class_wp_error))
|
26 |
+
require_once $class_wp_error;
|
27 |
+
}
|
28 |
+
|
29 |
+
|
30 |
+
require_once ABSPATH . WPINC . '/class-feed.php';
|
31 |
+
|
32 |
+
// Custom classes used by importer
|
33 |
+
require_once dirname( __FILE__ ) . '/blogger-importer-sanitize.php';
|
34 |
+
require_once dirname( __FILE__ ) . '/blogger-importer-blogitem.php';
|
35 |
+
require_once dirname( __FILE__ ) . '/blogger-importer-blog.php';
|
36 |
+
require_once dirname( __FILE__ ) . '/blogger-importer-connector.php';
|
37 |
+
require_once dirname( __FILE__ ) . '/blogger-entry.php';
|
38 |
+
require_once dirname( __FILE__ ) . '/comment-entry.php';
|
39 |
+
require_once dirname( __FILE__ ) . '/oauth.php';
|
40 |
+
require_once dirname( __FILE__ ) . '/blogger-importer-table.php';
|
41 |
+
|
42 |
+
/**
|
43 |
+
* Blogger Importer
|
44 |
+
*
|
45 |
+
* @package WordPress
|
46 |
+
* @subpackage Importer
|
47 |
+
*/
|
48 |
+
if (class_exists('WP_Importer'))
|
49 |
+
{
|
50 |
+
class Blogger_Import extends WP_Importer
|
51 |
+
{
|
52 |
+
// 14/2/2013 switched from global defines to class constants
|
53 |
+
// These are used to change the behaviour of the importer from the default.
|
54 |
+
const MAX_RESULTS = 25; // How many records per GData query (int)
|
55 |
+
const MAX_EXECUTION_TIME = 20; // How many seconds to let the script run (int)
|
56 |
+
const STATUS_INTERVAL = 3; // How many seconds between status bar updates (int)
|
57 |
+
const IMPORT_IMG = true; // Should we import the images (boolean)
|
58 |
+
const REMOTE_TIMEOUT = 5; // How many seconds to wait until google responds (int)
|
59 |
+
const LARGE_IMAGE_SIZE = '1024'; // The size of large images downloaded (string)
|
60 |
+
const POST_PINGBACK = 0; // Turn off the post pingback, set to 1 to re-enabled(bool)
|
61 |
+
//N.B. Not all images will be this large and not all will be able to be remapped.
|
62 |
+
|
63 |
+
var $blogs = array();
|
64 |
+
var $connector;
|
65 |
+
|
66 |
+
function Blogger_Import()
|
67 |
+
{
|
68 |
+
if (isset($_GET['import']) && $_GET['import'] == 'blogger')
|
69 |
+
{
|
70 |
+
add_action('admin_print_scripts', array(&$this, 'queue_scripts'));
|
71 |
+
add_action('admin_print_styles', array(&$this, 'queue_style'));
|
72 |
+
add_filter('blogger_importer_congrats', array(&$this, 'congrats_options'), 10, 2);
|
73 |
+
//Looking at the Wordpress importer there appear to be quite a few "filters" there that perhaps should be standardised across all importers.
|
74 |
+
}
|
75 |
+
//Get Data
|
76 |
+
$this->read_options();
|
77 |
+
}
|
78 |
+
|
79 |
+
static function register_importer()
|
80 |
+
{
|
81 |
+
if (!defined('WP_LOAD_IMPORTERS'))
|
82 |
+
return;
|
83 |
+
//Moved in from blogger_importer_init
|
84 |
+
load_plugin_textdomain('blogger-importer', false, dirname(plugin_basename(__file__)) . '/languages');
|
85 |
+
|
86 |
+
$blogger_import = new Blogger_Import();
|
87 |
+
register_importer('blogger', __('Blogger', 'blogger-importer'), __('Import categories, posts, images and comments then maps users from a Blogger blog.', 'blogger-importer'), array($blogger_import,'start'));
|
88 |
+
}
|
89 |
+
|
90 |
+
function queue_scripts($hook)
|
91 |
+
{
|
92 |
+
$interval = self::STATUS_INTERVAL * 1000;
|
93 |
+
wp_enqueue_script('BloggerImporter', plugins_url('/blogger-importer.js', __file__), array('jquery','jquery-ui-progressbar'), '', true);
|
94 |
+
wp_localize_script('BloggerImporter', 'BL_strings', array('ajaxURL' => admin_url('admin-ajax.php'), 'cont' => esc_js(__('Continue', 'blogger-importer')), 'stop' => esc_js(__('Importing...', 'blogger-importer')), 'stopping' => '', 'authors' =>
|
95 |
+
esc_js(__('Set Authors', 'blogger-importer')), 'nothing' => esc_js(__('Nothing was imported. Had you already imported this blog?', 'blogger-importer')), 'loadauth' => esc_js(__('Preparing author mapping form...',
|
96 |
+
'blogger-importer')), 'authhead' => esc_js(__('Final Step: Author Mapping', 'blogger-importer')), 'interval' => $interval));
|
97 |
+
}
|
98 |
+
|
99 |
+
function queue_style()
|
100 |
+
{
|
101 |
+
wp_enqueue_style('BloggerImporter', plugins_url('/blogger-importer.css', __file__));
|
102 |
+
}
|
103 |
+
|
104 |
+
// Shows the welcome screen and the magic auth link.
|
105 |
+
function greet()
|
106 |
+
{
|
107 |
+
$title = __('Import Blogger', 'blogger-importer');
|
108 |
+
$welcome = __('Howdy! This importer allows you to import posts and comments from your Blogger account into your WordPress site.', 'blogger-importer');
|
109 |
+
$prereqs = __('To use this importer, you must have a Google account and an upgraded (New, was Beta) blog hosted on blogspot.com or a custom domain (not FTP).', 'blogger-importer');
|
110 |
+
$stepone = __('The first thing you need to do is tell Blogger to let WordPress access your account. You will be sent back here after providing authorization.', 'blogger-importer');
|
111 |
+
$errormsg = __('Error occurred getting OAuth tokens from Google', 'blogger-importer');
|
112 |
+
|
113 |
+
echo "
|
114 |
+
|
115 |
+
<div class='wrap'>
|
116 |
+
" . screen_icon() . "
|
117 |
+
<h2>$title</h2>
|
118 |
+
<p>$welcome</p><p>$prereqs</p><p>$stepone</p>";
|
119 |
+
|
120 |
+
$isconnected = $this->connector->connect(admin_url('admin-ajax.php?action=BL_auth'));
|
121 |
+
if (!is_wp_error($isconnected))
|
122 |
+
{
|
123 |
+
echo($this->connector->auth_form());
|
124 |
+
} else
|
125 |
+
{
|
126 |
+
|
127 |
+
echo '<p>' . $errormsg . '</p>';
|
128 |
+
echo '<p><pre>
|
129 |
+
' . $isconnected->get_error_message() . '
|
130 |
+
</pre></p>';
|
131 |
+
}
|
132 |
+
}
|
133 |
+
|
134 |
+
|
135 |
+
function uh_oh($title, $message, $info)
|
136 |
+
{
|
137 |
+
echo "<div class='wrap'>";
|
138 |
+
screen_icon();
|
139 |
+
echo "<h2>$title</h2><p>$message</p><pre>$info</pre></div>";
|
140 |
+
}
|
141 |
+
|
142 |
+
/**
|
143 |
+
* Gets the list of blogs from blogger into an array $this->blogs[]
|
144 |
+
*/
|
145 |
+
function get_blogs($iter = 0)
|
146 |
+
{
|
147 |
+
$xml = $this->connector->oauth_get('https://www.blogger.com/feeds/default/blogs');
|
148 |
+
|
149 |
+
// Give it a few retries... apparently this step often flakes out the first time.
|
150 |
+
if (empty($xml))
|
151 |
+
{
|
152 |
+
if ($iter < 3)
|
153 |
+
{
|
154 |
+
return $this->get_blogs($iter + 1);
|
155 |
+
} else
|
156 |
+
{
|
157 |
+
return false;
|
158 |
+
}
|
159 |
+
}
|
160 |
+
|
161 |
+
$feed = new SimplePie();
|
162 |
+
$feed->set_raw_data($xml);
|
163 |
+
$feed->init();
|
164 |
+
|
165 |
+
$i = 0;
|
166 |
+
foreach ($feed->get_items() as $item)
|
167 |
+
{
|
168 |
+
$blog = new Blogger_Importer_Blog();
|
169 |
+
$blog->ID = $i++;
|
170 |
+
//Perhaps these could be passed as parameters to the new blog object or the init defaults call?
|
171 |
+
$blog->title = $item->get_title();
|
172 |
+
$blog->summary = $item->get_description();
|
173 |
+
|
174 |
+
//id is of the form tag:blogger.com,1999:blog-417730729915399755
|
175 |
+
//We need that number from the end
|
176 |
+
$rawid = explode('-', $item->get_id());
|
177 |
+
$blog->id = $rawid[count($rawid) - 1];
|
178 |
+
|
179 |
+
$parts = parse_url($item->get_link(0, 'alternate'));
|
180 |
+
$blog->host = $parts['host'];
|
181 |
+
$blog->gateway = $item->get_link(0, 'edit');
|
182 |
+
$blog->posts_url = $item->get_link(0, 'http://schemas.google.com/g/2005#post');
|
183 |
+
//AGC:20/4/2012 Developers guide suggests that the correct feed is located as follows
|
184 |
+
//See https://developers.google.com/blogger/docs/1.0/developers_guide_php
|
185 |
+
$blog->comments_url = "http://www.blogger.com/feeds/{$blog->id}/comments/default";
|
186 |
+
|
187 |
+
$blog->init_defaults($this->connector->get_total_results($blog->posts_url),$this->connector->get_total_results($blog->comments_url));
|
188 |
+
|
189 |
+
$this->blogs[] = $blog;
|
190 |
+
}
|
191 |
+
|
192 |
+
return true;
|
193 |
+
}
|
194 |
+
|
195 |
+
/**
|
196 |
+
* Shows the list of your bloger blogs as a table
|
197 |
+
* 25/1/2013 AGC: Moved the table rendering to be a wp_list_table
|
198 |
+
*
|
199 |
+
* @See Blogger_Import_List_Table
|
200 |
+
* @Link http://wpengineer.com/2426/wp_list_table-a-step-by-step-guide/
|
201 |
+
*/
|
202 |
+
function show_blogs()
|
203 |
+
{
|
204 |
+
global $wp_importers;
|
205 |
+
$noscript = esc_html__('This feature requires Javascript but it seems to be disabled. Please enable Javascript and then reload this page. Don’t worry, you can turn it back off when you’re done.',
|
206 |
+
'blogger-importer');
|
207 |
+
$title = esc_html__('Import Blogger', 'blogger-importer');
|
208 |
+
$refreshbutton = esc_attr__('Refresh blog list', 'blogger-importer');
|
209 |
+
|
210 |
+
$intro = $wp_importers['blogger'][1];
|
211 |
+
|
212 |
+
$init = '';
|
213 |
+
foreach ($this->blogs as $i => $blog)
|
214 |
+
{
|
215 |
+
$blogtitle = esc_js($blog->title);
|
216 |
+
$init .= "blogs[$i]=new blog($i,'$blogtitle','{$blog->mode}','" . $blog->get_js_status() . '\');';
|
217 |
+
}
|
218 |
+
|
219 |
+
echo screen_icon() . "<h2>" . $title . "</h2><noscript>" . $noscript . "</noscript>";
|
220 |
+
|
221 |
+
echo '<p>' . $intro . '</p>';
|
222 |
+
|
223 |
+
$myListTable = new Blogger_Import_List_Table(array('ajax' => true));
|
224 |
+
$myListTable->prepare_items($this->blogs, $init);
|
225 |
+
echo '<div id="BlogList" class="wrap">';
|
226 |
+
$myListTable->display();
|
227 |
+
echo '</div>';
|
228 |
+
|
229 |
+
//Refresh button
|
230 |
+
echo ("<form method='post' action='?import=blogger'>
|
231 |
+
<p class='submit' style='text-align:left;'>");
|
232 |
+
wp_nonce_field( 'blogger-importer-refresh', 'blogger-importer-refresh-nonce' );
|
233 |
+
printf("<input type='submit' class='button' value='%s' name='refresh' /></p></form>",$refreshbutton);
|
234 |
+
}
|
235 |
+
|
236 |
+
/**
|
237 |
+
* A clean return function for Ajax calls,
|
238 |
+
* http://www.dagondesign.com/articles/wordpress-hook-for-entire-page-using-output-buffering/
|
239 |
+
* 11/3/2014 Simplified this as the output buffer cleaning rarely seemed to work
|
240 |
+
*/
|
241 |
+
|
242 |
+
function ajax_die($data)
|
243 |
+
{
|
244 |
+
header( 'Content-Type: text/plain' );
|
245 |
+
echo $data;
|
246 |
+
exit;
|
247 |
+
}
|
248 |
+
|
249 |
+
//AJAX functions
|
250 |
+
static function ajax_getstatus()
|
251 |
+
{
|
252 |
+
$blogID = $_POST['blogID'];
|
253 |
+
$blog = Blogger_Importer_Blog::read_option($blogID);
|
254 |
+
Blogger_Import::ajax_die($blog->get_js_status());
|
255 |
+
}
|
256 |
+
|
257 |
+
static function ajax_doimport()
|
258 |
+
{
|
259 |
+
$blogID = $_POST['blogID'];
|
260 |
+
$blog = Blogger_Importer_Blog::read_option($blogID);
|
261 |
+
$connector = Blogger_Importer_Connector::read_option();
|
262 |
+
Blogger_Import::ajax_die($blog->import_blog($connector));
|
263 |
+
}
|
264 |
+
|
265 |
+
static function ajax_doauth()
|
266 |
+
{
|
267 |
+
$connector = Blogger_Importer_Connector::read_option();
|
268 |
+
$connector->auth($_GET['token'],$_GET['secret']);
|
269 |
+
wp_redirect(admin_url('admin.php?import=blogger'));
|
270 |
+
}
|
271 |
+
|
272 |
+
function get_author_form($blog)
|
273 |
+
{
|
274 |
+
//Present a form to the users so they can re-map the imported authors to Wordpress users.
|
275 |
+
$blog->get_authors();
|
276 |
+
|
277 |
+
$directions = sprintf(__('All posts were imported with the current user as author. Use this form to move each Blogger user’s posts to a different WordPress user. You may <a href="%s">add users</a> and then return to this page and complete the user mapping. This form may be used as many times as you like until you activate the “Restart” function below.',
|
278 |
+
'blogger-importer'), 'users.php');
|
279 |
+
$heading = __('Author mapping', 'blogger-importer');
|
280 |
+
$blogtitle = "{$blog->title} ({$blog->host})";
|
281 |
+
$mapthis = __('Blogger username', 'blogger-importer');
|
282 |
+
$tothis = __('WordPress login', 'blogger-importer');
|
283 |
+
$submit = esc_js(__('Save Changes', 'blogger-importer'));
|
284 |
+
$rows = '';
|
285 |
+
|
286 |
+
foreach ($blog->authors as $i => $author)
|
287 |
+
$rows .= "<tr><td><label for='authors[$i]'>{$author[0]}</label></td><td><select name='authors[$i]' id='authors[$i]'>" . $this->get_user_options($author[1]) . "</select></td></tr>";
|
288 |
+
|
289 |
+
return "<div class='wrap'>" . screen_icon() . "<h2>$heading</h2><h3>$blogtitle</h3><p>$directions</p><form action='index.php?import=blogger&noheader=true&saveauthors=1' method='post'><input type='hidden' name='blog' value='" .
|
290 |
+
esc_attr($blog->ID) . "' />".wp_nonce_field( 'blogger-importer-saveauthors', 'blogger-importer-saveauthors-nonce',true,false )."<table cellpadding='5'><thead><td>$mapthis</td><td>$tothis</td></thead>$rows<tr><td></td><td class='submit'><input type='submit' class='button authorsubmit' value='$submit' /></td></tr></table></form></div>";
|
291 |
+
}
|
292 |
+
|
293 |
+
function get_user_options($current)
|
294 |
+
{
|
295 |
+
//AGC: 21/10/2013 Simplified function, caching using a global variable not needed as this is not a function that is called frequently.
|
296 |
+
$importer_users = (array )get_users(); //Function: get_users_of_blog() Deprecated in version 3.1. Use get_users() instead.
|
297 |
+
|
298 |
+
$options = '';
|
299 |
+
|
300 |
+
foreach ($importer_users as $user)
|
301 |
+
{
|
302 |
+
$sel = ($user->ID == $current) ? " selected='selected'" : '';
|
303 |
+
$options .= "<option value='$user->ID'$sel>$user->display_name</option>";
|
304 |
+
}
|
305 |
+
|
306 |
+
return $options;
|
307 |
+
}
|
308 |
+
|
309 |
+
function restart()
|
310 |
+
{
|
311 |
+
$this->connector->reset();
|
312 |
+
$options = get_option('blogger_importer');
|
313 |
+
|
314 |
+
delete_option('blogger_importer');
|
315 |
+
foreach ($this->blogs as $i => $blog)
|
316 |
+
{
|
317 |
+
delete_option('blogger_importer_blog_'.$blog->ID);
|
318 |
+
}
|
319 |
+
}
|
320 |
+
|
321 |
+
// Step 9: Congratulate the user
|
322 |
+
function congrats()
|
323 |
+
{
|
324 |
+
echo "<div class='wrap'>";
|
325 |
+
screen_icon();
|
326 |
+
|
327 |
+
echo '<h2>' . __('Congratulations!', 'blogger-importer') . '</h2><p>' . __('Now that you have imported your Blogger blog into WordPress, what are you going to do? Here are some suggestions:',
|
328 |
+
'blogger-importer') . '</p><ul>';
|
329 |
+
|
330 |
+
$congrats = apply_filters('blogger_importer_congrats', '', count($this->blogs));
|
331 |
+
$congrats = $congrats . '<li>' . __('For security, click the link below to reset this importer.', 'blogger-importer') . '</li>';
|
332 |
+
echo $congrats;
|
333 |
+
echo '</ul>';
|
334 |
+
echo '</div>';
|
335 |
+
}
|
336 |
+
|
337 |
+
function congrats_options($optionlist, $blogcount)
|
338 |
+
{
|
339 |
+
//Plugable list of options called by filter 'blogger_importer_congrats'
|
340 |
+
$optionlist = $optionlist . '<li><a href="' . admin_url('edit.php') . '">' . __('Review posts', 'blogger-importer') . '</a></li>';
|
341 |
+
$optionlist = $optionlist . '<li><a href="' . admin_url('edit-tags.php?taxonomy=category') . '">' . __('Review categories', 'blogger-importer') . '</a></li>';
|
342 |
+
$optionlist = $optionlist . '<li><a href="' . admin_url('import.php') . '">' . __('Convert categories to tags', 'blogger-importer') . '</a></li>';
|
343 |
+
$optionlist = $optionlist . '<li><a href="' . admin_url('edit-comments.php') . '">' . __('Review comments', 'blogger-importer') . '</a></li>';
|
344 |
+
$optionlist = $optionlist . '<li><a href="' . admin_url('upload.php') . '">' . __('Review media', 'blogger-importer') . '</a></li>';
|
345 |
+
|
346 |
+
if ($blogcount > 1)
|
347 |
+
{
|
348 |
+
$optionlist = $optionlist . '<li>' . __('In case you haven’t done it already, you can import the posts from your other blogs:', 'blogger-importer');
|
349 |
+
$optionlist = $optionlist . ' <a href="' . admin_url('?import=blogger') . '">' . __('Show blogs', 'blogger-importer') . '</a></li>';
|
350 |
+
}
|
351 |
+
return $optionlist;
|
352 |
+
}
|
353 |
+
|
354 |
+
function read_options()
|
355 |
+
{
|
356 |
+
$options = get_option('blogger_importer');
|
357 |
+
if (is_array($options))
|
358 |
+
foreach ($options as $key => $value)
|
359 |
+
$this->$key = $value;
|
360 |
+
|
361 |
+
$this->connector = Blogger_Importer_Connector::read_option();
|
362 |
+
|
363 |
+
if (count($this->blogs) == 0) {
|
364 |
+
$blog = true;
|
365 |
+
for ($i = 0; $blog ; $i++) {
|
366 |
+
$blog = Blogger_Importer_Blog::read_option($i);
|
367 |
+
if ($blog) {
|
368 |
+
$this->blogs[] = $blog;
|
369 |
+
}
|
370 |
+
}
|
371 |
+
}
|
372 |
+
}
|
373 |
+
|
374 |
+
function save_vars()
|
375 |
+
{
|
376 |
+
//Todo: return false if errors occur
|
377 |
+
$vars = get_object_vars($this);
|
378 |
+
|
379 |
+
if (array_key_exists('blogs',$vars)){
|
380 |
+
unset($vars['blogs']);
|
381 |
+
}
|
382 |
+
if (array_key_exists('connector',$vars)){
|
383 |
+
unset($vars['connector']);
|
384 |
+
}
|
385 |
+
|
386 |
+
//http://core.trac.wordpress.org/ticket/13480
|
387 |
+
//Calling update options multiple times in a page (or ajax call) means that the cache kicks in and does not save to DB
|
388 |
+
update_option('blogger_importer', $vars);
|
389 |
+
|
390 |
+
//How to check for errors here?
|
391 |
+
if (isset($connector)) {
|
392 |
+
$connector->save_vars();
|
393 |
+
}
|
394 |
+
foreach ($this->blogs as $i => $blog) {
|
395 |
+
$blog->save_vars();
|
396 |
+
}
|
397 |
+
return true;
|
398 |
+
}
|
399 |
+
|
400 |
+
|
401 |
+
/**
|
402 |
+
* The start function is what is called when the importer runs
|
403 |
+
* it is used to parse the parameters and select the appropriate
|
404 |
+
* action such as importing a blog
|
405 |
+
* Moved status and import out to separate ajax calls
|
406 |
+
*/
|
407 |
+
function start()
|
408 |
+
{
|
409 |
+
if (isset($_POST['restart'])) {
|
410 |
+
if ( check_admin_referer( 'blogger-importer-clear', 'blogger-importer-clear-nonce' ) ) {
|
411 |
+
$this->restart();
|
412 |
+
wp_redirect('?import=blogger');
|
413 |
+
} else {
|
414 |
+
wp_die('Error');
|
415 |
+
}
|
416 |
+
}
|
417 |
+
if (isset($_POST['refresh'])) {
|
418 |
+
if ( check_admin_referer( 'blogger-importer-refresh', 'blogger-importer-refresh-nonce' ) ) {
|
419 |
+
$this->blogs = array();
|
420 |
+
} else {
|
421 |
+
wp_die('Error');
|
422 |
+
}
|
423 |
+
}
|
424 |
+
if (isset($_REQUEST['blog'])) {
|
425 |
+
$importing_blog = (int)(is_array($_REQUEST['blog']) ? array_shift($keys = array_keys($_REQUEST['blog'])) : $_REQUEST['blog']);
|
426 |
+
$blog = $this->blogs[$importing_blog];
|
427 |
+
|
428 |
+
if (isset($_GET['authors'])) {
|
429 |
+
print ($this->get_author_form($blog));
|
430 |
+
return;
|
431 |
+
}
|
432 |
+
if (isset($_GET['saveauthors'])) {
|
433 |
+
if ( check_admin_referer( 'blogger-importer-saveauthors', 'blogger-importer-saveauthors-nonce' )) {
|
434 |
+
$blog->save_authors();
|
435 |
+
wp_redirect('?import=blogger&congrats=1');
|
436 |
+
}
|
437 |
+
}
|
438 |
+
} elseif (isset($_GET['congrats'])) {
|
439 |
+
$this->congrats();
|
440 |
+
} elseif (isset($this->connector) && $this->connector->isconnected()) {
|
441 |
+
if (empty($this->blogs)) {
|
442 |
+
if (!$this->get_blogs()) {
|
443 |
+
$this->uh_oh(__('Trouble signing in', 'blogger-importer'), __('We were not able to gain access to your account. Try starting over.', 'blogger-importer'), '');
|
444 |
+
}
|
445 |
+
if (empty($this->blogs)) {
|
446 |
+
$this->uh_oh(__('No blogs found', 'blogger-importer'), __('We were able to log in but there were no blogs. Try a different account next time.', 'blogger-importer'), '');
|
447 |
+
}
|
448 |
+
}
|
449 |
+
if (!empty($this->blogs)) {
|
450 |
+
$this->show_blogs();
|
451 |
+
}
|
452 |
+
} else {
|
453 |
+
$this->connector = new Blogger_Importer_Connector();
|
454 |
+
$this->greet();
|
455 |
+
}
|
456 |
+
|
457 |
+
if (isset($this->connector) && $this->connector->isconnected())
|
458 |
+
{
|
459 |
+
$restart = __('Restart', 'blogger-importer');
|
460 |
+
$message = __('We have saved some information about your Blogger account in your WordPress database. Clearing this information will allow you to start over. Restarting will not affect any posts you have already imported. If you attempt to re-import a blog, duplicate posts and comments will be skipped.',
|
461 |
+
'blogger-importer');
|
462 |
+
$submit = esc_attr__('Clear account information', 'blogger-importer');
|
463 |
+
echo "<div class='wrap'><h2>$restart</h2><p>$message</p>";
|
464 |
+
echo "<form method='post' action='?import=blogger&noheader=true'>";
|
465 |
+
wp_nonce_field( 'blogger-importer-clear', 'blogger-importer-clear-nonce' );
|
466 |
+
printf("<p class='submit' style='text-align:left;'><input type='submit' class='button' value='%s' name='restart' /></p></form>",$submit);
|
467 |
+
}
|
468 |
+
$this->save_vars();
|
469 |
+
}
|
470 |
+
|
471 |
+
function _log( $message ) {
|
472 |
+
//Log to file only, we can't log to display as this is a background(ajax) call and there is no display
|
473 |
+
if( WP_DEBUG === true && WP_DEBUG_DISPLAY === false ){
|
474 |
+
if( is_array( $message ) || is_object( $message ) ){
|
475 |
+
error_log( print_r( $message, true ) );
|
476 |
+
} else {
|
477 |
+
error_log( $message );
|
478 |
+
}
|
479 |
+
}
|
480 |
+
}
|
481 |
+
}
|
482 |
+
} // class_exists( 'WP_Importer' )
|
483 |
+
|
484 |
+
add_action('admin_init', array('Blogger_Import','register_importer'));
|
485 |
+
//Ajax calls
|
486 |
+
add_action('wp_ajax_BL_import', array('Blogger_Import','ajax_doimport'));
|
487 |
+
add_action('wp_ajax_BL_status', array('Blogger_Import','ajax_getstatus'));
|
488 |
+
add_action('wp_ajax_BL_auth', array('Blogger_Import','ajax_doauth'));
|
489 |
+
|
490 |
+
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
comment-entry.php
ADDED
@@ -0,0 +1,93 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* @author Andy Clark
|
5 |
+
* @copyright 2013
|
6 |
+
* A data object representing the data to be added into Wordpress
|
7 |
+
* 10/3/2014 Added errorhandling to cope with comments that don't link to a post
|
8 |
+
*/
|
9 |
+
|
10 |
+
if (!class_exists('CommentEntry'))
|
11 |
+
{
|
12 |
+
class CommentEntry {
|
13 |
+
|
14 |
+
var $links = array();
|
15 |
+
var $categories = array();
|
16 |
+
|
17 |
+
function parselinks() {
|
18 |
+
// Drop the #fragment and we have the comment's old post permalink.
|
19 |
+
foreach ($this->links as $link)
|
20 |
+
{
|
21 |
+
if ($link['rel'] == 'alternate')
|
22 |
+
{
|
23 |
+
$parts = parse_url($link['href']);
|
24 |
+
if (isset($parts['fragment'])){
|
25 |
+
$this->old_permalink = $parts['fragment'];
|
26 |
+
}
|
27 |
+
}
|
28 |
+
//Parent post for nested links
|
29 |
+
if ($link['rel'] == 'related')
|
30 |
+
{
|
31 |
+
$parts = parse_url($link['href']);
|
32 |
+
$this->related = $parts['path'];
|
33 |
+
}
|
34 |
+
if ($link['rel'] == 'self')
|
35 |
+
{
|
36 |
+
$parts = parse_url($link['href']);
|
37 |
+
$this->self = $parts['path'];
|
38 |
+
}
|
39 |
+
}
|
40 |
+
}
|
41 |
+
|
42 |
+
function import() {
|
43 |
+
|
44 |
+
$comment_author = $this->author;
|
45 |
+
$comment_author_url = $this->authoruri;
|
46 |
+
$comment_author_email = $this->authoremail;
|
47 |
+
$comment_date = $this->updated;
|
48 |
+
|
49 |
+
$comment_content = $this->content;
|
50 |
+
$comment_post_ID = $this->post_ID;
|
51 |
+
$comment_author_IP = '127.0.0.1'; //Blogger does not supply the IP so default this
|
52 |
+
|
53 |
+
// Clean up content
|
54 |
+
// Simplepie does some cleaning but does not do these.
|
55 |
+
$comment_content = str_replace('<br>', '<br />', $comment_content);
|
56 |
+
$comment_content = str_replace('<hr>', '<hr />', $comment_content);
|
57 |
+
|
58 |
+
$comment_parent = isset($this->parentcommentid) ? $this->parentcommentid : 0;
|
59 |
+
|
60 |
+
$comment = compact('comment_post_ID', 'comment_author', 'comment_author_url', 'comment_author_email','comment_author_IP','comment_date', 'comment_content', 'comment_parent');
|
61 |
+
|
62 |
+
$comment = wp_filter_comment($comment);
|
63 |
+
$comment_id = wp_insert_comment($comment);
|
64 |
+
|
65 |
+
//links of the form /feeds/417730729915399755/8397846992898424746/comments/default/7732208643735403000
|
66 |
+
add_comment_meta($comment_id, 'blogger_internal', $this->self, true);
|
67 |
+
|
68 |
+
return $comment_id;
|
69 |
+
}
|
70 |
+
|
71 |
+
function exists()
|
72 |
+
{
|
73 |
+
//Do we have 2 comments for the same author at the same time, on the same post?
|
74 |
+
//returns comment id
|
75 |
+
return ($this->get_comment_by_oldID($this->self));
|
76 |
+
}
|
77 |
+
|
78 |
+
function get_comment_by_oldID($oldID) {
|
79 |
+
//Check to see if this post has been loaded already
|
80 |
+
//Can we use get_comments for this?
|
81 |
+
global $wpdb;
|
82 |
+
$query = "SELECT c.comment_id FROM $wpdb->commentmeta m inner join $wpdb->comments c on c.comment_ID = m.comment_id where meta_key = 'blogger_internal' and meta_value = '%s' LIMIT 0 , 1";
|
83 |
+
$c = (int) $wpdb->get_var( $wpdb->prepare($query, $oldID) );
|
84 |
+
|
85 |
+
return $c;
|
86 |
+
}
|
87 |
+
|
88 |
+
|
89 |
+
|
90 |
+
}
|
91 |
+
}
|
92 |
+
|
93 |
+
?>
|
languages/blogger-importer-el.mo
ADDED
Binary file
|
languages/blogger-importer-el.po
ADDED
@@ -0,0 +1,371 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Translation of the WordPress plugin Blogger Importer 0.2 by wordpressdotorg.
|
2 |
+
# Copyright (C) 2010 wordpressdotorg
|
3 |
+
# This file is distributed under the same license as the Blogger Importer package.
|
4 |
+
# FIRST AUTHOR <EMAIL@ADDRESS>, 2010.
|
5 |
+
#
|
6 |
+
msgid ""
|
7 |
+
msgstr ""
|
8 |
+
"Project-Id-Version: Blogger Importer 0.2\n"
|
9 |
+
"Report-Msgid-Bugs-To: \n"
|
10 |
+
"POT-Creation-Date: 2013-10-29 12:10-0000\n"
|
11 |
+
"PO-Revision-Date: 2013-10-29 15:45+0200\n"
|
12 |
+
"Last-Translator: Stergatu Lena <stergatu@cti.gr>\n"
|
13 |
+
"Language-Team: \n"
|
14 |
+
"Language: el_GR\n"
|
15 |
+
"MIME-Version: 1.0\n"
|
16 |
+
"Content-Type: text/plain; charset=UTF-8\n"
|
17 |
+
"Content-Transfer-Encoding: 8bit\n"
|
18 |
+
"X-Poedit-SourceCharset: utf-8\n"
|
19 |
+
"X-Generator: Poedit 1.5.7\n"
|
20 |
+
|
21 |
+
#: ../blogger-importer-blog.php:462 ../blogger-importer-blog.php:787
|
22 |
+
msgid "Attempting to write back empty content"
|
23 |
+
msgstr "Προσπάθειας εγγραφής κενού περιεχομένου"
|
24 |
+
|
25 |
+
#: ../blogger-importer-blog.php:502 ../blogger-importer-blog.php:505
|
26 |
+
#: ../blogger-importer-table.php:55 ../blogger-importer-table.php:59
|
27 |
+
#, php-format
|
28 |
+
msgid "%d of %d"
|
29 |
+
msgstr "%d από %d"
|
30 |
+
|
31 |
+
#: ../blogger-importer-blog.php:503
|
32 |
+
#, php-format
|
33 |
+
msgid "%d posts skipped"
|
34 |
+
msgstr "%d άρθρα αγνοήθηκαν"
|
35 |
+
|
36 |
+
#: ../blogger-importer-blog.php:506
|
37 |
+
#, php-format
|
38 |
+
msgid "%d comments skipped"
|
39 |
+
msgstr "%d σχόλια αγνοήθηκαν"
|
40 |
+
|
41 |
+
#: ../blogger-importer-blog.php:508
|
42 |
+
#, php-format
|
43 |
+
msgid "%d images skipped"
|
44 |
+
msgstr "%d εικόνες αγνοήθηκαν"
|
45 |
+
|
46 |
+
#: ../blogger-importer-blog.php:695
|
47 |
+
msgid "Lowrez not an image"
|
48 |
+
msgstr "Lowrez δεν είναι εικόνα"
|
49 |
+
|
50 |
+
#: ../blogger-importer-blog.php:731
|
51 |
+
msgid "None of the images are valid"
|
52 |
+
msgstr "Καμία από τις εικόνες δεν είναι έγκυρη"
|
53 |
+
|
54 |
+
#: ../blogger-importer-table.php:19
|
55 |
+
msgid "Blog Name"
|
56 |
+
msgstr "Όνομα Ιστολογίου"
|
57 |
+
|
58 |
+
#: ../blogger-importer-table.php:20
|
59 |
+
msgid "Blog URL"
|
60 |
+
msgstr "URL Ιστολογίου"
|
61 |
+
|
62 |
+
#: ../blogger-importer-table.php:21
|
63 |
+
msgid "Posts"
|
64 |
+
msgstr "Άρθρα"
|
65 |
+
|
66 |
+
#: ../blogger-importer-table.php:22
|
67 |
+
msgid "Comments"
|
68 |
+
msgstr "Σχόλια"
|
69 |
+
|
70 |
+
#: ../blogger-importer-table.php:23
|
71 |
+
msgid "Images"
|
72 |
+
msgstr "Εικόνες"
|
73 |
+
|
74 |
+
#: ../blogger-importer-table.php:24
|
75 |
+
msgid "Links"
|
76 |
+
msgstr "Σύνδεσμοι"
|
77 |
+
|
78 |
+
#: ../blogger-importer-table.php:25
|
79 |
+
msgid "Action Button"
|
80 |
+
msgstr "Κουμπί ενέργειας"
|
81 |
+
|
82 |
+
#: ../blogger-importer-table.php:71
|
83 |
+
msgid "Import"
|
84 |
+
msgstr "Εισαγωγή"
|
85 |
+
|
86 |
+
#: ../blogger-importer-table.php:72 ../blogger-importer.php:84
|
87 |
+
msgid "Continue"
|
88 |
+
msgstr "Συνέχεια"
|
89 |
+
|
90 |
+
#: ../blogger-importer-table.php:73 ../blogger-importer.php:84
|
91 |
+
msgid "Importing..."
|
92 |
+
msgstr "Εισαγωγή…"
|
93 |
+
|
94 |
+
#: ../blogger-importer-table.php:74 ../blogger-importer.php:85
|
95 |
+
msgid "Set Authors"
|
96 |
+
msgstr "Ορισμός συντακτών"
|
97 |
+
|
98 |
+
#: ../blogger-importer.php:77
|
99 |
+
msgid "Blogger"
|
100 |
+
msgstr "Blogger"
|
101 |
+
|
102 |
+
#: ../blogger-importer.php:77
|
103 |
+
msgid ""
|
104 |
+
"Import categories, posts, images and comments then maps users from a Blogger "
|
105 |
+
"blog."
|
106 |
+
msgstr ""
|
107 |
+
"Εισαγωγή κατηγοριών, άρθρων, εικόνων και σχολίων και στην συνέχεια "
|
108 |
+
"αντιστοίχιση μελών από ιστολόγιο Blogger."
|
109 |
+
|
110 |
+
#: ../blogger-importer.php:85
|
111 |
+
msgid "Nothing was imported. Had you already imported this blog?"
|
112 |
+
msgstr "Δεν εισήχθη τίποτα. Μήπως είχατε εισάγει ήδη το ιστολόγιο;"
|
113 |
+
|
114 |
+
#: ../blogger-importer.php:85
|
115 |
+
msgid "Preparing author mapping form..."
|
116 |
+
msgstr "Προετοιμασία χάρτη συντακτών…"
|
117 |
+
|
118 |
+
#: ../blogger-importer.php:86
|
119 |
+
msgid "Final Step: Author Mapping"
|
120 |
+
msgstr "Τελευταίο βήμα: Χάρτης συντακτών"
|
121 |
+
|
122 |
+
#: ../blogger-importer.php:99
|
123 |
+
msgid "Import Blogger"
|
124 |
+
msgstr "Εισαγωγή Blogger"
|
125 |
+
|
126 |
+
#: ../blogger-importer.php:100
|
127 |
+
msgid ""
|
128 |
+
"Howdy! This importer allows you to import posts and comments from your "
|
129 |
+
"Blogger account into your WordPress site."
|
130 |
+
msgstr ""
|
131 |
+
"Καλώς ήρθατε! Ο εισαγωγέας αυτός είναι για να εισάγετε άρθρα και σχόλια από "
|
132 |
+
"λογαριασμό Blogger σε ένα ιστολόγιο WordPress."
|
133 |
+
|
134 |
+
#: ../blogger-importer.php:101
|
135 |
+
msgid ""
|
136 |
+
"To use this importer, you must have a Google account and an upgraded (New, "
|
137 |
+
"was Beta) blog hosted on blogspot.com or a custom domain (not FTP)."
|
138 |
+
msgstr ""
|
139 |
+
"Για να χρησιμοποιήσετε τον εισαγωγέα, πρέπει να έχετε λογαριασμό Google και "
|
140 |
+
"αναβαθμισμένο ιστολόγιο στο blogspot.com ή σε δικό σας όνομα (όχι FTP)."
|
141 |
+
|
142 |
+
#: ../blogger-importer.php:102
|
143 |
+
msgid ""
|
144 |
+
"The first thing you need to do is tell Blogger to let WordPress access your "
|
145 |
+
"account. You will be sent back here after providing authorization."
|
146 |
+
msgstr ""
|
147 |
+
"Καταρχάς, πρέπει να πείτε στο Blogger να δώσει πρόσβαση στον λογαριασμό του "
|
148 |
+
"WordPress. Αφού δώσετε την έγκριση, θα επιστρέψετε αυτόματα εδώ."
|
149 |
+
|
150 |
+
#: ../blogger-importer.php:104
|
151 |
+
msgid "Error occurred getting OAuth tokens from Google"
|
152 |
+
msgstr "Σφάλμα ανάκτησης OAuth tokens από την Google"
|
153 |
+
|
154 |
+
#: ../blogger-importer.php:142
|
155 |
+
msgid ""
|
156 |
+
"OpenSSL is not installed check your PHP.INI or ask your server provider to "
|
157 |
+
"enable the OpenSSL module."
|
158 |
+
msgstr ""
|
159 |
+
"Το OpenSSL δεν είναι εγκατεστημένο. Ελέξτε το PHP.INI ή ζητήστε από τον "
|
160 |
+
"server provider να ενεργοποιήσει το OpenSSL module."
|
161 |
+
|
162 |
+
#: ../blogger-importer.php:175
|
163 |
+
msgid ""
|
164 |
+
"Invalid Token: Check server date, firewall settings and transports (curl, "
|
165 |
+
"streams and fsockopen)"
|
166 |
+
msgstr ""
|
167 |
+
"Μη έγκυρο Token: Ελέξτε την ημερομηνία του server, τις ρυθμίσεις firewall "
|
168 |
+
"και μεταφοράς δεδομένων (curl, streams and fsockopen)"
|
169 |
+
|
170 |
+
#: ../blogger-importer.php:394
|
171 |
+
#, php-format
|
172 |
+
msgid ""
|
173 |
+
"All posts were imported with the current user as author. Use this form to "
|
174 |
+
"move each Blogger user’s posts to a different WordPress user. You may "
|
175 |
+
"<a href=\"%s\">add users</a> and then return to this page and complete the "
|
176 |
+
"user mapping. This form may be used as many times as you like until you "
|
177 |
+
"activate the “Restart” function below."
|
178 |
+
msgstr ""
|
179 |
+
"Όλα τα άρθρα εισήχθησαν με συντάκτη τον τρέχοντα χρήστη. Χρησιμοποιήστε τη "
|
180 |
+
"φόρμα για να μεταφέρετε τα άρθρα κάθε συντάκτη του Blogger σε διαφορετικό "
|
181 |
+
"συντάκτη του WordPress. Μπορείτε να <a href=\"%s\">προσθέσετε μέλη</a> και "
|
182 |
+
"να επιστρέψετε για να ολοκληρώσετε την αναχαρτογράφηση. Μέχρι να "
|
183 |
+
"ενεργοποιήσετε την «Επανεκκίνηση», γίνεται να χρησιμοποιήσετε τη φόρμα όσες "
|
184 |
+
"φορές θέλετε."
|
185 |
+
|
186 |
+
#: ../blogger-importer.php:396
|
187 |
+
msgid "Author mapping"
|
188 |
+
msgstr "Χάρτης συντακτών"
|
189 |
+
|
190 |
+
#: ../blogger-importer.php:398
|
191 |
+
msgid "Blogger username"
|
192 |
+
msgstr "Όνομα χρήστη Blogger"
|
193 |
+
|
194 |
+
#: ../blogger-importer.php:399
|
195 |
+
msgid "WordPress login"
|
196 |
+
msgstr "Σύνδεση WordPress"
|
197 |
+
|
198 |
+
#: ../blogger-importer.php:400
|
199 |
+
msgid "Save Changes"
|
200 |
+
msgstr "Αποθήκευση αλλαγών"
|
201 |
+
|
202 |
+
#: ../blogger-importer.php:447
|
203 |
+
msgid "Congratulations!"
|
204 |
+
msgstr "Συγχαρητήρια!"
|
205 |
+
|
206 |
+
#: ../blogger-importer.php:447
|
207 |
+
msgid ""
|
208 |
+
"Now that you have imported your Blogger blog into WordPress, what are you "
|
209 |
+
"going to do? Here are some suggestions:"
|
210 |
+
msgstr ""
|
211 |
+
"Η εισαγωγή του ιστολογίου Blogger στο WordPress τελείωσε. Τι θα κάνετε τώρα; "
|
212 |
+
"Να μερικές ιδέες:"
|
213 |
+
|
214 |
+
#: ../blogger-importer.php:451
|
215 |
+
msgid "For security, click the link below to reset this importer."
|
216 |
+
msgstr "Για λόγους ασφαλείας, κάντε κλικ πιο κάτω να μηδενίσετε τον εισαγωγέα."
|
217 |
+
|
218 |
+
#: ../blogger-importer.php:460
|
219 |
+
msgid "Review posts"
|
220 |
+
msgstr "Ανασκόπηση άρθρων"
|
221 |
+
|
222 |
+
#: ../blogger-importer.php:461
|
223 |
+
msgid "Review categories"
|
224 |
+
msgstr "Ανασκόπηση ετικετών"
|
225 |
+
|
226 |
+
#: ../blogger-importer.php:462
|
227 |
+
msgid "Convert categories to tags"
|
228 |
+
msgstr "Μετατροπή κατηγοριών σε ετικέτες"
|
229 |
+
|
230 |
+
#: ../blogger-importer.php:463
|
231 |
+
msgid "Review comments"
|
232 |
+
msgstr "Ανασκόπηση σχολίων"
|
233 |
+
|
234 |
+
#: ../blogger-importer.php:464
|
235 |
+
msgid "Review media"
|
236 |
+
msgstr "Ανασκόπηση πολυμέσων"
|
237 |
+
|
238 |
+
#: ../blogger-importer.php:468
|
239 |
+
msgid ""
|
240 |
+
"In case you haven’t done it already, you can import the posts from "
|
241 |
+
"your other blogs:"
|
242 |
+
msgstr ""
|
243 |
+
"Αν δεν το έχετε κάνει ήδη, μπορείτε να εισάγετε άρθρα από άλλα ιστολόγια σας:"
|
244 |
+
|
245 |
+
#: ../blogger-importer.php:469
|
246 |
+
msgid "Show blogs"
|
247 |
+
msgstr "Προβολή ιστολογίων"
|
248 |
+
|
249 |
+
#: ../blogger-importer.php:554
|
250 |
+
msgid "Trouble signing in"
|
251 |
+
msgstr "Πρόβλημα στη σύνδεση"
|
252 |
+
|
253 |
+
#: ../blogger-importer.php:554
|
254 |
+
msgid "We were not able to gain access to your account. Try starting over."
|
255 |
+
msgstr ""
|
256 |
+
"Δεν ήταν δυνατή η πρόσβαση στον λογαριασμό σας. Δοκιμάστε άλλη μία φορά."
|
257 |
+
|
258 |
+
#: ../blogger-importer.php:557
|
259 |
+
msgid "No blogs found"
|
260 |
+
msgstr "Δεν βρέθηκαν ιστολόγια"
|
261 |
+
|
262 |
+
#: ../blogger-importer.php:557
|
263 |
+
msgid ""
|
264 |
+
"We were able to log in but there were no blogs. Try a different account next "
|
265 |
+
"time."
|
266 |
+
msgstr ""
|
267 |
+
"Η είσοδος έγινε κανονικά αλλά δεν βρέθηκε ιστολόγιο. Δοκιμάστε έναν άλλο "
|
268 |
+
"λογαριασμό."
|
269 |
+
|
270 |
+
#: ../blogger-importer.php:568
|
271 |
+
msgid "Restart"
|
272 |
+
msgstr "Επανεκκίνηση"
|
273 |
+
|
274 |
+
#: ../blogger-importer.php:569
|
275 |
+
msgid ""
|
276 |
+
"We have saved some information about your Blogger account in your WordPress "
|
277 |
+
"database. Clearing this information will allow you to start over. Restarting "
|
278 |
+
"will not affect any posts you have already imported. If you attempt to re-"
|
279 |
+
"import a blog, duplicate posts and comments will be skipped."
|
280 |
+
msgstr ""
|
281 |
+
"Αποθηκεύσαμε στη βάση δεδομένων του WordPress κάποια στοιχεία από τον "
|
282 |
+
"λογαριασμό σας στο Blogger. Η αφαίρεση αυτών των στοιχείων σας επιτρέπει να "
|
283 |
+
"ξεκινήσετε από την αρχή. Άρθρα που έχουν ήδη εισαχθεί δεν επηρεάζονται. Αν "
|
284 |
+
"επιχειρήσετε να επανεισάγετε ένα ιστολόγιο, τα ήδη εισηγμένα άρθρα και "
|
285 |
+
"σχόλια δεν επανεισάγονται."
|
286 |
+
|
287 |
+
#~ msgid "Authorize"
|
288 |
+
#~ msgstr "Εξουσιοδότηση"
|
289 |
+
|
290 |
+
#~ msgid "Blogger Blogs"
|
291 |
+
#~ msgstr "Ιστολόγια Blogger"
|
292 |
+
|
293 |
+
#~ msgid "The Magic Button"
|
294 |
+
#~ msgstr "Το Μαγικό Κουμπί"
|
295 |
+
|
296 |
+
#~ msgid ""
|
297 |
+
#~ "This feature requires Javascript but it seems to be disabled. Please "
|
298 |
+
#~ "enable Javascript and then reload this page. Don’t worry, you can "
|
299 |
+
#~ "turn it back off when you’re done."
|
300 |
+
#~ msgstr ""
|
301 |
+
#~ "Η λειτουργία αυτή απαιτεί JavaScript, που φαίνεται ότι είναι "
|
302 |
+
#~ "απενεργοποιημένη. Παρακαλούμε ενεργοποιήστε την JavaScript και ανανεώστε "
|
303 |
+
#~ "τη σελίδα. (Μόλις τελειώσετε, μπορείτε να απενεργοποιήσετε πάλι την "
|
304 |
+
#~ "JavaScript.)"
|
305 |
+
|
306 |
+
#~ msgid "That was hard work! Take a break."
|
307 |
+
#~ msgstr "Ήταν κουραστική εργασία! Κάντε ένα διάλειμμα!"
|
308 |
+
|
309 |
+
#~ msgid ""
|
310 |
+
#~ "Go to <a href=\"%s\" target=\"%s\">Authors & Users</a>, where you can "
|
311 |
+
#~ "modify the new user(s) or delete them. If you want to make all of the "
|
312 |
+
#~ "imported posts yours, you will be given that option when you delete the "
|
313 |
+
#~ "new authors."
|
314 |
+
#~ msgstr ""
|
315 |
+
#~ "Στο <a href=\"%s\" target=\"%s\">Συντάκτες & Μέλη</a> μπορείτε να "
|
316 |
+
#~ "επεξεργαστείτε νέα μέλη ή να τα διαγράψετε. Αν θέλετε να μεταφέρετε στον "
|
317 |
+
#~ "δικό σας λογαριασμό όλα τα εισηγμένα άρθρα, σας δίνεται η επιλογή όταν "
|
318 |
+
#~ "διαγράφετε τους νέους συντάκτες."
|
319 |
+
|
320 |
+
#~ msgid "Clear account information"
|
321 |
+
#~ msgstr "Καθαρισμός πληροφοριών λογαριασμού"
|
322 |
+
|
323 |
+
#~ msgid "Blogger Importer"
|
324 |
+
#~ msgstr "Εισαγωγέας Blogger"
|
325 |
+
|
326 |
+
#~ msgid "http://wordpress.org/extend/plugins/blogger-importer/"
|
327 |
+
#~ msgstr "http://wordpress.org/extend/plugins/blogger-importer/"
|
328 |
+
|
329 |
+
#~ msgid ""
|
330 |
+
#~ "Import posts, comments and tags from a Blogger blog and migrate authors "
|
331 |
+
#~ "to WordPress users."
|
332 |
+
#~ msgstr ""
|
333 |
+
#~ "Εισαγωγή άρθρων, σχολίων και ετικετών από ιστολόγιο Blogger και στην "
|
334 |
+
#~ "συνέχεια αντιστοίχιση συντακτών με μέλη του ιστολογίου σας."
|
335 |
+
|
336 |
+
#~ msgid "wordpressdotorg"
|
337 |
+
#~ msgstr "wordpressdotorg"
|
338 |
+
|
339 |
+
#~ msgid "http://wordpress.org/"
|
340 |
+
#~ msgstr "http://wordpress.org/"
|
341 |
+
|
342 |
+
#~ msgid "Authorization failed"
|
343 |
+
#~ msgstr "Εξουσιοδότηση απέτυχε"
|
344 |
+
|
345 |
+
#~ msgid ""
|
346 |
+
#~ "Something went wrong. If the problem persists, send this info to support:"
|
347 |
+
#~ msgstr ""
|
348 |
+
#~ "Κάτι δεν πήγε καλά. Αν επιμείνει το πρόβλημα, στείλτε αυτά τα στοιχεία "
|
349 |
+
#~ "στην υποστήριξη:"
|
350 |
+
|
351 |
+
#~ msgid "Could not connect to https://www.google.com"
|
352 |
+
#~ msgstr "Αδύνατη η σύνδεση με https://www.google.com"
|
353 |
+
|
354 |
+
#~ msgid ""
|
355 |
+
#~ "There was a problem opening a secure connection to Google. This is what "
|
356 |
+
#~ "went wrong:"
|
357 |
+
#~ msgstr ""
|
358 |
+
#~ "Δεν ήταν δυνατή η ασφαλής σύνδεση με το Google. Να τι δεν πήγε καλά:"
|
359 |
+
|
360 |
+
#~ msgid "Could not connect to %s"
|
361 |
+
#~ msgstr "Αδύνατη η σύνδεση με %s"
|
362 |
+
|
363 |
+
#~ msgid ""
|
364 |
+
#~ "There was a problem opening a connection to Blogger. This is what went "
|
365 |
+
#~ "wrong:"
|
366 |
+
#~ msgstr "Δεν ήταν δυνατή η σύνδεση με το Blogger. Να τι δεν πήγε καλά:"
|
367 |
+
|
368 |
+
#~ msgid "Import posts, comments, tags, and attachments from a Blogger blog."
|
369 |
+
#~ msgstr ""
|
370 |
+
#~ "Εγκαταστήστε τον εισαγωγέα Blogger για να εισάγετε άρθρα, σχόλια και μέλη "
|
371 |
+
#~ "ιστολόγιο Blogger."
|
languages/blogger-importer.pot
CHANGED
@@ -1,220 +1,292 @@
|
|
1 |
-
# Copyright (C)
|
2 |
-
# This file is distributed under the same license as the Blogger Importer package.
|
3 |
-
msgid ""
|
4 |
-
msgstr ""
|
5 |
-
"Project-Id-Version: Blogger Importer 0.
|
6 |
-
"Report-Msgid-Bugs-To: http://wordpress.org/
|
7 |
-
"POT-Creation-Date:
|
8 |
-
"MIME-Version: 1.0\n"
|
9 |
-
"Content-Type: text/plain; charset=UTF-8\n"
|
10 |
-
"Content-Transfer-Encoding: 8bit\n"
|
11 |
-
"PO-Revision-Date:
|
12 |
-
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
13 |
-
"Language-Team: LANGUAGE <LL@li.org>\n"
|
14 |
-
|
15 |
-
#: blogger-importer.php:
|
16 |
-
msgid "
|
17 |
-
msgstr ""
|
18 |
-
|
19 |
-
#: blogger-importer.php:
|
20 |
-
|
21 |
-
"
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
msgstr ""
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
|
45 |
-
|
46 |
-
|
47 |
-
|
48 |
-
|
49 |
-
|
50 |
-
|
51 |
-
|
52 |
-
|
53 |
-
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
60 |
-
|
61 |
-
|
62 |
-
|
63 |
-
|
64 |
-
|
65 |
-
|
66 |
-
|
67 |
-
|
68 |
-
|
69 |
-
|
70 |
-
|
71 |
-
|
72 |
-
|
73 |
-
|
74 |
-
|
75 |
-
|
76 |
-
|
77 |
-
|
78 |
-
|
79 |
-
|
80 |
-
|
81 |
-
|
82 |
-
|
83 |
-
|
84 |
-
|
85 |
-
|
86 |
-
|
87 |
-
|
88 |
-
|
89 |
-
|
90 |
-
|
91 |
-
|
92 |
-
|
93 |
-
|
94 |
-
|
95 |
-
|
96 |
-
|
97 |
-
|
98 |
-
|
99 |
-
|
100 |
-
|
101 |
-
|
102 |
-
|
103 |
-
|
104 |
-
|
105 |
-
|
106 |
-
|
107 |
-
|
108 |
-
|
109 |
-
|
110 |
-
|
111 |
-
|
112 |
-
|
113 |
-
|
114 |
-
|
115 |
-
|
116 |
-
|
117 |
-
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
"
|
122 |
-
|
123 |
-
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
|
136 |
-
|
137 |
-
|
138 |
-
|
139 |
-
|
140 |
-
|
141 |
-
|
142 |
-
|
143 |
-
|
144 |
-
|
145 |
-
|
146 |
-
|
147 |
-
|
148 |
-
|
149 |
-
|
150 |
-
"
|
151 |
-
|
152 |
-
|
153 |
-
|
154 |
-
|
155 |
-
|
156 |
-
|
157 |
-
|
158 |
-
|
159 |
-
|
160 |
-
"
|
161 |
-
|
162 |
-
|
163 |
-
|
164 |
-
|
165 |
-
|
166 |
-
"
|
167 |
-
|
168 |
-
|
169 |
-
|
170 |
-
|
171 |
-
|
172 |
-
|
173 |
-
|
174 |
-
|
175 |
-
|
176 |
-
|
177 |
-
|
178 |
-
|
179 |
-
|
180 |
-
|
181 |
-
|
182 |
-
|
183 |
-
"
|
184 |
-
|
185 |
-
|
186 |
-
|
187 |
-
|
188 |
-
|
189 |
-
|
190 |
-
|
191 |
-
|
192 |
-
|
193 |
-
|
194 |
-
|
195 |
-
|
196 |
-
|
197 |
-
|
198 |
-
|
199 |
-
|
200 |
-
|
201 |
-
|
202 |
-
msgstr ""
|
203 |
-
|
204 |
-
|
205 |
-
msgid "
|
206 |
-
msgstr ""
|
207 |
-
|
208 |
-
|
209 |
-
msgid ""
|
210 |
-
|
211 |
-
|
212 |
-
|
213 |
-
|
214 |
-
|
215 |
-
|
216 |
-
|
217 |
-
|
218 |
-
|
219 |
-
|
220 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
# Copyright (C) 2014 Blogger Importer
|
2 |
+
# This file is distributed under the same license as the Blogger Importer package.
|
3 |
+
msgid ""
|
4 |
+
msgstr ""
|
5 |
+
"Project-Id-Version: Blogger Importer 0.8\n"
|
6 |
+
"Report-Msgid-Bugs-To: http://wordpress.org/support/plugin/blogger-importer\n"
|
7 |
+
"POT-Creation-Date: 2014-12-20 19:44:23+00:00\n"
|
8 |
+
"MIME-Version: 1.0\n"
|
9 |
+
"Content-Type: text/plain; charset=UTF-8\n"
|
10 |
+
"Content-Transfer-Encoding: 8bit\n"
|
11 |
+
"PO-Revision-Date: 2014-MO-DA HO:MI+ZONE\n"
|
12 |
+
"Last-Translator: FULL NAME <EMAIL@ADDRESS>\n"
|
13 |
+
"Language-Team: LANGUAGE <LL@li.org>\n"
|
14 |
+
|
15 |
+
#: blogger-importer-blog.php:447 blogger-importer-blog.php:768
|
16 |
+
msgid "Attempting to write back empty content"
|
17 |
+
msgstr ""
|
18 |
+
|
19 |
+
#: blogger-importer-blog.php:487 blogger-importer-blog.php:490
|
20 |
+
#: blogger-importer-table.php:55 blogger-importer-table.php:59
|
21 |
+
msgid "%d of %d"
|
22 |
+
msgstr ""
|
23 |
+
|
24 |
+
#: blogger-importer-blog.php:488
|
25 |
+
msgid "%d post skipped"
|
26 |
+
msgid_plural "%d posts skipped"
|
27 |
+
msgstr[0] ""
|
28 |
+
msgstr[1] ""
|
29 |
+
|
30 |
+
#: blogger-importer-blog.php:491
|
31 |
+
msgid "%d comment skipped"
|
32 |
+
msgid_plural "%d comments skipped"
|
33 |
+
msgstr[0] ""
|
34 |
+
msgstr[1] ""
|
35 |
+
|
36 |
+
#: blogger-importer-blog.php:493
|
37 |
+
msgid "%d image skipped"
|
38 |
+
msgid_plural "%d images skipped"
|
39 |
+
msgstr[0] ""
|
40 |
+
msgstr[1] ""
|
41 |
+
|
42 |
+
#: blogger-importer-blog.php:676
|
43 |
+
msgid "Lowrez not an image"
|
44 |
+
msgstr ""
|
45 |
+
|
46 |
+
#: blogger-importer-blog.php:712
|
47 |
+
msgid "None of the images are valid"
|
48 |
+
msgstr ""
|
49 |
+
|
50 |
+
#: blogger-importer-connector.php:32
|
51 |
+
msgid ""
|
52 |
+
"OpenSSL is not installed check your PHP.INI or ask your server provider to "
|
53 |
+
"enable the OpenSSL module."
|
54 |
+
msgstr ""
|
55 |
+
|
56 |
+
#: blogger-importer-connector.php:65
|
57 |
+
msgid ""
|
58 |
+
"Invalid Token: Check server date, firewall settings and transports (curl, "
|
59 |
+
"streams and fsockopen)"
|
60 |
+
msgstr ""
|
61 |
+
|
62 |
+
#: blogger-importer-connector.php:80
|
63 |
+
msgid "Authorize"
|
64 |
+
msgstr ""
|
65 |
+
|
66 |
+
#: blogger-importer-table.php:19
|
67 |
+
msgid "Blog Name"
|
68 |
+
msgstr ""
|
69 |
+
|
70 |
+
#: blogger-importer-table.php:20
|
71 |
+
msgid "Blog URL"
|
72 |
+
msgstr ""
|
73 |
+
|
74 |
+
#: blogger-importer-table.php:21
|
75 |
+
msgid "Posts"
|
76 |
+
msgstr ""
|
77 |
+
|
78 |
+
#: blogger-importer-table.php:22
|
79 |
+
msgid "Comments"
|
80 |
+
msgstr ""
|
81 |
+
|
82 |
+
#: blogger-importer-table.php:23
|
83 |
+
msgid "Images"
|
84 |
+
msgstr ""
|
85 |
+
|
86 |
+
#: blogger-importer-table.php:24
|
87 |
+
msgid "Links"
|
88 |
+
msgstr ""
|
89 |
+
|
90 |
+
#: blogger-importer-table.php:25
|
91 |
+
msgid "Action Button"
|
92 |
+
msgstr ""
|
93 |
+
|
94 |
+
#: blogger-importer-table.php:71
|
95 |
+
msgid "Import"
|
96 |
+
msgstr ""
|
97 |
+
|
98 |
+
#: blogger-importer-table.php:72 blogger-importer.php:94
|
99 |
+
msgid "Continue"
|
100 |
+
msgstr ""
|
101 |
+
|
102 |
+
#: blogger-importer-table.php:73 blogger-importer.php:94
|
103 |
+
msgid "Importing..."
|
104 |
+
msgstr ""
|
105 |
+
|
106 |
+
#: blogger-importer-table.php:74 blogger-importer.php:95
|
107 |
+
msgid "Set Authors"
|
108 |
+
msgstr ""
|
109 |
+
|
110 |
+
#: blogger-importer.php:87
|
111 |
+
msgid "Blogger"
|
112 |
+
msgstr ""
|
113 |
+
|
114 |
+
#: blogger-importer.php:87
|
115 |
+
msgid ""
|
116 |
+
"Import categories, posts, images and comments then maps users from a Blogger "
|
117 |
+
"blog."
|
118 |
+
msgstr ""
|
119 |
+
|
120 |
+
#: blogger-importer.php:95
|
121 |
+
msgid "Nothing was imported. Had you already imported this blog?"
|
122 |
+
msgstr ""
|
123 |
+
|
124 |
+
#: blogger-importer.php:95
|
125 |
+
msgid "Preparing author mapping form..."
|
126 |
+
msgstr ""
|
127 |
+
|
128 |
+
#: blogger-importer.php:96
|
129 |
+
msgid "Final Step: Author Mapping"
|
130 |
+
msgstr ""
|
131 |
+
|
132 |
+
#: blogger-importer.php:107 blogger-importer.php:207
|
133 |
+
msgid "Import Blogger"
|
134 |
+
msgstr ""
|
135 |
+
|
136 |
+
#: blogger-importer.php:108
|
137 |
+
msgid ""
|
138 |
+
"Howdy! This importer allows you to import posts and comments from your "
|
139 |
+
"Blogger account into your WordPress site."
|
140 |
+
msgstr ""
|
141 |
+
|
142 |
+
#: blogger-importer.php:109
|
143 |
+
msgid ""
|
144 |
+
"To use this importer, you must have a Google account and an upgraded (New, "
|
145 |
+
"was Beta) blog hosted on blogspot.com or a custom domain (not FTP)."
|
146 |
+
msgstr ""
|
147 |
+
|
148 |
+
#: blogger-importer.php:110
|
149 |
+
msgid ""
|
150 |
+
"The first thing you need to do is tell Blogger to let WordPress access your "
|
151 |
+
"account. You will be sent back here after providing authorization."
|
152 |
+
msgstr ""
|
153 |
+
|
154 |
+
#: blogger-importer.php:111
|
155 |
+
msgid "Error occurred getting OAuth tokens from Google"
|
156 |
+
msgstr ""
|
157 |
+
|
158 |
+
#: blogger-importer.php:205
|
159 |
+
msgid ""
|
160 |
+
"This feature requires Javascript but it seems to be disabled. Please enable "
|
161 |
+
"Javascript and then reload this page. Don’t worry, you can turn it "
|
162 |
+
"back off when you’re done."
|
163 |
+
msgstr ""
|
164 |
+
|
165 |
+
#: blogger-importer.php:208
|
166 |
+
msgid "Refresh blog list"
|
167 |
+
msgstr ""
|
168 |
+
|
169 |
+
#: blogger-importer.php:277
|
170 |
+
msgid ""
|
171 |
+
"All posts were imported with the current user as author. Use this form to "
|
172 |
+
"move each Blogger user’s posts to a different WordPress user. You may "
|
173 |
+
"<a href=\"%s\">add users</a> and then return to this page and complete the "
|
174 |
+
"user mapping. This form may be used as many times as you like until you "
|
175 |
+
"activate the “Restart” function below."
|
176 |
+
msgstr ""
|
177 |
+
|
178 |
+
#: blogger-importer.php:279
|
179 |
+
msgid "Author mapping"
|
180 |
+
msgstr ""
|
181 |
+
|
182 |
+
#: blogger-importer.php:281
|
183 |
+
msgid "Blogger username"
|
184 |
+
msgstr ""
|
185 |
+
|
186 |
+
#: blogger-importer.php:282
|
187 |
+
msgid "WordPress login"
|
188 |
+
msgstr ""
|
189 |
+
|
190 |
+
#: blogger-importer.php:283
|
191 |
+
msgid "Save Changes"
|
192 |
+
msgstr ""
|
193 |
+
|
194 |
+
#: blogger-importer.php:327
|
195 |
+
msgid "Congratulations!"
|
196 |
+
msgstr ""
|
197 |
+
|
198 |
+
#: blogger-importer.php:327
|
199 |
+
msgid ""
|
200 |
+
"Now that you have imported your Blogger blog into WordPress, what are you "
|
201 |
+
"going to do? Here are some suggestions:"
|
202 |
+
msgstr ""
|
203 |
+
|
204 |
+
#: blogger-importer.php:331
|
205 |
+
msgid "For security, click the link below to reset this importer."
|
206 |
+
msgstr ""
|
207 |
+
|
208 |
+
#: blogger-importer.php:340
|
209 |
+
msgid "Review posts"
|
210 |
+
msgstr ""
|
211 |
+
|
212 |
+
#: blogger-importer.php:341
|
213 |
+
msgid "Review categories"
|
214 |
+
msgstr ""
|
215 |
+
|
216 |
+
#: blogger-importer.php:342
|
217 |
+
msgid "Convert categories to tags"
|
218 |
+
msgstr ""
|
219 |
+
|
220 |
+
#: blogger-importer.php:343
|
221 |
+
msgid "Review comments"
|
222 |
+
msgstr ""
|
223 |
+
|
224 |
+
#: blogger-importer.php:344
|
225 |
+
msgid "Review media"
|
226 |
+
msgstr ""
|
227 |
+
|
228 |
+
#: blogger-importer.php:348
|
229 |
+
msgid ""
|
230 |
+
"In case you haven’t done it already, you can import the posts from "
|
231 |
+
"your other blogs:"
|
232 |
+
msgstr ""
|
233 |
+
|
234 |
+
#: blogger-importer.php:349
|
235 |
+
msgid "Show blogs"
|
236 |
+
msgstr ""
|
237 |
+
|
238 |
+
#: blogger-importer.php:443
|
239 |
+
msgid "Trouble signing in"
|
240 |
+
msgstr ""
|
241 |
+
|
242 |
+
#: blogger-importer.php:443
|
243 |
+
msgid "We were not able to gain access to your account. Try starting over."
|
244 |
+
msgstr ""
|
245 |
+
|
246 |
+
#: blogger-importer.php:446
|
247 |
+
msgid "No blogs found"
|
248 |
+
msgstr ""
|
249 |
+
|
250 |
+
#: blogger-importer.php:446
|
251 |
+
msgid ""
|
252 |
+
"We were able to log in but there were no blogs. Try a different account next "
|
253 |
+
"time."
|
254 |
+
msgstr ""
|
255 |
+
|
256 |
+
#: blogger-importer.php:459
|
257 |
+
msgid "Restart"
|
258 |
+
msgstr ""
|
259 |
+
|
260 |
+
#: blogger-importer.php:460
|
261 |
+
msgid ""
|
262 |
+
"We have saved some information about your Blogger account in your WordPress "
|
263 |
+
"database. Clearing this information will allow you to start over. Restarting "
|
264 |
+
"will not affect any posts you have already imported. If you attempt to re-"
|
265 |
+
"import a blog, duplicate posts and comments will be skipped."
|
266 |
+
msgstr ""
|
267 |
+
|
268 |
+
#: blogger-importer.php:462
|
269 |
+
msgid "Clear account information"
|
270 |
+
msgstr ""
|
271 |
+
|
272 |
+
#. Plugin Name of the plugin/theme
|
273 |
+
msgid "Blogger Importer"
|
274 |
+
msgstr ""
|
275 |
+
|
276 |
+
#. Plugin URI of the plugin/theme
|
277 |
+
msgid "http://wordpress.org/extend/plugins/blogger-importer/"
|
278 |
+
msgstr ""
|
279 |
+
|
280 |
+
#. Description of the plugin/theme
|
281 |
+
msgid ""
|
282 |
+
"Imports posts, comments, images and tags from a Blogger blog then migrates "
|
283 |
+
"authors to WordPress users."
|
284 |
+
msgstr ""
|
285 |
+
|
286 |
+
#. Author of the plugin/theme
|
287 |
+
msgid "wordpressdotorg"
|
288 |
+
msgstr ""
|
289 |
+
|
290 |
+
#. Author URI of the plugin/theme
|
291 |
+
msgid "http://wordpress.org/"
|
292 |
+
msgstr ""
|
oauth.php
CHANGED
@@ -1,926 +1,930 @@
|
|
1 |
-
<?php
|
2 |
-
/*
|
3 |
-
Original File from: http://oauth.googlecode.com/svn/code/php/
|
4 |
-
License: MIT License:
|
5 |
-
|
6 |
-
The MIT License
|
7 |
-
|
8 |
-
Copyright (c) 2007 Andy Smith
|
9 |
-
|
10 |
-
Permission is hereby granted, free of charge, to any person obtaining a copy
|
11 |
-
of this software and associated documentation files (the "Software"), to deal
|
12 |
-
in the Software without restriction, including without limitation the rights
|
13 |
-
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
14 |
-
copies of the Software, and to permit persons to whom the Software is
|
15 |
-
furnished to do so, subject to the following conditions:
|
16 |
-
|
17 |
-
The above copyright notice and this permission notice shall be included in
|
18 |
-
all copies or substantial portions of the Software.
|
19 |
-
|
20 |
-
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
21 |
-
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
22 |
-
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
23 |
-
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
24 |
-
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
25 |
-
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
26 |
-
THE SOFTWARE.
|
27 |
-
|
28 |
-
|
29 |
-
Modified for use in WordPress by Otto
|
30 |
-
|
31 |
-
Changes: All classes renamed to Blogger_* to prevent conflicts with other plugins/code using the same OAuth library.
|
32 |
-
|
33 |
-
*/
|
34 |
-
// vim: foldmethod=marker
|
35 |
-
|
36 |
-
/* Generic exception class
|
37 |
-
*/
|
38 |
-
class Blogger_OAuthException extends Exception {
|
39 |
-
// pass
|
40 |
-
}
|
41 |
-
|
42 |
-
class Blogger_OAuthConsumer {
|
43 |
-
public $key;
|
44 |
-
public $secret;
|
45 |
-
|
46 |
-
function __construct($key, $secret, $callback_url=NULL) {
|
47 |
-
$this->key = $key;
|
48 |
-
$this->secret = $secret;
|
49 |
-
$this->callback_url = $callback_url;
|
50 |
-
}
|
51 |
-
|
52 |
-
function __toString() {
|
53 |
-
return "Blogger_OAuthConsumer[key=$this->key,secret=$this->secret]";
|
54 |
-
}
|
55 |
-
}
|
56 |
-
|
57 |
-
class Blogger_OAuthToken {
|
58 |
-
// access tokens and request tokens
|
59 |
-
public $key;
|
60 |
-
public $secret;
|
61 |
-
|
62 |
-
/**
|
63 |
-
* key = the token
|
64 |
-
* secret = the token secret
|
65 |
-
*/
|
66 |
-
function __construct($key, $secret) {
|
67 |
-
$this->key = $key;
|
68 |
-
$this->secret = $secret;
|
69 |
-
}
|
70 |
-
|
71 |
-
/**
|
72 |
-
* generates the basic string serialization of a token that a server
|
73 |
-
* would respond to request_token and access_token calls with
|
74 |
-
*/
|
75 |
-
function to_string() {
|
76 |
-
return "oauth_token=" .
|
77 |
-
Blogger_OAuthUtil::urlencode_rfc3986($this->key) .
|
78 |
-
"&oauth_token_secret=" .
|
79 |
-
Blogger_OAuthUtil::urlencode_rfc3986($this->secret);
|
80 |
-
}
|
81 |
-
|
82 |
-
function __toString() {
|
83 |
-
return $this->to_string();
|
84 |
-
}
|
85 |
-
}
|
86 |
-
|
87 |
-
/**
|
88 |
-
* A class for implementing a Signature Method
|
89 |
-
* See section 9 ("Signing Requests") in the spec
|
90 |
-
*/
|
91 |
-
abstract class Blogger_OAuthSignatureMethod {
|
92 |
-
/**
|
93 |
-
* Needs to return the name of the Signature Method (ie HMAC-SHA1)
|
94 |
-
* @return string
|
95 |
-
*/
|
96 |
-
abstract public function get_name();
|
97 |
-
|
98 |
-
/**
|
99 |
-
* Build up the signature
|
100 |
-
* NOTE: The output of this function MUST NOT be urlencoded.
|
101 |
-
* the encoding is handled in Blogger_OAuthRequest when the final
|
102 |
-
* request is serialized
|
103 |
-
* @param Blogger_OAuthRequest $request
|
104 |
-
* @param Blogger_OAuthConsumer $consumer
|
105 |
-
* @param Blogger_OAuthToken $token
|
106 |
-
* @return string
|
107 |
-
*/
|
108 |
-
abstract public function build_signature($request, $consumer, $token);
|
109 |
-
|
110 |
-
/**
|
111 |
-
* Verifies that a given signature is correct
|
112 |
-
* @param Blogger_OAuthRequest $request
|
113 |
-
* @param Blogger_OAuthConsumer $consumer
|
114 |
-
* @param Blogger_OAuthToken $token
|
115 |
-
* @param string $signature
|
116 |
-
* @return bool
|
117 |
-
*/
|
118 |
-
public function check_signature($request, $consumer, $token, $signature) {
|
119 |
-
$built = $this->build_signature($request, $consumer, $token);
|
120 |
-
|
121 |
-
// Check for zero length, although unlikely here
|
122 |
-
if (strlen($built) == 0 || strlen($signature) == 0) {
|
123 |
-
return false;
|
124 |
-
}
|
125 |
-
|
126 |
-
if (strlen($built) != strlen($signature)) {
|
127 |
-
return false;
|
128 |
-
}
|
129 |
-
|
130 |
-
// Avoid a timing leak with a (hopefully) time insensitive compare
|
131 |
-
$result = 0;
|
132 |
-
for ($i = 0; $i < strlen($signature); $i++) {
|
133 |
-
$result |= ord($built{$i}) ^ ord($signature{$i});
|
134 |
-
}
|
135 |
-
|
136 |
-
return $result == 0;
|
137 |
-
}
|
138 |
-
}
|
139 |
-
|
140 |
-
/**
|
141 |
-
* The HMAC-SHA1 signature method uses the HMAC-SHA1 signature algorithm as defined in [RFC2104]
|
142 |
-
* where the Signature Base String is the text and the key is the concatenated values (each first
|
143 |
-
* encoded per Parameter Encoding) of the Consumer Secret and Token Secret, separated by an '&'
|
144 |
-
* character (ASCII code 38) even if empty.
|
145 |
-
* - Chapter 9.2 ("HMAC-SHA1")
|
146 |
-
*/
|
147 |
-
class Blogger_OAuthSignatureMethod_HMAC_SHA1 extends Blogger_OAuthSignatureMethod {
|
148 |
-
function get_name() {
|
149 |
-
return "HMAC-SHA1";
|
150 |
-
}
|
151 |
-
|
152 |
-
public function build_signature($request, $consumer, $token) {
|
153 |
-
$base_string = $request->get_signature_base_string();
|
154 |
-
$request->base_string = $base_string;
|
155 |
-
|
156 |
-
$key_parts = array(
|
157 |
-
$consumer->secret,
|
158 |
-
($token) ? $token->secret : ""
|
159 |
-
);
|
160 |
-
|
161 |
-
$key_parts = Blogger_OAuthUtil::urlencode_rfc3986($key_parts);
|
162 |
-
$key = implode('&', $key_parts);
|
163 |
-
|
164 |
-
return base64_encode(hash_hmac('sha1', $base_string, $key, true));
|
165 |
-
}
|
166 |
-
}
|
167 |
-
|
168 |
-
/**
|
169 |
-
* The PLAINTEXT method does not provide any security protection and SHOULD only be used
|
170 |
-
* over a secure channel such as HTTPS. It does not use the Signature Base String.
|
171 |
-
* - Chapter 9.4 ("PLAINTEXT")
|
172 |
-
*/
|
173 |
-
class Blogger_OAuthSignatureMethod_PLAINTEXT extends Blogger_OAuthSignatureMethod {
|
174 |
-
public function get_name() {
|
175 |
-
return "PLAINTEXT";
|
176 |
-
}
|
177 |
-
|
178 |
-
/**
|
179 |
-
* oauth_signature is set to the concatenated encoded values of the Consumer Secret and
|
180 |
-
* Token Secret, separated by a '&' character (ASCII code 38), even if either secret is
|
181 |
-
* empty. The result MUST be encoded again.
|
182 |
-
* - Chapter 9.4.1 ("Generating Signatures")
|
183 |
-
*
|
184 |
-
* Please note that the second encoding MUST NOT happen in the SignatureMethod, as
|
185 |
-
* Blogger_OAuthRequest handles this!
|
186 |
-
*/
|
187 |
-
public function build_signature($request, $consumer, $token) {
|
188 |
-
$key_parts = array(
|
189 |
-
$consumer->secret,
|
190 |
-
($token) ? $token->secret : ""
|
191 |
-
);
|
192 |
-
|
193 |
-
$key_parts = Blogger_OAuthUtil::urlencode_rfc3986($key_parts);
|
194 |
-
$key = implode('&', $key_parts);
|
195 |
-
$request->base_string = $key;
|
196 |
-
|
197 |
-
return $key;
|
198 |
-
}
|
199 |
-
}
|
200 |
-
|
201 |
-
/**
|
202 |
-
* The RSA-SHA1 signature method uses the RSASSA-PKCS1-v1_5 signature algorithm as defined in
|
203 |
-
* [RFC3447] section 8.2 (more simply known as PKCS#1), using SHA-1 as the hash function for
|
204 |
-
* EMSA-PKCS1-v1_5. It is assumed that the Consumer has provided its RSA public key in a
|
205 |
-
* verified way to the Service Provider, in a manner which is beyond the scope of this
|
206 |
-
* specification.
|
207 |
-
* - Chapter 9.3 ("RSA-SHA1")
|
208 |
-
*/
|
209 |
-
abstract class Blogger_OAuthSignatureMethod_RSA_SHA1 extends Blogger_OAuthSignatureMethod {
|
210 |
-
public function get_name() {
|
211 |
-
return "RSA-SHA1";
|
212 |
-
}
|
213 |
-
|
214 |
-
// Up to the SP to implement this lookup of keys. Possible ideas are:
|
215 |
-
// (1) do a lookup in a table of trusted certs keyed off of consumer
|
216 |
-
// (2) fetch via http using a url provided by the requester
|
217 |
-
// (3) some sort of specific discovery code based on request
|
218 |
-
//
|
219 |
-
// Either way should return a string representation of the certificate
|
220 |
-
protected abstract function fetch_public_cert(&$request);
|
221 |
-
|
222 |
-
// Up to the SP to implement this lookup of keys. Possible ideas are:
|
223 |
-
// (1) do a lookup in a table of trusted certs keyed off of consumer
|
224 |
-
//
|
225 |
-
// Either way should return a string representation of the certificate
|
226 |
-
protected abstract function fetch_private_cert(&$request);
|
227 |
-
|
228 |
-
public function build_signature($request, $consumer, $token) {
|
229 |
-
$base_string = $request->get_signature_base_string();
|
230 |
-
$request->base_string = $base_string;
|
231 |
-
|
232 |
-
// Fetch the private key cert based on the request
|
233 |
-
$cert = $this->fetch_private_cert($request);
|
234 |
-
|
235 |
-
// Pull the private key ID from the certificate
|
236 |
-
$privatekeyid = openssl_get_privatekey($cert);
|
237 |
-
|
238 |
-
// Sign using the key
|
239 |
-
$ok = openssl_sign($base_string, $signature, $privatekeyid);
|
240 |
-
|
241 |
-
// Release the key resource
|
242 |
-
openssl_free_key($privatekeyid);
|
243 |
-
|
244 |
-
return base64_encode($signature);
|
245 |
-
}
|
246 |
-
|
247 |
-
public function check_signature($request, $consumer, $token, $signature) {
|
248 |
-
$decoded_sig = base64_decode($signature);
|
249 |
-
|
250 |
-
$base_string = $request->get_signature_base_string();
|
251 |
-
|
252 |
-
// Fetch the public key cert based on the request
|
253 |
-
$cert = $this->fetch_public_cert($request);
|
254 |
-
|
255 |
-
// Pull the public key ID from the certificate
|
256 |
-
$publickeyid = openssl_get_publickey($cert);
|
257 |
-
|
258 |
-
// Check the computed signature against the one passed in the query
|
259 |
-
$ok = openssl_verify($base_string, $decoded_sig, $publickeyid);
|
260 |
-
|
261 |
-
// Release the key resource
|
262 |
-
openssl_free_key($publickeyid);
|
263 |
-
|
264 |
-
return $ok == 1;
|
265 |
-
}
|
266 |
-
}
|
267 |
-
|
268 |
-
class Blogger_OAuthRequest {
|
269 |
-
protected $parameters;
|
270 |
-
protected $http_method;
|
271 |
-
protected $http_url;
|
272 |
-
// for debug purposes
|
273 |
-
public $base_string;
|
274 |
-
public static $version = '1.0';
|
275 |
-
public static $POST_INPUT = 'php://input';
|
276 |
-
|
277 |
-
function __construct($http_method, $http_url, $parameters=NULL) {
|
278 |
-
$parameters = ($parameters) ? $parameters : array();
|
279 |
-
$parameters = array_merge( Blogger_OAuthUtil::parse_parameters(parse_url($http_url, PHP_URL_QUERY)), $parameters);
|
280 |
-
$this->parameters = $parameters;
|
281 |
-
$this->http_method = $http_method;
|
282 |
-
$this->http_url = $http_url;
|
283 |
-
}
|
284 |
-
|
285 |
-
|
286 |
-
/**
|
287 |
-
* attempt to build up a request from what was passed to the server
|
288 |
-
*/
|
289 |
-
public static function from_request($http_method=NULL, $http_url=NULL, $parameters=NULL) {
|
290 |
-
$scheme = (!isset($_SERVER['HTTPS']) || $_SERVER['HTTPS'] != "on")
|
291 |
-
? 'http'
|
292 |
-
: 'https';
|
293 |
-
$http_url = ($http_url) ? $http_url : $scheme .
|
294 |
-
'://' . $_SERVER['SERVER_NAME'] .
|
295 |
-
':' .
|
296 |
-
$_SERVER['SERVER_PORT'] .
|
297 |
-
$_SERVER['REQUEST_URI'];
|
298 |
-
$http_method = ($http_method) ? $http_method : $_SERVER['REQUEST_METHOD'];
|
299 |
-
|
300 |
-
// We weren't handed any parameters, so let's find the ones relevant to
|
301 |
-
// this request.
|
302 |
-
// If you run XML-RPC or similar you should use this to provide your own
|
303 |
-
// parsed parameter-list
|
304 |
-
if (!$parameters) {
|
305 |
-
// Find request headers
|
306 |
-
$request_headers = Blogger_OAuthUtil::get_headers();
|
307 |
-
|
308 |
-
// Parse the query-string to find GET parameters
|
309 |
-
$parameters = Blogger_OAuthUtil::parse_parameters($_SERVER['QUERY_STRING']);
|
310 |
-
|
311 |
-
// It's a POST request of the proper content-type, so parse POST
|
312 |
-
// parameters and add those overriding any duplicates from GET
|
313 |
-
if ($http_method == "POST"
|
314 |
-
&& isset($request_headers['Content-Type'])
|
315 |
-
&& strstr($request_headers['Content-Type'],
|
316 |
-
'application/x-www-form-urlencoded')
|
317 |
-
) {
|
318 |
-
$post_data = Blogger_OAuthUtil::parse_parameters(
|
319 |
-
file_get_contents(self::$POST_INPUT)
|
320 |
-
);
|
321 |
-
$parameters = array_merge($parameters, $post_data);
|
322 |
-
}
|
323 |
-
|
324 |
-
// We have a Authorization-header with Blogger_OAuth data. Parse the header
|
325 |
-
// and add those overriding any duplicates from GET or POST
|
326 |
-
if (isset($request_headers['Authorization']) && substr($request_headers['Authorization'], 0, 6) == 'Blogger_OAuth ') {
|
327 |
-
$header_parameters = Blogger_OAuthUtil::split_header(
|
328 |
-
$request_headers['Authorization']
|
329 |
-
);
|
330 |
-
$parameters = array_merge($parameters, $header_parameters);
|
331 |
-
}
|
332 |
-
|
333 |
-
}
|
334 |
-
|
335 |
-
return new Blogger_OAuthRequest($http_method, $http_url, $parameters);
|
336 |
-
}
|
337 |
-
|
338 |
-
/**
|
339 |
-
* pretty much a helper function to set up the request
|
340 |
-
*/
|
341 |
-
public static function from_consumer_and_token($consumer, $token, $http_method, $http_url, $parameters=NULL) {
|
342 |
-
$parameters = ($parameters) ? $parameters : array();
|
343 |
-
$defaults = array("oauth_version" => Blogger_OAuthRequest::$version,
|
344 |
-
"oauth_nonce" => Blogger_OAuthRequest::generate_nonce(),
|
345 |
-
"oauth_timestamp" => Blogger_OAuthRequest::generate_timestamp(),
|
346 |
-
"oauth_consumer_key" => $consumer->key);
|
347 |
-
if ($token)
|
348 |
-
$defaults['oauth_token'] = $token->key;
|
349 |
-
|
350 |
-
$parameters = array_merge($defaults, $parameters);
|
351 |
-
|
352 |
-
return new Blogger_OAuthRequest($http_method, $http_url, $parameters);
|
353 |
-
}
|
354 |
-
|
355 |
-
public function set_parameter($name, $value, $allow_duplicates = true) {
|
356 |
-
if ($allow_duplicates && isset($this->parameters[$name])) {
|
357 |
-
// We have already added parameter(s) with this name, so add to the list
|
358 |
-
if (is_scalar($this->parameters[$name])) {
|
359 |
-
// This is the first duplicate, so transform scalar (string)
|
360 |
-
// into an array so we can add the duplicates
|
361 |
-
$this->parameters[$name] = array($this->parameters[$name]);
|
362 |
-
}
|
363 |
-
|
364 |
-
$this->parameters[$name][] = $value;
|
365 |
-
} else {
|
366 |
-
$this->parameters[$name] = $value;
|
367 |
-
}
|
368 |
-
}
|
369 |
-
|
370 |
-
public function get_parameter($name) {
|
371 |
-
return isset($this->parameters[$name]) ? $this->parameters[$name] : null;
|
372 |
-
}
|
373 |
-
|
374 |
-
public function get_parameters() {
|
375 |
-
return $this->parameters;
|
376 |
-
}
|
377 |
-
|
378 |
-
public function unset_parameter($name) {
|
379 |
-
unset($this->parameters[$name]);
|
380 |
-
}
|
381 |
-
|
382 |
-
/**
|
383 |
-
* The request parameters, sorted and concatenated into a normalized string.
|
384 |
-
* @return string
|
385 |
-
*/
|
386 |
-
public function get_signable_parameters() {
|
387 |
-
// Grab all parameters
|
388 |
-
$params = $this->parameters;
|
389 |
-
|
390 |
-
// Remove oauth_signature if present
|
391 |
-
// Ref: Spec: 9.1.1 ("The oauth_signature parameter MUST be excluded.")
|
392 |
-
if (isset($params['oauth_signature'])) {
|
393 |
-
unset($params['oauth_signature']);
|
394 |
-
}
|
395 |
-
|
396 |
-
return Blogger_OAuthUtil::build_http_query($params);
|
397 |
-
}
|
398 |
-
|
399 |
-
/**
|
400 |
-
* Returns the base string of this request
|
401 |
-
*
|
402 |
-
* The base string defined as the method, the url
|
403 |
-
* and the parameters (normalized), each urlencoded
|
404 |
-
* and the concated with &.
|
405 |
-
*/
|
406 |
-
public function get_signature_base_string() {
|
407 |
-
$parts = array(
|
408 |
-
$this->get_normalized_http_method(),
|
409 |
-
$this->get_normalized_http_url(),
|
410 |
-
$this->get_signable_parameters()
|
411 |
-
);
|
412 |
-
|
413 |
-
$parts = Blogger_OAuthUtil::urlencode_rfc3986($parts);
|
414 |
-
|
415 |
-
return implode('&', $parts);
|
416 |
-
}
|
417 |
-
|
418 |
-
/**
|
419 |
-
* just uppercases the http method
|
420 |
-
*/
|
421 |
-
public function get_normalized_http_method() {
|
422 |
-
return strtoupper($this->http_method);
|
423 |
-
}
|
424 |
-
|
425 |
-
/**
|
426 |
-
* parses the url and rebuilds it to be
|
427 |
-
* scheme://host/path
|
428 |
-
*/
|
429 |
-
public function get_normalized_http_url() {
|
430 |
-
$parts = parse_url($this->http_url);
|
431 |
-
|
432 |
-
$scheme = (isset($parts['scheme'])) ? $parts['scheme'] : 'http';
|
433 |
-
$port = (isset($parts['port'])) ? $parts['port'] : (($scheme == 'https') ? '443' : '80');
|
434 |
-
$host = (isset($parts['host'])) ? strtolower($parts['host']) : '';
|
435 |
-
$path = (isset($parts['path'])) ? $parts['path'] : '';
|
436 |
-
|
437 |
-
if (($scheme == 'https' && $port != '443')
|
438 |
-
|| ($scheme == 'http' && $port != '80')) {
|
439 |
-
$host = "$host:$port";
|
440 |
-
}
|
441 |
-
return "$scheme://$host$path";
|
442 |
-
}
|
443 |
-
|
444 |
-
/**
|
445 |
-
* builds a url usable for a GET request
|
446 |
-
*/
|
447 |
-
public function to_url() {
|
448 |
-
$post_data = $this->to_postdata();
|
449 |
-
$out = $this->get_normalized_http_url();
|
450 |
-
if ($post_data) {
|
451 |
-
$out .= '?'.$post_data;
|
452 |
-
}
|
453 |
-
return $out;
|
454 |
-
}
|
455 |
-
|
456 |
-
/**
|
457 |
-
* builds the data one would send in a POST request
|
458 |
-
*/
|
459 |
-
public function to_postdata() {
|
460 |
-
return Blogger_OAuthUtil::build_http_query($this->parameters);
|
461 |
-
}
|
462 |
-
|
463 |
-
/**
|
464 |
-
* builds the Authorization: header
|
465 |
-
*/
|
466 |
-
public function to_header($realm=null) {
|
467 |
-
$first = true;
|
468 |
-
if($realm) {
|
469 |
-
$out = 'Authorization: Blogger_OAuth realm="' . Blogger_OAuthUtil::urlencode_rfc3986($realm) . '"';
|
470 |
-
$first = false;
|
471 |
-
} else
|
472 |
-
$out = 'Authorization: Blogger_OAuth';
|
473 |
-
|
474 |
-
$total = array();
|
475 |
-
foreach ($this->parameters as $k => $v) {
|
476 |
-
if (substr($k, 0, 5) != "oauth") continue;
|
477 |
-
if (is_array($v)) {
|
478 |
-
throw new Blogger_OAuthException('Arrays not supported in headers');
|
479 |
-
}
|
480 |
-
$out .= ($first) ? ' ' : ',';
|
481 |
-
$out .= Blogger_OAuthUtil::urlencode_rfc3986($k) .
|
482 |
-
'="' .
|
483 |
-
Blogger_OAuthUtil::urlencode_rfc3986($v) .
|
484 |
-
'"';
|
485 |
-
$first = false;
|
486 |
-
}
|
487 |
-
return $out;
|
488 |
-
}
|
489 |
-
|
490 |
-
public function __toString() {
|
491 |
-
return $this->to_url();
|
492 |
-
}
|
493 |
-
|
494 |
-
|
495 |
-
public function sign_request($signature_method, $consumer, $token) {
|
496 |
-
$this->set_parameter(
|
497 |
-
"oauth_signature_method",
|
498 |
-
$signature_method->get_name(),
|
499 |
-
false
|
500 |
-
);
|
501 |
-
$signature = $this->build_signature($signature_method, $consumer, $token);
|
502 |
-
$this->set_parameter("oauth_signature", $signature, false);
|
503 |
-
}
|
504 |
-
|
505 |
-
public function build_signature($signature_method, $consumer, $token) {
|
506 |
-
$signature = $signature_method->build_signature($this, $consumer, $token);
|
507 |
-
return $signature;
|
508 |
-
}
|
509 |
-
|
510 |
-
/**
|
511 |
-
* util function: current timestamp
|
512 |
-
*/
|
513 |
-
private static function generate_timestamp() {
|
514 |
-
return time();
|
515 |
-
}
|
516 |
-
|
517 |
-
/**
|
518 |
-
* util function: current nonce
|
519 |
-
*/
|
520 |
-
private static function generate_nonce() {
|
521 |
-
$mt = microtime();
|
522 |
-
$rand = mt_rand();
|
523 |
-
|
524 |
-
return md5($mt . $rand); // md5s look nicer than numbers
|
525 |
-
}
|
526 |
-
}
|
527 |
-
|
528 |
-
class Blogger_OAuthServer {
|
529 |
-
protected $timestamp_threshold = 300; // in seconds, five minutes
|
530 |
-
protected $version = '1.0'; // hi blaine
|
531 |
-
protected $signature_methods = array();
|
532 |
-
|
533 |
-
protected $data_store;
|
534 |
-
|
535 |
-
function __construct($data_store) {
|
536 |
-
$this->data_store = $data_store;
|
537 |
-
}
|
538 |
-
|
539 |
-
public function add_signature_method($signature_method) {
|
540 |
-
$this->signature_methods[$signature_method->get_name()] =
|
541 |
-
$signature_method;
|
542 |
-
}
|
543 |
-
|
544 |
-
// high level functions
|
545 |
-
|
546 |
-
/**
|
547 |
-
* process a request_token request
|
548 |
-
* returns the request token on success
|
549 |
-
*/
|
550 |
-
public function fetch_request_token(&$request) {
|
551 |
-
$this->get_version($request);
|
552 |
-
|
553 |
-
$consumer = $this->get_consumer($request);
|
554 |
-
|
555 |
-
// no token required for the initial token request
|
556 |
-
$token = NULL;
|
557 |
-
|
558 |
-
$this->check_signature($request, $consumer, $token);
|
559 |
-
|
560 |
-
// Rev A change
|
561 |
-
$callback = $request->get_parameter('oauth_callback');
|
562 |
-
$new_token = $this->data_store->new_request_token($consumer, $callback);
|
563 |
-
|
564 |
-
return $new_token;
|
565 |
-
}
|
566 |
-
|
567 |
-
/**
|
568 |
-
* process an access_token request
|
569 |
-
* returns the access token on success
|
570 |
-
*/
|
571 |
-
public function fetch_access_token(&$request) {
|
572 |
-
$this->get_version($request);
|
573 |
-
|
574 |
-
$consumer = $this->get_consumer($request);
|
575 |
-
|
576 |
-
// requires authorized request token
|
577 |
-
$token = $this->get_token($request, $consumer, "request");
|
578 |
-
|
579 |
-
$this->check_signature($request, $consumer, $token);
|
580 |
-
|
581 |
-
// Rev A change
|
582 |
-
$verifier = $request->get_parameter('oauth_verifier');
|
583 |
-
$new_token = $this->data_store->new_access_token($token, $consumer, $verifier);
|
584 |
-
|
585 |
-
return $new_token;
|
586 |
-
}
|
587 |
-
|
588 |
-
/**
|
589 |
-
* verify an api call, checks all the parameters
|
590 |
-
*/
|
591 |
-
public function verify_request(&$request) {
|
592 |
-
$this->get_version($request);
|
593 |
-
$consumer = $this->get_consumer($request);
|
594 |
-
$token = $this->get_token($request, $consumer, "access");
|
595 |
-
$this->check_signature($request, $consumer, $token);
|
596 |
-
return array($consumer, $token);
|
597 |
-
}
|
598 |
-
|
599 |
-
// Internals from here
|
600 |
-
/**
|
601 |
-
* version 1
|
602 |
-
*/
|
603 |
-
private function get_version(&$request) {
|
604 |
-
$version = $request->get_parameter("oauth_version");
|
605 |
-
if (!$version) {
|
606 |
-
// Service Providers MUST assume the protocol version to be 1.0 if this parameter is not present.
|
607 |
-
// Chapter 7.0 ("Accessing Protected Ressources")
|
608 |
-
$version = '1.0';
|
609 |
-
}
|
610 |
-
if ($version !== $this->version) {
|
611 |
-
throw new Blogger_OAuthException("Blogger_OAuth version '$version' not supported");
|
612 |
-
}
|
613 |
-
return $version;
|
614 |
-
}
|
615 |
-
|
616 |
-
/**
|
617 |
-
* figure out the signature with some defaults
|
618 |
-
*/
|
619 |
-
private function get_signature_method($request) {
|
620 |
-
$signature_method = $request instanceof Blogger_OAuthRequest
|
621 |
-
? $request->get_parameter("oauth_signature_method")
|
622 |
-
: NULL;
|
623 |
-
|
624 |
-
if (!$signature_method) {
|
625 |
-
// According to chapter 7 ("Accessing Protected Ressources") the signature-method
|
626 |
-
// parameter is required, and we can't just fallback to PLAINTEXT
|
627 |
-
throw new Blogger_OAuthException('No signature method parameter. This parameter is required');
|
628 |
-
}
|
629 |
-
|
630 |
-
if (!in_array($signature_method,
|
631 |
-
array_keys($this->signature_methods))) {
|
632 |
-
throw new Blogger_OAuthException(
|
633 |
-
"Signature method '$signature_method' not supported " .
|
634 |
-
"try one of the following: " .
|
635 |
-
implode(", ", array_keys($this->signature_methods))
|
636 |
-
);
|
637 |
-
}
|
638 |
-
return $this->signature_methods[$signature_method];
|
639 |
-
}
|
640 |
-
|
641 |
-
/**
|
642 |
-
* try to find the consumer for the provided request's consumer key
|
643 |
-
*/
|
644 |
-
private function get_consumer($request) {
|
645 |
-
$consumer_key = $request instanceof Blogger_OAuthRequest
|
646 |
-
? $request->get_parameter("oauth_consumer_key")
|
647 |
-
: NULL;
|
648 |
-
|
649 |
-
if (!$consumer_key) {
|
650 |
-
throw new Blogger_OAuthException("Invalid consumer key");
|
651 |
-
}
|
652 |
-
|
653 |
-
$consumer = $this->data_store->lookup_consumer($consumer_key);
|
654 |
-
if (!$consumer) {
|
655 |
-
throw new Blogger_OAuthException("Invalid consumer");
|
656 |
-
}
|
657 |
-
|
658 |
-
return $consumer;
|
659 |
-
}
|
660 |
-
|
661 |
-
/**
|
662 |
-
* try to find the token for the provided request's token key
|
663 |
-
*/
|
664 |
-
private function get_token($request, $consumer, $token_type="access") {
|
665 |
-
$token_field = $request instanceof Blogger_OAuthRequest
|
666 |
-
? $request->get_parameter('oauth_token')
|
667 |
-
: NULL;
|
668 |
-
|
669 |
-
$token = $this->data_store->lookup_token(
|
670 |
-
$consumer, $token_type, $token_field
|
671 |
-
);
|
672 |
-
if (!$token) {
|
673 |
-
throw new Blogger_OAuthException("Invalid $token_type token: $token_field");
|
674 |
-
}
|
675 |
-
return $token;
|
676 |
-
}
|
677 |
-
|
678 |
-
/**
|
679 |
-
* all-in-one function to check the signature on a request
|
680 |
-
* should guess the signature method appropriately
|
681 |
-
*/
|
682 |
-
private function check_signature($request, $consumer, $token) {
|
683 |
-
// this should probably be in a different method
|
684 |
-
$timestamp = $request instanceof Blogger_OAuthRequest
|
685 |
-
? $request->get_parameter('oauth_timestamp')
|
686 |
-
: NULL;
|
687 |
-
$nonce = $request instanceof Blogger_OAuthRequest
|
688 |
-
? $request->get_parameter('oauth_nonce')
|
689 |
-
: NULL;
|
690 |
-
|
691 |
-
$this->check_timestamp($timestamp);
|
692 |
-
$this->check_nonce($consumer, $token, $nonce, $timestamp);
|
693 |
-
|
694 |
-
$signature_method = $this->get_signature_method($request);
|
695 |
-
|
696 |
-
$signature = $request->get_parameter('oauth_signature');
|
697 |
-
$valid_sig = $signature_method->check_signature(
|
698 |
-
$request,
|
699 |
-
$consumer,
|
700 |
-
$token,
|
701 |
-
$signature
|
702 |
-
);
|
703 |
-
|
704 |
-
if (!$valid_sig) {
|
705 |
-
throw new Blogger_OAuthException("Invalid signature");
|
706 |
-
}
|
707 |
-
}
|
708 |
-
|
709 |
-
/**
|
710 |
-
* check that the timestamp is new enough
|
711 |
-
*/
|
712 |
-
private function check_timestamp($timestamp) {
|
713 |
-
if( ! $timestamp )
|
714 |
-
throw new Blogger_OAuthException(
|
715 |
-
'Missing timestamp parameter. The parameter is required'
|
716 |
-
);
|
717 |
-
|
718 |
-
// verify that timestamp is recentish
|
719 |
-
$now = time();
|
720 |
-
if (abs($now - $timestamp) > $this->timestamp_threshold) {
|
721 |
-
throw new Blogger_OAuthException(
|
722 |
-
"Expired timestamp, yours $timestamp, ours $now"
|
723 |
-
);
|
724 |
-
}
|
725 |
-
}
|
726 |
-
|
727 |
-
/**
|
728 |
-
* check that the nonce is not repeated
|
729 |
-
*/
|
730 |
-
private function check_nonce($consumer, $token, $nonce, $timestamp) {
|
731 |
-
if( ! $nonce )
|
732 |
-
throw new Blogger_OAuthException(
|
733 |
-
'Missing nonce parameter. The parameter is required'
|
734 |
-
);
|
735 |
-
|
736 |
-
// verify that the nonce is uniqueish
|
737 |
-
$found = $this->data_store->lookup_nonce(
|
738 |
-
$consumer,
|
739 |
-
$token,
|
740 |
-
$nonce,
|
741 |
-
$timestamp
|
742 |
-
);
|
743 |
-
if ($found) {
|
744 |
-
throw new Blogger_OAuthException("Nonce already used: $nonce");
|
745 |
-
}
|
746 |
-
}
|
747 |
-
|
748 |
-
}
|
749 |
-
|
750 |
-
class Blogger_OAuthDataStore {
|
751 |
-
function lookup_consumer($consumer_key) {
|
752 |
-
// implement me
|
753 |
-
}
|
754 |
-
|
755 |
-
function lookup_token($consumer, $token_type, $token) {
|
756 |
-
// implement me
|
757 |
-
}
|
758 |
-
|
759 |
-
function lookup_nonce($consumer, $token, $nonce, $timestamp) {
|
760 |
-
// implement me
|
761 |
-
}
|
762 |
-
|
763 |
-
function new_request_token($consumer, $callback = null) {
|
764 |
-
// return a new token attached to this consumer
|
765 |
-
}
|
766 |
-
|
767 |
-
function new_access_token($token, $consumer, $verifier = null) {
|
768 |
-
// return a new access token attached to this consumer
|
769 |
-
// for the user associated with this token if the request token
|
770 |
-
// is authorized
|
771 |
-
// should also invalidate the request token
|
772 |
-
}
|
773 |
-
|
774 |
-
}
|
775 |
-
|
776 |
-
class Blogger_OAuthUtil {
|
777 |
-
public static function
|
778 |
-
|
779 |
-
|
780 |
-
|
781 |
-
|
782 |
-
|
783 |
-
|
784 |
-
|
785 |
-
|
786 |
-
|
787 |
-
|
788 |
-
|
789 |
-
|
790 |
-
|
791 |
-
|
792 |
-
|
793 |
-
|
794 |
-
|
795 |
-
|
796 |
-
|
797 |
-
|
798 |
-
|
799 |
-
|
800 |
-
|
801 |
-
|
802 |
-
|
803 |
-
//
|
804 |
-
|
805 |
-
|
806 |
-
|
807 |
-
|
808 |
-
|
809 |
-
|
810 |
-
|
811 |
-
|
812 |
-
|
813 |
-
|
814 |
-
|
815 |
-
|
816 |
-
|
817 |
-
|
818 |
-
|
819 |
-
|
820 |
-
|
821 |
-
|
822 |
-
|
823 |
-
|
824 |
-
//
|
825 |
-
//
|
826 |
-
|
827 |
-
|
828 |
-
|
829 |
-
|
830 |
-
|
831 |
-
|
832 |
-
|
833 |
-
|
834 |
-
|
835 |
-
|
836 |
-
|
837 |
-
|
838 |
-
|
839 |
-
|
840 |
-
|
841 |
-
|
842 |
-
|
843 |
-
|
844 |
-
|
845 |
-
|
846 |
-
|
847 |
-
|
848 |
-
|
849 |
-
|
850 |
-
|
851 |
-
|
852 |
-
|
853 |
-
|
854 |
-
|
855 |
-
|
856 |
-
|
857 |
-
|
858 |
-
|
859 |
-
|
860 |
-
|
861 |
-
|
862 |
-
|
863 |
-
|
864 |
-
|
865 |
-
|
866 |
-
|
867 |
-
|
868 |
-
|
869 |
-
|
870 |
-
|
871 |
-
$
|
872 |
-
|
873 |
-
|
874 |
-
|
875 |
-
|
876 |
-
|
877 |
-
|
878 |
-
|
879 |
-
|
880 |
-
|
881 |
-
|
882 |
-
|
883 |
-
|
884 |
-
|
885 |
-
|
886 |
-
|
887 |
-
|
888 |
-
|
889 |
-
|
890 |
-
|
891 |
-
|
892 |
-
|
893 |
-
|
894 |
-
|
895 |
-
|
896 |
-
|
897 |
-
|
898 |
-
|
899 |
-
|
900 |
-
|
901 |
-
|
902 |
-
|
903 |
-
|
904 |
-
|
905 |
-
|
906 |
-
|
907 |
-
|
908 |
-
|
909 |
-
|
910 |
-
|
911 |
-
|
912 |
-
|
913 |
-
|
914 |
-
|
915 |
-
|
916 |
-
|
917 |
-
|
918 |
-
$
|
919 |
-
|
920 |
-
|
921 |
-
|
922 |
-
|
923 |
-
|
924 |
-
|
925 |
-
|
926 |
-
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
/*
|
3 |
+
Original File from: http://oauth.googlecode.com/svn/code/php/
|
4 |
+
License: MIT License:
|
5 |
+
|
6 |
+
The MIT License
|
7 |
+
|
8 |
+
Copyright (c) 2007 Andy Smith
|
9 |
+
|
10 |
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
11 |
+
of this software and associated documentation files (the "Software"), to deal
|
12 |
+
in the Software without restriction, including without limitation the rights
|
13 |
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
14 |
+
copies of the Software, and to permit persons to whom the Software is
|
15 |
+
furnished to do so, subject to the following conditions:
|
16 |
+
|
17 |
+
The above copyright notice and this permission notice shall be included in
|
18 |
+
all copies or substantial portions of the Software.
|
19 |
+
|
20 |
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
21 |
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
22 |
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
23 |
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
24 |
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
25 |
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
26 |
+
THE SOFTWARE.
|
27 |
+
|
28 |
+
|
29 |
+
Modified for use in WordPress by Otto
|
30 |
+
|
31 |
+
Changes: All classes renamed to Blogger_* to prevent conflicts with other plugins/code using the same OAuth library.
|
32 |
+
|
33 |
+
*/
|
34 |
+
// vim: foldmethod=marker
|
35 |
+
|
36 |
+
/* Generic exception class
|
37 |
+
*/
|
38 |
+
class Blogger_OAuthException extends Exception {
|
39 |
+
// pass
|
40 |
+
}
|
41 |
+
|
42 |
+
class Blogger_OAuthConsumer {
|
43 |
+
public $key;
|
44 |
+
public $secret;
|
45 |
+
|
46 |
+
function __construct($key, $secret, $callback_url=NULL) {
|
47 |
+
$this->key = $key;
|
48 |
+
$this->secret = $secret;
|
49 |
+
$this->callback_url = $callback_url;
|
50 |
+
}
|
51 |
+
|
52 |
+
function __toString() {
|
53 |
+
return "Blogger_OAuthConsumer[key=$this->key,secret=$this->secret]";
|
54 |
+
}
|
55 |
+
}
|
56 |
+
|
57 |
+
class Blogger_OAuthToken {
|
58 |
+
// access tokens and request tokens
|
59 |
+
public $key;
|
60 |
+
public $secret;
|
61 |
+
|
62 |
+
/**
|
63 |
+
* key = the token
|
64 |
+
* secret = the token secret
|
65 |
+
*/
|
66 |
+
function __construct($key, $secret) {
|
67 |
+
$this->key = $key;
|
68 |
+
$this->secret = $secret;
|
69 |
+
}
|
70 |
+
|
71 |
+
/**
|
72 |
+
* generates the basic string serialization of a token that a server
|
73 |
+
* would respond to request_token and access_token calls with
|
74 |
+
*/
|
75 |
+
function to_string() {
|
76 |
+
return "oauth_token=" .
|
77 |
+
Blogger_OAuthUtil::urlencode_rfc3986($this->key) .
|
78 |
+
"&oauth_token_secret=" .
|
79 |
+
Blogger_OAuthUtil::urlencode_rfc3986($this->secret);
|
80 |
+
}
|
81 |
+
|
82 |
+
function __toString() {
|
83 |
+
return $this->to_string();
|
84 |
+
}
|
85 |
+
}
|
86 |
+
|
87 |
+
/**
|
88 |
+
* A class for implementing a Signature Method
|
89 |
+
* See section 9 ("Signing Requests") in the spec
|
90 |
+
*/
|
91 |
+
abstract class Blogger_OAuthSignatureMethod {
|
92 |
+
/**
|
93 |
+
* Needs to return the name of the Signature Method (ie HMAC-SHA1)
|
94 |
+
* @return string
|
95 |
+
*/
|
96 |
+
abstract public function get_name();
|
97 |
+
|
98 |
+
/**
|
99 |
+
* Build up the signature
|
100 |
+
* NOTE: The output of this function MUST NOT be urlencoded.
|
101 |
+
* the encoding is handled in Blogger_OAuthRequest when the final
|
102 |
+
* request is serialized
|
103 |
+
* @param Blogger_OAuthRequest $request
|
104 |
+
* @param Blogger_OAuthConsumer $consumer
|
105 |
+
* @param Blogger_OAuthToken $token
|
106 |
+
* @return string
|
107 |
+
*/
|
108 |
+
abstract public function build_signature($request, $consumer, $token);
|
109 |
+
|
110 |
+
/**
|
111 |
+
* Verifies that a given signature is correct
|
112 |
+
* @param Blogger_OAuthRequest $request
|
113 |
+
* @param Blogger_OAuthConsumer $consumer
|
114 |
+
* @param Blogger_OAuthToken $token
|
115 |
+
* @param string $signature
|
116 |
+
* @return bool
|
117 |
+
*/
|
118 |
+
public function check_signature($request, $consumer, $token, $signature) {
|
119 |
+
$built = $this->build_signature($request, $consumer, $token);
|
120 |
+
|
121 |
+
// Check for zero length, although unlikely here
|
122 |
+
if (strlen($built) == 0 || strlen($signature) == 0) {
|
123 |
+
return false;
|
124 |
+
}
|
125 |
+
|
126 |
+
if (strlen($built) != strlen($signature)) {
|
127 |
+
return false;
|
128 |
+
}
|
129 |
+
|
130 |
+
// Avoid a timing leak with a (hopefully) time insensitive compare
|
131 |
+
$result = 0;
|
132 |
+
for ($i = 0; $i < strlen($signature); $i++) {
|
133 |
+
$result |= ord($built{$i}) ^ ord($signature{$i});
|
134 |
+
}
|
135 |
+
|
136 |
+
return $result == 0;
|
137 |
+
}
|
138 |
+
}
|
139 |
+
|
140 |
+
/**
|
141 |
+
* The HMAC-SHA1 signature method uses the HMAC-SHA1 signature algorithm as defined in [RFC2104]
|
142 |
+
* where the Signature Base String is the text and the key is the concatenated values (each first
|
143 |
+
* encoded per Parameter Encoding) of the Consumer Secret and Token Secret, separated by an '&'
|
144 |
+
* character (ASCII code 38) even if empty.
|
145 |
+
* - Chapter 9.2 ("HMAC-SHA1")
|
146 |
+
*/
|
147 |
+
class Blogger_OAuthSignatureMethod_HMAC_SHA1 extends Blogger_OAuthSignatureMethod {
|
148 |
+
function get_name() {
|
149 |
+
return "HMAC-SHA1";
|
150 |
+
}
|
151 |
+
|
152 |
+
public function build_signature($request, $consumer, $token) {
|
153 |
+
$base_string = $request->get_signature_base_string();
|
154 |
+
$request->base_string = $base_string;
|
155 |
+
|
156 |
+
$key_parts = array(
|
157 |
+
$consumer->secret,
|
158 |
+
($token) ? $token->secret : ""
|
159 |
+
);
|
160 |
+
|
161 |
+
$key_parts = Blogger_OAuthUtil::urlencode_rfc3986($key_parts);
|
162 |
+
$key = implode('&', $key_parts);
|
163 |
+
|
164 |
+
return base64_encode(hash_hmac('sha1', $base_string, $key, true));
|
165 |
+
}
|
166 |
+
}
|
167 |
+
|
168 |
+
/**
|
169 |
+
* The PLAINTEXT method does not provide any security protection and SHOULD only be used
|
170 |
+
* over a secure channel such as HTTPS. It does not use the Signature Base String.
|
171 |
+
* - Chapter 9.4 ("PLAINTEXT")
|
172 |
+
*/
|
173 |
+
class Blogger_OAuthSignatureMethod_PLAINTEXT extends Blogger_OAuthSignatureMethod {
|
174 |
+
public function get_name() {
|
175 |
+
return "PLAINTEXT";
|
176 |
+
}
|
177 |
+
|
178 |
+
/**
|
179 |
+
* oauth_signature is set to the concatenated encoded values of the Consumer Secret and
|
180 |
+
* Token Secret, separated by a '&' character (ASCII code 38), even if either secret is
|
181 |
+
* empty. The result MUST be encoded again.
|
182 |
+
* - Chapter 9.4.1 ("Generating Signatures")
|
183 |
+
*
|
184 |
+
* Please note that the second encoding MUST NOT happen in the SignatureMethod, as
|
185 |
+
* Blogger_OAuthRequest handles this!
|
186 |
+
*/
|
187 |
+
public function build_signature($request, $consumer, $token) {
|
188 |
+
$key_parts = array(
|
189 |
+
$consumer->secret,
|
190 |
+
($token) ? $token->secret : ""
|
191 |
+
);
|
192 |
+
|
193 |
+
$key_parts = Blogger_OAuthUtil::urlencode_rfc3986($key_parts);
|
194 |
+
$key = implode('&', $key_parts);
|
195 |
+
$request->base_string = $key;
|
196 |
+
|
197 |
+
return $key;
|
198 |
+
}
|
199 |
+
}
|
200 |
+
|
201 |
+
/**
|
202 |
+
* The RSA-SHA1 signature method uses the RSASSA-PKCS1-v1_5 signature algorithm as defined in
|
203 |
+
* [RFC3447] section 8.2 (more simply known as PKCS#1), using SHA-1 as the hash function for
|
204 |
+
* EMSA-PKCS1-v1_5. It is assumed that the Consumer has provided its RSA public key in a
|
205 |
+
* verified way to the Service Provider, in a manner which is beyond the scope of this
|
206 |
+
* specification.
|
207 |
+
* - Chapter 9.3 ("RSA-SHA1")
|
208 |
+
*/
|
209 |
+
abstract class Blogger_OAuthSignatureMethod_RSA_SHA1 extends Blogger_OAuthSignatureMethod {
|
210 |
+
public function get_name() {
|
211 |
+
return "RSA-SHA1";
|
212 |
+
}
|
213 |
+
|
214 |
+
// Up to the SP to implement this lookup of keys. Possible ideas are:
|
215 |
+
// (1) do a lookup in a table of trusted certs keyed off of consumer
|
216 |
+
// (2) fetch via http using a url provided by the requester
|
217 |
+
// (3) some sort of specific discovery code based on request
|
218 |
+
//
|
219 |
+
// Either way should return a string representation of the certificate
|
220 |
+
protected abstract function fetch_public_cert(&$request);
|
221 |
+
|
222 |
+
// Up to the SP to implement this lookup of keys. Possible ideas are:
|
223 |
+
// (1) do a lookup in a table of trusted certs keyed off of consumer
|
224 |
+
//
|
225 |
+
// Either way should return a string representation of the certificate
|
226 |
+
protected abstract function fetch_private_cert(&$request);
|
227 |
+
|
228 |
+
public function build_signature($request, $consumer, $token) {
|
229 |
+
$base_string = $request->get_signature_base_string();
|
230 |
+
$request->base_string = $base_string;
|
231 |
+
|
232 |
+
// Fetch the private key cert based on the request
|
233 |
+
$cert = $this->fetch_private_cert($request);
|
234 |
+
|
235 |
+
// Pull the private key ID from the certificate
|
236 |
+
$privatekeyid = openssl_get_privatekey($cert);
|
237 |
+
|
238 |
+
// Sign using the key
|
239 |
+
$ok = openssl_sign($base_string, $signature, $privatekeyid);
|
240 |
+
|
241 |
+
// Release the key resource
|
242 |
+
openssl_free_key($privatekeyid);
|
243 |
+
|
244 |
+
return base64_encode($signature);
|
245 |
+
}
|
246 |
+
|
247 |
+
public function check_signature($request, $consumer, $token, $signature) {
|
248 |
+
$decoded_sig = base64_decode($signature);
|
249 |
+
|
250 |
+
$base_string = $request->get_signature_base_string();
|
251 |
+
|
252 |
+
// Fetch the public key cert based on the request
|
253 |
+
$cert = $this->fetch_public_cert($request);
|
254 |
+
|
255 |
+
// Pull the public key ID from the certificate
|
256 |
+
$publickeyid = openssl_get_publickey($cert);
|
257 |
+
|
258 |
+
// Check the computed signature against the one passed in the query
|
259 |
+
$ok = openssl_verify($base_string, $decoded_sig, $publickeyid);
|
260 |
+
|
261 |
+
// Release the key resource
|
262 |
+
openssl_free_key($publickeyid);
|
263 |
+
|
264 |
+
return $ok == 1;
|
265 |
+
}
|
266 |
+
}
|
267 |
+
|
268 |
+
class Blogger_OAuthRequest {
|
269 |
+
protected $parameters;
|
270 |
+
protected $http_method;
|
271 |
+
protected $http_url;
|
272 |
+
// for debug purposes
|
273 |
+
public $base_string;
|
274 |
+
public static $version = '1.0';
|
275 |
+
public static $POST_INPUT = 'php://input';
|
276 |
+
|
277 |
+
function __construct($http_method, $http_url, $parameters=NULL) {
|
278 |
+
$parameters = ($parameters) ? $parameters : array();
|
279 |
+
$parameters = array_merge( Blogger_OAuthUtil::parse_parameters(parse_url($http_url, PHP_URL_QUERY)), $parameters);
|
280 |
+
$this->parameters = $parameters;
|
281 |
+
$this->http_method = $http_method;
|
282 |
+
$this->http_url = $http_url;
|
283 |
+
}
|
284 |
+
|
285 |
+
|
286 |
+
/**
|
287 |
+
* attempt to build up a request from what was passed to the server
|
288 |
+
*/
|
289 |
+
public static function from_request($http_method=NULL, $http_url=NULL, $parameters=NULL) {
|
290 |
+
$scheme = (!isset($_SERVER['HTTPS']) || $_SERVER['HTTPS'] != "on")
|
291 |
+
? 'http'
|
292 |
+
: 'https';
|
293 |
+
$http_url = ($http_url) ? $http_url : $scheme .
|
294 |
+
'://' . $_SERVER['SERVER_NAME'] .
|
295 |
+
':' .
|
296 |
+
$_SERVER['SERVER_PORT'] .
|
297 |
+
$_SERVER['REQUEST_URI'];
|
298 |
+
$http_method = ($http_method) ? $http_method : $_SERVER['REQUEST_METHOD'];
|
299 |
+
|
300 |
+
// We weren't handed any parameters, so let's find the ones relevant to
|
301 |
+
// this request.
|
302 |
+
// If you run XML-RPC or similar you should use this to provide your own
|
303 |
+
// parsed parameter-list
|
304 |
+
if (!$parameters) {
|
305 |
+
// Find request headers
|
306 |
+
$request_headers = Blogger_OAuthUtil::get_headers();
|
307 |
+
|
308 |
+
// Parse the query-string to find GET parameters
|
309 |
+
$parameters = Blogger_OAuthUtil::parse_parameters($_SERVER['QUERY_STRING']);
|
310 |
+
|
311 |
+
// It's a POST request of the proper content-type, so parse POST
|
312 |
+
// parameters and add those overriding any duplicates from GET
|
313 |
+
if ($http_method == "POST"
|
314 |
+
&& isset($request_headers['Content-Type'])
|
315 |
+
&& strstr($request_headers['Content-Type'],
|
316 |
+
'application/x-www-form-urlencoded')
|
317 |
+
) {
|
318 |
+
$post_data = Blogger_OAuthUtil::parse_parameters(
|
319 |
+
file_get_contents(self::$POST_INPUT)
|
320 |
+
);
|
321 |
+
$parameters = array_merge($parameters, $post_data);
|
322 |
+
}
|
323 |
+
|
324 |
+
// We have a Authorization-header with Blogger_OAuth data. Parse the header
|
325 |
+
// and add those overriding any duplicates from GET or POST
|
326 |
+
if (isset($request_headers['Authorization']) && substr($request_headers['Authorization'], 0, 6) == 'Blogger_OAuth ') {
|
327 |
+
$header_parameters = Blogger_OAuthUtil::split_header(
|
328 |
+
$request_headers['Authorization']
|
329 |
+
);
|
330 |
+
$parameters = array_merge($parameters, $header_parameters);
|
331 |
+
}
|
332 |
+
|
333 |
+
}
|
334 |
+
|
335 |
+
return new Blogger_OAuthRequest($http_method, $http_url, $parameters);
|
336 |
+
}
|
337 |
+
|
338 |
+
/**
|
339 |
+
* pretty much a helper function to set up the request
|
340 |
+
*/
|
341 |
+
public static function from_consumer_and_token($consumer, $token, $http_method, $http_url, $parameters=NULL) {
|
342 |
+
$parameters = ($parameters) ? $parameters : array();
|
343 |
+
$defaults = array("oauth_version" => Blogger_OAuthRequest::$version,
|
344 |
+
"oauth_nonce" => Blogger_OAuthRequest::generate_nonce(),
|
345 |
+
"oauth_timestamp" => Blogger_OAuthRequest::generate_timestamp(),
|
346 |
+
"oauth_consumer_key" => $consumer->key);
|
347 |
+
if ($token)
|
348 |
+
$defaults['oauth_token'] = $token->key;
|
349 |
+
|
350 |
+
$parameters = array_merge($defaults, $parameters);
|
351 |
+
|
352 |
+
return new Blogger_OAuthRequest($http_method, $http_url, $parameters);
|
353 |
+
}
|
354 |
+
|
355 |
+
public function set_parameter($name, $value, $allow_duplicates = true) {
|
356 |
+
if ($allow_duplicates && isset($this->parameters[$name])) {
|
357 |
+
// We have already added parameter(s) with this name, so add to the list
|
358 |
+
if (is_scalar($this->parameters[$name])) {
|
359 |
+
// This is the first duplicate, so transform scalar (string)
|
360 |
+
// into an array so we can add the duplicates
|
361 |
+
$this->parameters[$name] = array($this->parameters[$name]);
|
362 |
+
}
|
363 |
+
|
364 |
+
$this->parameters[$name][] = $value;
|
365 |
+
} else {
|
366 |
+
$this->parameters[$name] = $value;
|
367 |
+
}
|
368 |
+
}
|
369 |
+
|
370 |
+
public function get_parameter($name) {
|
371 |
+
return isset($this->parameters[$name]) ? $this->parameters[$name] : null;
|
372 |
+
}
|
373 |
+
|
374 |
+
public function get_parameters() {
|
375 |
+
return $this->parameters;
|
376 |
+
}
|
377 |
+
|
378 |
+
public function unset_parameter($name) {
|
379 |
+
unset($this->parameters[$name]);
|
380 |
+
}
|
381 |
+
|
382 |
+
/**
|
383 |
+
* The request parameters, sorted and concatenated into a normalized string.
|
384 |
+
* @return string
|
385 |
+
*/
|
386 |
+
public function get_signable_parameters() {
|
387 |
+
// Grab all parameters
|
388 |
+
$params = $this->parameters;
|
389 |
+
|
390 |
+
// Remove oauth_signature if present
|
391 |
+
// Ref: Spec: 9.1.1 ("The oauth_signature parameter MUST be excluded.")
|
392 |
+
if (isset($params['oauth_signature'])) {
|
393 |
+
unset($params['oauth_signature']);
|
394 |
+
}
|
395 |
+
|
396 |
+
return Blogger_OAuthUtil::build_http_query($params);
|
397 |
+
}
|
398 |
+
|
399 |
+
/**
|
400 |
+
* Returns the base string of this request
|
401 |
+
*
|
402 |
+
* The base string defined as the method, the url
|
403 |
+
* and the parameters (normalized), each urlencoded
|
404 |
+
* and the concated with &.
|
405 |
+
*/
|
406 |
+
public function get_signature_base_string() {
|
407 |
+
$parts = array(
|
408 |
+
$this->get_normalized_http_method(),
|
409 |
+
$this->get_normalized_http_url(),
|
410 |
+
$this->get_signable_parameters()
|
411 |
+
);
|
412 |
+
|
413 |
+
$parts = Blogger_OAuthUtil::urlencode_rfc3986($parts);
|
414 |
+
|
415 |
+
return implode('&', $parts);
|
416 |
+
}
|
417 |
+
|
418 |
+
/**
|
419 |
+
* just uppercases the http method
|
420 |
+
*/
|
421 |
+
public function get_normalized_http_method() {
|
422 |
+
return strtoupper($this->http_method);
|
423 |
+
}
|
424 |
+
|
425 |
+
/**
|
426 |
+
* parses the url and rebuilds it to be
|
427 |
+
* scheme://host/path
|
428 |
+
*/
|
429 |
+
public function get_normalized_http_url() {
|
430 |
+
$parts = parse_url($this->http_url);
|
431 |
+
|
432 |
+
$scheme = (isset($parts['scheme'])) ? $parts['scheme'] : 'http';
|
433 |
+
$port = (isset($parts['port'])) ? $parts['port'] : (($scheme == 'https') ? '443' : '80');
|
434 |
+
$host = (isset($parts['host'])) ? strtolower($parts['host']) : '';
|
435 |
+
$path = (isset($parts['path'])) ? $parts['path'] : '';
|
436 |
+
|
437 |
+
if (($scheme == 'https' && $port != '443')
|
438 |
+
|| ($scheme == 'http' && $port != '80')) {
|
439 |
+
$host = "$host:$port";
|
440 |
+
}
|
441 |
+
return "$scheme://$host$path";
|
442 |
+
}
|
443 |
+
|
444 |
+
/**
|
445 |
+
* builds a url usable for a GET request
|
446 |
+
*/
|
447 |
+
public function to_url() {
|
448 |
+
$post_data = $this->to_postdata();
|
449 |
+
$out = $this->get_normalized_http_url();
|
450 |
+
if ($post_data) {
|
451 |
+
$out .= '?'.$post_data;
|
452 |
+
}
|
453 |
+
return $out;
|
454 |
+
}
|
455 |
+
|
456 |
+
/**
|
457 |
+
* builds the data one would send in a POST request
|
458 |
+
*/
|
459 |
+
public function to_postdata() {
|
460 |
+
return Blogger_OAuthUtil::build_http_query($this->parameters);
|
461 |
+
}
|
462 |
+
|
463 |
+
/**
|
464 |
+
* builds the Authorization: header
|
465 |
+
*/
|
466 |
+
public function to_header($realm=null) {
|
467 |
+
$first = true;
|
468 |
+
if($realm) {
|
469 |
+
$out = 'Authorization: Blogger_OAuth realm="' . Blogger_OAuthUtil::urlencode_rfc3986($realm) . '"';
|
470 |
+
$first = false;
|
471 |
+
} else
|
472 |
+
$out = 'Authorization: Blogger_OAuth';
|
473 |
+
|
474 |
+
$total = array();
|
475 |
+
foreach ($this->parameters as $k => $v) {
|
476 |
+
if (substr($k, 0, 5) != "oauth") continue;
|
477 |
+
if (is_array($v)) {
|
478 |
+
throw new Blogger_OAuthException('Arrays not supported in headers');
|
479 |
+
}
|
480 |
+
$out .= ($first) ? ' ' : ',';
|
481 |
+
$out .= Blogger_OAuthUtil::urlencode_rfc3986($k) .
|
482 |
+
'="' .
|
483 |
+
Blogger_OAuthUtil::urlencode_rfc3986($v) .
|
484 |
+
'"';
|
485 |
+
$first = false;
|
486 |
+
}
|
487 |
+
return $out;
|
488 |
+
}
|
489 |
+
|
490 |
+
public function __toString() {
|
491 |
+
return $this->to_url();
|
492 |
+
}
|
493 |
+
|
494 |
+
|
495 |
+
public function sign_request($signature_method, $consumer, $token) {
|
496 |
+
$this->set_parameter(
|
497 |
+
"oauth_signature_method",
|
498 |
+
$signature_method->get_name(),
|
499 |
+
false
|
500 |
+
);
|
501 |
+
$signature = $this->build_signature($signature_method, $consumer, $token);
|
502 |
+
$this->set_parameter("oauth_signature", $signature, false);
|
503 |
+
}
|
504 |
+
|
505 |
+
public function build_signature($signature_method, $consumer, $token) {
|
506 |
+
$signature = $signature_method->build_signature($this, $consumer, $token);
|
507 |
+
return $signature;
|
508 |
+
}
|
509 |
+
|
510 |
+
/**
|
511 |
+
* util function: current timestamp
|
512 |
+
*/
|
513 |
+
private static function generate_timestamp() {
|
514 |
+
return time();
|
515 |
+
}
|
516 |
+
|
517 |
+
/**
|
518 |
+
* util function: current nonce
|
519 |
+
*/
|
520 |
+
private static function generate_nonce() {
|
521 |
+
$mt = microtime();
|
522 |
+
$rand = mt_rand();
|
523 |
+
|
524 |
+
return md5($mt . $rand); // md5s look nicer than numbers
|
525 |
+
}
|
526 |
+
}
|
527 |
+
|
528 |
+
class Blogger_OAuthServer {
|
529 |
+
protected $timestamp_threshold = 300; // in seconds, five minutes
|
530 |
+
protected $version = '1.0'; // hi blaine
|
531 |
+
protected $signature_methods = array();
|
532 |
+
|
533 |
+
protected $data_store;
|
534 |
+
|
535 |
+
function __construct($data_store) {
|
536 |
+
$this->data_store = $data_store;
|
537 |
+
}
|
538 |
+
|
539 |
+
public function add_signature_method($signature_method) {
|
540 |
+
$this->signature_methods[$signature_method->get_name()] =
|
541 |
+
$signature_method;
|
542 |
+
}
|
543 |
+
|
544 |
+
// high level functions
|
545 |
+
|
546 |
+
/**
|
547 |
+
* process a request_token request
|
548 |
+
* returns the request token on success
|
549 |
+
*/
|
550 |
+
public function fetch_request_token(&$request) {
|
551 |
+
$this->get_version($request);
|
552 |
+
|
553 |
+
$consumer = $this->get_consumer($request);
|
554 |
+
|
555 |
+
// no token required for the initial token request
|
556 |
+
$token = NULL;
|
557 |
+
|
558 |
+
$this->check_signature($request, $consumer, $token);
|
559 |
+
|
560 |
+
// Rev A change
|
561 |
+
$callback = $request->get_parameter('oauth_callback');
|
562 |
+
$new_token = $this->data_store->new_request_token($consumer, $callback);
|
563 |
+
|
564 |
+
return $new_token;
|
565 |
+
}
|
566 |
+
|
567 |
+
/**
|
568 |
+
* process an access_token request
|
569 |
+
* returns the access token on success
|
570 |
+
*/
|
571 |
+
public function fetch_access_token(&$request) {
|
572 |
+
$this->get_version($request);
|
573 |
+
|
574 |
+
$consumer = $this->get_consumer($request);
|
575 |
+
|
576 |
+
// requires authorized request token
|
577 |
+
$token = $this->get_token($request, $consumer, "request");
|
578 |
+
|
579 |
+
$this->check_signature($request, $consumer, $token);
|
580 |
+
|
581 |
+
// Rev A change
|
582 |
+
$verifier = $request->get_parameter('oauth_verifier');
|
583 |
+
$new_token = $this->data_store->new_access_token($token, $consumer, $verifier);
|
584 |
+
|
585 |
+
return $new_token;
|
586 |
+
}
|
587 |
+
|
588 |
+
/**
|
589 |
+
* verify an api call, checks all the parameters
|
590 |
+
*/
|
591 |
+
public function verify_request(&$request) {
|
592 |
+
$this->get_version($request);
|
593 |
+
$consumer = $this->get_consumer($request);
|
594 |
+
$token = $this->get_token($request, $consumer, "access");
|
595 |
+
$this->check_signature($request, $consumer, $token);
|
596 |
+
return array($consumer, $token);
|
597 |
+
}
|
598 |
+
|
599 |
+
// Internals from here
|
600 |
+
/**
|
601 |
+
* version 1
|
602 |
+
*/
|
603 |
+
private function get_version(&$request) {
|
604 |
+
$version = $request->get_parameter("oauth_version");
|
605 |
+
if (!$version) {
|
606 |
+
// Service Providers MUST assume the protocol version to be 1.0 if this parameter is not present.
|
607 |
+
// Chapter 7.0 ("Accessing Protected Ressources")
|
608 |
+
$version = '1.0';
|
609 |
+
}
|
610 |
+
if ($version !== $this->version) {
|
611 |
+
throw new Blogger_OAuthException("Blogger_OAuth version '$version' not supported");
|
612 |
+
}
|
613 |
+
return $version;
|
614 |
+
}
|
615 |
+
|
616 |
+
/**
|
617 |
+
* figure out the signature with some defaults
|
618 |
+
*/
|
619 |
+
private function get_signature_method($request) {
|
620 |
+
$signature_method = $request instanceof Blogger_OAuthRequest
|
621 |
+
? $request->get_parameter("oauth_signature_method")
|
622 |
+
: NULL;
|
623 |
+
|
624 |
+
if (!$signature_method) {
|
625 |
+
// According to chapter 7 ("Accessing Protected Ressources") the signature-method
|
626 |
+
// parameter is required, and we can't just fallback to PLAINTEXT
|
627 |
+
throw new Blogger_OAuthException('No signature method parameter. This parameter is required');
|
628 |
+
}
|
629 |
+
|
630 |
+
if (!in_array($signature_method,
|
631 |
+
array_keys($this->signature_methods))) {
|
632 |
+
throw new Blogger_OAuthException(
|
633 |
+
"Signature method '$signature_method' not supported " .
|
634 |
+
"try one of the following: " .
|
635 |
+
implode(", ", array_keys($this->signature_methods))
|
636 |
+
);
|
637 |
+
}
|
638 |
+
return $this->signature_methods[$signature_method];
|
639 |
+
}
|
640 |
+
|
641 |
+
/**
|
642 |
+
* try to find the consumer for the provided request's consumer key
|
643 |
+
*/
|
644 |
+
private function get_consumer($request) {
|
645 |
+
$consumer_key = $request instanceof Blogger_OAuthRequest
|
646 |
+
? $request->get_parameter("oauth_consumer_key")
|
647 |
+
: NULL;
|
648 |
+
|
649 |
+
if (!$consumer_key) {
|
650 |
+
throw new Blogger_OAuthException("Invalid consumer key");
|
651 |
+
}
|
652 |
+
|
653 |
+
$consumer = $this->data_store->lookup_consumer($consumer_key);
|
654 |
+
if (!$consumer) {
|
655 |
+
throw new Blogger_OAuthException("Invalid consumer");
|
656 |
+
}
|
657 |
+
|
658 |
+
return $consumer;
|
659 |
+
}
|
660 |
+
|
661 |
+
/**
|
662 |
+
* try to find the token for the provided request's token key
|
663 |
+
*/
|
664 |
+
private function get_token($request, $consumer, $token_type="access") {
|
665 |
+
$token_field = $request instanceof Blogger_OAuthRequest
|
666 |
+
? $request->get_parameter('oauth_token')
|
667 |
+
: NULL;
|
668 |
+
|
669 |
+
$token = $this->data_store->lookup_token(
|
670 |
+
$consumer, $token_type, $token_field
|
671 |
+
);
|
672 |
+
if (!$token) {
|
673 |
+
throw new Blogger_OAuthException("Invalid $token_type token: $token_field");
|
674 |
+
}
|
675 |
+
return $token;
|
676 |
+
}
|
677 |
+
|
678 |
+
/**
|
679 |
+
* all-in-one function to check the signature on a request
|
680 |
+
* should guess the signature method appropriately
|
681 |
+
*/
|
682 |
+
private function check_signature($request, $consumer, $token) {
|
683 |
+
// this should probably be in a different method
|
684 |
+
$timestamp = $request instanceof Blogger_OAuthRequest
|
685 |
+
? $request->get_parameter('oauth_timestamp')
|
686 |
+
: NULL;
|
687 |
+
$nonce = $request instanceof Blogger_OAuthRequest
|
688 |
+
? $request->get_parameter('oauth_nonce')
|
689 |
+
: NULL;
|
690 |
+
|
691 |
+
$this->check_timestamp($timestamp);
|
692 |
+
$this->check_nonce($consumer, $token, $nonce, $timestamp);
|
693 |
+
|
694 |
+
$signature_method = $this->get_signature_method($request);
|
695 |
+
|
696 |
+
$signature = $request->get_parameter('oauth_signature');
|
697 |
+
$valid_sig = $signature_method->check_signature(
|
698 |
+
$request,
|
699 |
+
$consumer,
|
700 |
+
$token,
|
701 |
+
$signature
|
702 |
+
);
|
703 |
+
|
704 |
+
if (!$valid_sig) {
|
705 |
+
throw new Blogger_OAuthException("Invalid signature");
|
706 |
+
}
|
707 |
+
}
|
708 |
+
|
709 |
+
/**
|
710 |
+
* check that the timestamp is new enough
|
711 |
+
*/
|
712 |
+
private function check_timestamp($timestamp) {
|
713 |
+
if( ! $timestamp )
|
714 |
+
throw new Blogger_OAuthException(
|
715 |
+
'Missing timestamp parameter. The parameter is required'
|
716 |
+
);
|
717 |
+
|
718 |
+
// verify that timestamp is recentish
|
719 |
+
$now = time();
|
720 |
+
if (abs($now - $timestamp) > $this->timestamp_threshold) {
|
721 |
+
throw new Blogger_OAuthException(
|
722 |
+
"Expired timestamp, yours $timestamp, ours $now"
|
723 |
+
);
|
724 |
+
}
|
725 |
+
}
|
726 |
+
|
727 |
+
/**
|
728 |
+
* check that the nonce is not repeated
|
729 |
+
*/
|
730 |
+
private function check_nonce($consumer, $token, $nonce, $timestamp) {
|
731 |
+
if( ! $nonce )
|
732 |
+
throw new Blogger_OAuthException(
|
733 |
+
'Missing nonce parameter. The parameter is required'
|
734 |
+
);
|
735 |
+
|
736 |
+
// verify that the nonce is uniqueish
|
737 |
+
$found = $this->data_store->lookup_nonce(
|
738 |
+
$consumer,
|
739 |
+
$token,
|
740 |
+
$nonce,
|
741 |
+
$timestamp
|
742 |
+
);
|
743 |
+
if ($found) {
|
744 |
+
throw new Blogger_OAuthException("Nonce already used: $nonce");
|
745 |
+
}
|
746 |
+
}
|
747 |
+
|
748 |
+
}
|
749 |
+
|
750 |
+
class Blogger_OAuthDataStore {
|
751 |
+
function lookup_consumer($consumer_key) {
|
752 |
+
// implement me
|
753 |
+
}
|
754 |
+
|
755 |
+
function lookup_token($consumer, $token_type, $token) {
|
756 |
+
// implement me
|
757 |
+
}
|
758 |
+
|
759 |
+
function lookup_nonce($consumer, $token, $nonce, $timestamp) {
|
760 |
+
// implement me
|
761 |
+
}
|
762 |
+
|
763 |
+
function new_request_token($consumer, $callback = null) {
|
764 |
+
// return a new token attached to this consumer
|
765 |
+
}
|
766 |
+
|
767 |
+
function new_access_token($token, $consumer, $verifier = null) {
|
768 |
+
// return a new access token attached to this consumer
|
769 |
+
// for the user associated with this token if the request token
|
770 |
+
// is authorized
|
771 |
+
// should also invalidate the request token
|
772 |
+
}
|
773 |
+
|
774 |
+
}
|
775 |
+
|
776 |
+
class Blogger_OAuthUtil {
|
777 |
+
public static function sslinstalled() {
|
778 |
+
return function_exists('openssl_sign');
|
779 |
+
}
|
780 |
+
|
781 |
+
public static function urlencode_rfc3986($input) {
|
782 |
+
if (is_array($input)) {
|
783 |
+
return array_map(array('Blogger_OAuthUtil', 'urlencode_rfc3986'), $input);
|
784 |
+
} else if (is_scalar($input)) {
|
785 |
+
return str_replace(
|
786 |
+
'+',
|
787 |
+
' ',
|
788 |
+
str_replace('%7E', '~', rawurlencode($input))
|
789 |
+
);
|
790 |
+
} else {
|
791 |
+
return '';
|
792 |
+
}
|
793 |
+
}
|
794 |
+
|
795 |
+
|
796 |
+
// This decode function isn't taking into consideration the above
|
797 |
+
// modifications to the encoding process. However, this method doesn't
|
798 |
+
// seem to be used anywhere so leaving it as is.
|
799 |
+
public static function urldecode_rfc3986($string) {
|
800 |
+
return urldecode($string);
|
801 |
+
}
|
802 |
+
|
803 |
+
// Utility function for turning the Authorization: header into
|
804 |
+
// parameters, has to do some unescaping
|
805 |
+
// Can filter out any non-oauth parameters if needed (default behaviour)
|
806 |
+
// May 28th, 2010 - method updated to tjerk.meesters for a speed improvement.
|
807 |
+
// see http://code.google.com/p/oauth/issues/detail?id=163
|
808 |
+
public static function split_header($header, $only_allow_oauth_parameters = true) {
|
809 |
+
$params = array();
|
810 |
+
if (preg_match_all('/('.($only_allow_oauth_parameters ? 'oauth_' : '').'[a-z_-]*)=(:?"([^"]*)"|([^,]*))/', $header, $matches)) {
|
811 |
+
foreach ($matches[1] as $i => $h) {
|
812 |
+
$params[$h] = Blogger_OAuthUtil::urldecode_rfc3986(empty($matches[3][$i]) ? $matches[4][$i] : $matches[3][$i]);
|
813 |
+
}
|
814 |
+
if (isset($params['realm'])) {
|
815 |
+
unset($params['realm']);
|
816 |
+
}
|
817 |
+
}
|
818 |
+
return $params;
|
819 |
+
}
|
820 |
+
|
821 |
+
// helper to try to sort out headers for people who aren't running apache
|
822 |
+
public static function get_headers() {
|
823 |
+
if (function_exists('apache_request_headers')) {
|
824 |
+
// we need this to get the actual Authorization: header
|
825 |
+
// because apache tends to tell us it doesn't exist
|
826 |
+
$headers = apache_request_headers();
|
827 |
+
|
828 |
+
// sanitize the output of apache_request_headers because
|
829 |
+
// we always want the keys to be Cased-Like-This and arh()
|
830 |
+
// returns the headers in the same case as they are in the
|
831 |
+
// request
|
832 |
+
$out = array();
|
833 |
+
foreach ($headers AS $key => $value) {
|
834 |
+
$key = str_replace(
|
835 |
+
" ",
|
836 |
+
"-",
|
837 |
+
ucwords(strtolower(str_replace("-", " ", $key)))
|
838 |
+
);
|
839 |
+
$out[$key] = $value;
|
840 |
+
}
|
841 |
+
} else {
|
842 |
+
// otherwise we don't have apache and are just going to have to hope
|
843 |
+
// that $_SERVER actually contains what we need
|
844 |
+
$out = array();
|
845 |
+
if( isset($_SERVER['CONTENT_TYPE']) )
|
846 |
+
$out['Content-Type'] = $_SERVER['CONTENT_TYPE'];
|
847 |
+
if( isset($_ENV['CONTENT_TYPE']) )
|
848 |
+
$out['Content-Type'] = $_ENV['CONTENT_TYPE'];
|
849 |
+
|
850 |
+
foreach ($_SERVER as $key => $value) {
|
851 |
+
if (substr($key, 0, 5) == "HTTP_") {
|
852 |
+
// this is chaos, basically it is just there to capitalize the first
|
853 |
+
// letter of every word that is not an initial HTTP and strip HTTP
|
854 |
+
// code from przemek
|
855 |
+
$key = str_replace(
|
856 |
+
" ",
|
857 |
+
"-",
|
858 |
+
ucwords(strtolower(str_replace("_", " ", substr($key, 5))))
|
859 |
+
);
|
860 |
+
$out[$key] = $value;
|
861 |
+
}
|
862 |
+
}
|
863 |
+
}
|
864 |
+
return $out;
|
865 |
+
}
|
866 |
+
|
867 |
+
// This function takes a input like a=b&a=c&d=e and returns the parsed
|
868 |
+
// parameters like this
|
869 |
+
// array('a' => array('b','c'), 'd' => 'e')
|
870 |
+
public static function parse_parameters( $input ) {
|
871 |
+
if (!isset($input) || !$input) return array();
|
872 |
+
|
873 |
+
$pairs = explode('&', $input);
|
874 |
+
|
875 |
+
$parsed_parameters = array();
|
876 |
+
foreach ($pairs as $pair) {
|
877 |
+
$split = explode('=', $pair, 2);
|
878 |
+
$parameter = Blogger_OAuthUtil::urldecode_rfc3986($split[0]);
|
879 |
+
$value = isset($split[1]) ? Blogger_OAuthUtil::urldecode_rfc3986($split[1]) : '';
|
880 |
+
|
881 |
+
if (isset($parsed_parameters[$parameter])) {
|
882 |
+
// We have already recieved parameter(s) with this name, so add to the list
|
883 |
+
// of parameters with this name
|
884 |
+
|
885 |
+
if (is_scalar($parsed_parameters[$parameter])) {
|
886 |
+
// This is the first duplicate, so transform scalar (string) into an array
|
887 |
+
// so we can add the duplicates
|
888 |
+
$parsed_parameters[$parameter] = array($parsed_parameters[$parameter]);
|
889 |
+
}
|
890 |
+
|
891 |
+
$parsed_parameters[$parameter][] = $value;
|
892 |
+
} else {
|
893 |
+
$parsed_parameters[$parameter] = $value;
|
894 |
+
}
|
895 |
+
}
|
896 |
+
return $parsed_parameters;
|
897 |
+
}
|
898 |
+
|
899 |
+
public static function build_http_query($params) {
|
900 |
+
if (!$params) return '';
|
901 |
+
|
902 |
+
// Urlencode both keys and values
|
903 |
+
$keys = Blogger_OAuthUtil::urlencode_rfc3986(array_keys($params));
|
904 |
+
$values = Blogger_OAuthUtil::urlencode_rfc3986(array_values($params));
|
905 |
+
$params = array_combine($keys, $values);
|
906 |
+
|
907 |
+
// Parameters are sorted by name, using lexicographical byte value ordering.
|
908 |
+
// Ref: Spec: 9.1.1 (1)
|
909 |
+
uksort($params, 'strcmp');
|
910 |
+
|
911 |
+
$pairs = array();
|
912 |
+
foreach ($params as $parameter => $value) {
|
913 |
+
if (is_array($value)) {
|
914 |
+
// If two or more parameters share the same name, they are sorted by their value
|
915 |
+
// Ref: Spec: 9.1.1 (1)
|
916 |
+
// June 12th, 2010 - changed to sort because of issue 164 by hidetaka
|
917 |
+
sort($value, SORT_STRING);
|
918 |
+
foreach ($value as $duplicate_value) {
|
919 |
+
$pairs[] = $parameter . '=' . $duplicate_value;
|
920 |
+
}
|
921 |
+
} else {
|
922 |
+
$pairs[] = $parameter . '=' . $value;
|
923 |
+
}
|
924 |
+
}
|
925 |
+
// For each parameter, the name is separated from the corresponding value by an '=' character (ASCII code 61)
|
926 |
+
// Each name-value pair is separated by an '&' character (ASCII code 38)
|
927 |
+
return implode('&', $pairs);
|
928 |
+
}
|
929 |
+
}
|
930 |
+
|
readme.txt
CHANGED
@@ -1,135 +1,275 @@
|
|
1 |
-
===
|
2 |
-
Contributors: wordpressdotorg, Otto42, Workshopshed, SergeyBiryukov, rmccue
|
3 |
-
Donate link:
|
4 |
-
Tags: importer, blogger
|
5 |
-
Requires at least: 3.0
|
6 |
-
Tested up to:
|
7 |
-
Stable tag: 0.
|
8 |
-
License: GPLv2 or later
|
9 |
-
|
10 |
-
Imports posts, comments, and categories (blogger tags) from a Blogger blog then migrates authors to
|
11 |
-
|
12 |
-
== Description ==
|
13 |
-
|
14 |
-
The Blogger Importer imports your blog data from a Blogger site into a WordPress.org installation.
|
15 |
-
|
16 |
-
= Items imported =
|
17 |
-
|
18 |
-
* Categories
|
19 |
-
* Posts (published, scheduled and draft)
|
20 |
-
* Comments (not spam)
|
21 |
-
|
22 |
-
|
23 |
-
|
24 |
-
|
25 |
-
*
|
26 |
-
|
27 |
-
|
28 |
-
|
29 |
-
|
30 |
-
|
31 |
-
|
32 |
-
|
33 |
-
|
34 |
-
|
35 |
-
|
36 |
-
|
37 |
-
|
38 |
-
|
39 |
-
|
40 |
-
|
41 |
-
|
42 |
-
|
43 |
-
|
44 |
-
|
45 |
-
|
46 |
-
|
47 |
-
|
48 |
-
|
49 |
-
|
50 |
-
|
51 |
-
|
52 |
-
|
53 |
-
|
54 |
-
|
55 |
-
|
56 |
-
|
57 |
-
|
58 |
-
|
59 |
-
|
60 |
-
|
61 |
-
|
62 |
-
|
63 |
-
|
64 |
-
|
65 |
-
|
66 |
-
|
67 |
-
|
68 |
-
|
69 |
-
|
70 |
-
|
71 |
-
|
72 |
-
|
73 |
-
|
74 |
-
|
75 |
-
|
76 |
-
|
77 |
-
|
78 |
-
|
79 |
-
|
80 |
-
|
81 |
-
|
82 |
-
|
83 |
-
|
84 |
-
|
85 |
-
|
86 |
-
|
87 |
-
|
88 |
-
|
89 |
-
|
90 |
-
|
91 |
-
|
92 |
-
|
93 |
-
|
94 |
-
|
95 |
-
|
96 |
-
|
97 |
-
|
98 |
-
|
99 |
-
|
100 |
-
|
101 |
-
|
102 |
-
|
103 |
-
|
104 |
-
|
105 |
-
|
106 |
-
|
107 |
-
|
108 |
-
|
109 |
-
|
110 |
-
|
111 |
-
|
112 |
-
|
113 |
-
|
114 |
-
|
115 |
-
|
116 |
-
|
117 |
-
|
118 |
-
|
119 |
-
|
120 |
-
|
121 |
-
|
122 |
-
|
123 |
-
=
|
124 |
-
|
125 |
-
|
126 |
-
|
127 |
-
|
128 |
-
|
129 |
-
|
130 |
-
|
131 |
-
=
|
132 |
-
|
133 |
-
|
134 |
-
|
135 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
=== Blogger Importer ===
|
2 |
+
Contributors: wordpressdotorg, Otto42, Workshopshed, SergeyBiryukov, rmccue
|
3 |
+
Donate link:
|
4 |
+
Tags: importer, blogger
|
5 |
+
Requires at least: 3.0
|
6 |
+
Tested up to: 4.1
|
7 |
+
Stable tag: 0.7
|
8 |
+
License: GPLv2 or later
|
9 |
+
|
10 |
+
Imports posts, images, comments, and categories (blogger tags) from a Blogger blog then migrates authors to WordPress users.
|
11 |
+
|
12 |
+
== Description ==
|
13 |
+
|
14 |
+
The Blogger Importer imports your blog data from a Google Blogger site into a WordPress.org installation.
|
15 |
+
|
16 |
+
= Items imported =
|
17 |
+
|
18 |
+
* Categories
|
19 |
+
* Posts (published, scheduled and draft)
|
20 |
+
* Comments (not spam)
|
21 |
+
* Images
|
22 |
+
|
23 |
+
= Items not imported =
|
24 |
+
|
25 |
+
* Pages
|
26 |
+
* Widgets/Widget Data
|
27 |
+
* Templates/Theme
|
28 |
+
* Comment and author Avatars
|
29 |
+
|
30 |
+
== Installation ==
|
31 |
+
|
32 |
+
1. Upload the `blogger-importer` folder to the `/wp-content/plugins/` directory
|
33 |
+
1. Activate the plugin through the 'Plugins' menu in WordPress
|
34 |
+
|
35 |
+
= Prerequisites =
|
36 |
+
|
37 |
+
The importer connects your server to the blogger server to copy across the posts. For this to work you need to have connectivity from the server to the internet and also have at least one of the remote access protocols enabled, e.g. curl, streams or fsockopen. You can use the Core Control plugin to test if these are working correctly. The importer connects to Google over a secure connection so OpenSSL needs to be enabled on your server.
|
38 |
+
The importer uses the SimplePie classes to read and process the data from blogger so you will need the php-xml module installed on your webserver.
|
39 |
+
|
40 |
+
= Preparation =
|
41 |
+
|
42 |
+
It is strongly recommended that you **disable all other plugins and caching** during the import.
|
43 |
+
|
44 |
+
This will ensure that the information transfers across as smoothly as possible and that posts and comments are correctly transferrred.
|
45 |
+
|
46 |
+
= How to use =
|
47 |
+
|
48 |
+
1. Blogger Importer is available from the WordPress Tools->Import screen.
|
49 |
+
1. Press Authorise
|
50 |
+
1. If you are not already logged into Google you will be asked to login
|
51 |
+
1. You will be asked to grant WordPress access to your Blogger information, to continue press Grant Access
|
52 |
+
1. You will be presented with a list of all your blogs
|
53 |
+
1. Select the appropriate blog and press the import button
|
54 |
+
1. Wait whilst the posts, comments and images are imported
|
55 |
+
1. Press the Set Authors button
|
56 |
+
1. Select the appropriate mapping for the authors
|
57 |
+
1. Review categories, posts and comments
|
58 |
+
|
59 |
+
You can now remove the importer plugin if you no longer need to use it.
|
60 |
+
|
61 |
+
== Frequently Asked Questions ==
|
62 |
+
|
63 |
+
= How do I re-import? =
|
64 |
+
|
65 |
+
From the list of blogs, press refresh blog list, this should update the counts and re-enable the import button. The importer is designed not to re-import the same posts. If you need to do a full re-import then delete the posts and then empty the trash before re-importing.
|
66 |
+
|
67 |
+
= Once I've imported the posts do I need to keep the plugin? =
|
68 |
+
|
69 |
+
No, you can remove the plugin once you've completed your migration.
|
70 |
+
|
71 |
+
= How do I know which posts were imported? =
|
72 |
+
|
73 |
+
Each of the posts loaded is tagged with a meta tags indicating where the posts were loaded from. The permalink will be set to the visible URL if the post was published or the internal ID if it was still a draft or scheduled post
|
74 |
+
|
75 |
+
* blogger_author
|
76 |
+
* blogger_blog
|
77 |
+
* blogger_permalink
|
78 |
+
|
79 |
+
= After importing there are a lot of categories =
|
80 |
+
|
81 |
+
Blogger does not distinguish between tags and categories so you will likely want to review what was imported and then use the categories to tags converter
|
82 |
+
|
83 |
+
= What about pages? =
|
84 |
+
|
85 |
+
This importer does not handle blogger pages, you will need to manually transfer them.
|
86 |
+
|
87 |
+
= What about images? =
|
88 |
+
|
89 |
+
This version of the importer imports these too, but you can disable this via a setting in the blogger-importer.php file. Tracking images of size 1x1 are not processed. If you with to specifically exclude other images you could code something for the image_filter function.
|
90 |
+
|
91 |
+
= What size are the images? =
|
92 |
+
|
93 |
+
The importer will attempt to download the a large version of the file if one is available. This is controlled by the setting "LARGE_IMAGE_SIZE" and defaults to a width of 1024. The display size of the images is the "medium" size of images as defined on Wordpress. You can change this in advance if you want to show a different size.
|
94 |
+
|
95 |
+
= How do I know what images are skipped? =
|
96 |
+
|
97 |
+
If you hover over the progress bar for images it will tell you how many images are skipped. To see the filenames of these images you will need to enable Wordpress debugging to log to file. See http://codex.wordpress.org/Debugging_in_WordPress
|
98 |
+
|
99 |
+
= What about future posts? =
|
100 |
+
|
101 |
+
The scheduled posts will be transferred and will be published as specified. However, Blogger and WordPress handle drafts differently, WordPress does not support dates on draft posts so you will need to use a plugin if you wish to plan your writing schedule.
|
102 |
+
|
103 |
+
= Are the permalinks the same? =
|
104 |
+
|
105 |
+
No, WordPress and Blogger handle the permalinks differently. However, it is possible to use the redirection plugin or your .htaccess file to map the old URLs across to the new URLs.
|
106 |
+
|
107 |
+
= My posts and comments moved across but some things are stripped out =
|
108 |
+
|
109 |
+
The importer uses the SimplePie classes to process the data, these in turn use a Simplepie_Sanitize class to remove potentially malicious code from the source data. If the php-xml module is not installed then this may result in your entire comment text being stripped out and the error "PHP Warning: DOMDocument not found, unable to use sanitizer" to appear in your logs.
|
110 |
+
|
111 |
+
= The dashboard is reporting that there are 0 comments in blogger =
|
112 |
+
|
113 |
+
This can occur if your blogger blog is set to private.
|
114 |
+
|
115 |
+
= The comments don't have avatars =
|
116 |
+
|
117 |
+
This is a know limitation of the data that is provided from Blogger. The Wordpress system uses Gravatar to provide the images for the comment avatars. This relies the email of the person making the comment. Blogger does not provide the email address in the data feed so Wordpress does not display the correct images. You can manually update or script change to the comment email addresses to work around this issue.
|
118 |
+
|
119 |
+
= How do I diagnose communication issues? =
|
120 |
+
|
121 |
+
If you've got issues with the blogger importer talking to Google then you can use the Core Control plugin to first check that your communication is working with the HTTP Module and if that is all ok then check the HTTP Logging Module to see if any of the remote calls to Google are returning errors. Also the OAuth Playground is a good way to test your access to the blogger data. http://googlecodesamples.com/oauth_playground/
|
122 |
+
|
123 |
+
= I'm getting timeouts connecting to Google =
|
124 |
+
|
125 |
+
Try the core control plugin as mentioned above. Also you try changing the constant REMOTE_TIMEOUT in the blogger-importer.php file to be a larger number of seconds.
|
126 |
+
|
127 |
+
= Why does it keep stopping? =
|
128 |
+
|
129 |
+
The imported is not expect to stop, so this could be to an incompatibility with another plugin, disable any other plugins and see if the problem persists. Also check your error log to see if any error messages have been reported.
|
130 |
+
|
131 |
+
= It does not seem to be processing the images =
|
132 |
+
|
133 |
+
The most common reasons for this are lack of memory and timeouts, these should appear in your error log. Also check you've not run out of disk space on your server. Because Wordpress stores the files in multiple resolutions one image might take up as much as 250kb spread across 5 files of different sizes.
|
134 |
+
|
135 |
+
= How do I make the images bigger or smaller? / My images are fuzzy =
|
136 |
+
|
137 |
+
The importer will attempt to download a large version of images but it displays them on the blog at the medium size. If you go into your settings->media options then you can display a different size "medium" image by default. You can't make this bigger than the file that has been downloaded which is where the next setting comes in.
|
138 |
+
|
139 |
+
The default size for the large images is 1024, you can change this to an even larger size by changing the following line in the blogger-import.php file.
|
140 |
+
|
141 |
+
const LARGE_IMAGE_SIZE = '1024';
|
142 |
+
|
143 |
+
The file downloaded won't be bigger than the origional file so if it was only 800x600 to start with then it won't be any bigger than that.
|
144 |
+
|
145 |
+
If your origional blog has hardcoded width and height values that are larger than the medium size settings then that might result in your images becoming fuzzy.
|
146 |
+
|
147 |
+
= I've run out of disk space processing the images =
|
148 |
+
|
149 |
+
The importer is designed to download the high resolution images where they are available. You can either disable the downloading of images or you can change the constant LARGE_IMAGE_SIZE string in the blogger-importer.php file to swap the links with a smaller image.
|
150 |
+
|
151 |
+
== Screenshots ==
|
152 |
+
|
153 |
+
1. Import in progress
|
154 |
+
2. Custom Fields added to Posts, Attachements and Comments
|
155 |
+
|
156 |
+
== Reference ==
|
157 |
+
|
158 |
+
* https://developers.google.com/blogger/docs/1.0/developers_guide_php
|
159 |
+
* https://developers.google.com/gdata/articles/oauth
|
160 |
+
* http://www.simplepie.org/
|
161 |
+
|
162 |
+
The following were referenced for implementing the images and links
|
163 |
+
|
164 |
+
* http://wordpress.org/extend/plugins/remote-images-grabber
|
165 |
+
* http://notions.okuda.ca/wordpress-plugins/blogger-image-import/
|
166 |
+
* http://wordpress.org/extend/plugins/cache-images/
|
167 |
+
* http://wordpress.org/extend/plugins/tumblr-importer/
|
168 |
+
* http://core.trac.wordpress.org/ticket/14525
|
169 |
+
* http://wpengineer.com/1735/easier-better-solutions-to-get-pictures-on-your-posts/
|
170 |
+
* http://www.velvetblues.com/web-development-blog/wordpress-plugin-update-urls/
|
171 |
+
* http://wordpress.stackexchange.com/questions//media-sideload-image-file-name
|
172 |
+
* http://wp.tutsplus.com/tutorials/plugins/a-guide-to-the-wordpress-http-api-the-basics/
|
173 |
+
|
174 |
+
== Known Issues ==
|
175 |
+
|
176 |
+
* Some users have reported that their IFrames are stripped out of the post content.
|
177 |
+
* Requests for better performance of larger transfers and tranfers of images
|
178 |
+
* Drop the check for OpenSSL and Blogger_OAuthSignatureMethod_RSA_SHA1 is not used by any other code in the plugin and can probably just be removed.
|
179 |
+
* Review of behavior when it re-imports, partiularly are the counts correct
|
180 |
+
* Review using get_posts or get_comments with the appropriate parameters to get the counts and exists instead of using SQL
|
181 |
+
* Incorrect notice, PHP Notice: The data could not be converted to UTF-8. You MUST have either the iconv or mbstring extension installed. This occurs even when Iconv is installed, could be related to Blogger reporting 0 comments
|
182 |
+
* When the importer is running it's not possible to stop it using the stop button
|
183 |
+
* Blogger's count of comments include those not linked to a post e.g. the post has been deleted.
|
184 |
+
|
185 |
+
== Filters and Actions ==
|
186 |
+
|
187 |
+
These actions and filters have been added so that you can extend the functionality of the importer without needing to modify the code.
|
188 |
+
|
189 |
+
Action - import_start - This is run when the import starts processing the records for a new blog
|
190 |
+
|
191 |
+
Action - import_done - This is run when the import finishes processing the records for a blog.
|
192 |
+
|
193 |
+
Filter - blogger_importer_congrats - Passes the list of options shown to the user when the blog is complete, options can be added or removed.
|
194 |
+
|
195 |
+
== Changelog ==
|
196 |
+
|
197 |
+
= 0.8 =
|
198 |
+
* Fixed issue with the authors form not showing a the list of authors for a blog
|
199 |
+
* Simplified check for duplicate comments
|
200 |
+
* Code simplified for get_authors and get_author_form
|
201 |
+
* Fixed issue with wpdb prepare and integer keys by switching to a sub select query
|
202 |
+
* Make comment handling more robust
|
203 |
+
* Simplified functions to reduce messages in the log
|
204 |
+
|
205 |
+
= 0.7 =
|
206 |
+
* Fixed issue with drafts not being imported in the right state
|
207 |
+
* Added extra error handling for get_oauth_link to stop blank tokens being sent to the form
|
208 |
+
* Restructured code to keep similar steps in single function and to allow testing of components to be done
|
209 |
+
* Re-incorporated the "congrats" function and provided a sensible list of what to do next
|
210 |
+
* Add a geo_public flag to posts with geotags
|
211 |
+
* Dropped _normalize_tag after confirming that it's handled by SimplePie
|
212 |
+
* Added image handling http://core.trac.wordpress.org/ticket/4010
|
213 |
+
* Added setting author on images
|
214 |
+
* Added error handling in get_oauth_link() as suggested by daniel_henrique ref http://core.trac.wordpress.org/ticket/21163
|
215 |
+
* Added a check for OpenSSL as suggested by digitalsensus
|
216 |
+
* Fixed issue with SimplePie santizer not getting set in Wordpress 3.5
|
217 |
+
* Added filter for the congrats function 'blogger_importer_congrats' so other plugins can add in new options
|
218 |
+
* Converted manual HTML table to WP_LIST_TABLE
|
219 |
+
* Moved inline Javascript to separate file to aid debugging and testing
|
220 |
+
* Wrapped data sent to Javascript in I18n functions.
|
221 |
+
* Fixed timeout error in the Javascript, timeouts were not being used.
|
222 |
+
* Supress post revisions when importing so that DB does not grow
|
223 |
+
* Added processing of internal links
|
224 |
+
* Added uninstall.php to remove options on uninstall
|
225 |
+
* Added a timeout value to all of the wp_remote_get calls as people have reported timeout issues
|
226 |
+
* Added a setting to control the large images downloaded from blogger.
|
227 |
+
* Stopped logging all the post and comment IDs in arrays and storing in option this improved the importing of very large blogs
|
228 |
+
* Fixed issue with comment_author_IP notice
|
229 |
+
* Code restructuring to use classes for blog objects
|
230 |
+
* Changed AJAX calls to use technique described here http://codex.wordpress.org/AJAX_in_Plugins#Ajax_on_the_Administration_Side
|
231 |
+
* Added AdminURL to the greet function rather than hardcoded path
|
232 |
+
* Defaulted to turn off post pingbacks
|
233 |
+
* Fix to stop it counting pingbacks, issue reported by realdoublebee
|
234 |
+
* Retrofitted Security enhancement from 0.6, nonce added to form buttons on main screen
|
235 |
+
* Security enhancement, nonce added to form button on authors screen
|
236 |
+
* Updated POT file
|
237 |
+
* Greek Translation from Stergatou Eleni http://buddypress.org/community/members/lenasterg/
|
238 |
+
|
239 |
+
= 0.6 =
|
240 |
+
* Security enhancement, nonce added to form button on main screen
|
241 |
+
|
242 |
+
= 0.5 =
|
243 |
+
* Merged in fix by SergeyBiryukov http://core.trac.wordpress.org/ticket/16012
|
244 |
+
* Merged in rmccue change to get_total_results to also use SimplePie from http://core.trac.wordpress.org/attachment/ticket/7652/7652-blogger.diff
|
245 |
+
* Reviewed in rmccue's changes in http://core.trac.wordpress.org/attachment/ticket/7652/7652-separate.diff issues with date handling functions so skipped those
|
246 |
+
* Moved SimplePie functions in new class WP_SimplePie_Blog_Item incorporating get_draft_status and get_updated and convert date
|
247 |
+
* Tested comments from source blog GMT-8, destination London (currently GMT-1), comment dates transferred correctly.
|
248 |
+
* Fixed typo in oauth_get
|
249 |
+
* Added screen_icon() to all pages
|
250 |
+
* Added GeoTags as per spec on http://codex.wordpress.org/Geodata
|
251 |
+
* Change by Otto42, rmccue to use Simplepie XML processing rather than Atomparser, http://core.trac.wordpress.org/ticket/14525 ref: http://core.trac.wordpress.org/attachment/ticket/7652/7652-blogger.diff
|
252 |
+
this also fixes http://core.trac.wordpress.org/ticket/15560
|
253 |
+
* Change by Otto42 to use OAuth rather than AuthSub authentication, should make authentication more reliable
|
254 |
+
* Fix by Andy from Workshopshed to load comments and nested comments correctly
|
255 |
+
* Fix by Andy from Workshopshed to correctly pass the blogger start-index and max-results parameters to oAuth functions and to process more than one batch http://core.trac.wordpress.org/ticket/19096
|
256 |
+
* Fix by Andy from Workshopshed error about incorrect enqueuing of scripts also changed styles to work the same
|
257 |
+
* Change by Andy from Workshopshed testing in debug mode and wrapped ajax return into a function to suppress debug messages
|
258 |
+
* Fix by Andy from Workshopshed notices for undefined variables.
|
259 |
+
* Change by Andy from Workshopshed Added tooltip to results table to show numbers of posts and comments skipped (duplicates / missing key)
|
260 |
+
* Fix by Andy from Workshopshed incorrectly checking for duplicates based on only the date and username, this gave false positives when large numbers of comments, particularly anonymous ones.
|
261 |
+
|
262 |
+
= 0.4 =
|
263 |
+
* Fix for tracking images being added by Blogger to non-authenticated feeds http://core.trac.wordpress.org/ticket/17623
|
264 |
+
|
265 |
+
= 0.3 =
|
266 |
+
* Bugfix for 403 Invalid AuthSub Token http://core.trac.wordpress.org/ticket/14629
|
267 |
+
|
268 |
+
= 0.1 =
|
269 |
+
* Initial release
|
270 |
+
|
271 |
+
== Upgrade Notice ==
|
272 |
+
|
273 |
+
= 0.8 =
|
274 |
+
|
275 |
+
Some bug fixes and simplified code see change log.
|
uninstall.php
ADDED
@@ -0,0 +1,21 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1 |
+
<?php
|
2 |
+
|
3 |
+
/**
|
4 |
+
* @author Andy Clark
|
5 |
+
* @copyright 2013
|
6 |
+
*
|
7 |
+
* Tidy up things left over by the plugin
|
8 |
+
* Keep this as simple as possible so we don't stall the uninstallation process
|
9 |
+
*/
|
10 |
+
|
11 |
+
delete_option('blogger_importer');
|
12 |
+
delete_option('blogger_importer_connector');
|
13 |
+
$blogopt = true;
|
14 |
+
for ($i = 1; $blogopt ; $i++) {
|
15 |
+
$blogopt = get_option('blogger_importer_blog_'.$i);
|
16 |
+
if (is_array($blogopt))
|
17 |
+
delete_option('blogger_importer_blog_'.$i);
|
18 |
+
}
|
19 |
+
|
20 |
+
|
21 |
+
?>
|