如何使用 WordPress 作为 Laravel 应用程序的后端

发布于 作者:

How to use WordPress as a backend for a Laravel Application image

上周我重新发布了 Laravel 新闻,新网站运行在 Laravel 上,后端使用 WordPress。过去两年我一直使用 WordPress,并且越来越喜欢它提供的功能。发布体验、媒体管理器、移动应用程序和 Jetpack 用于跟踪统计信息。

我还没有准备好放弃这些功能,而且我没有时间构建自己的系统,所以我决定保留 WordPress 并使用 API 插件来提取我需要的所有内容,然后将它们存储在我的 Laravel 应用程序中。在本教程中,我想概述我是如何设置这一切的。

请记住,有几种不同的方法可以做到这一点,您可以查看 WordPress 和 Laravel 文章,以获取其他人创建的用于解决同一任务的第三方软件包和资源列表。主要选项是连接 WordPress 数据库,从 API 拉取或将数据从一个系统同步到另一个系统。

我选择同步选项,因为我希望一切都分开。在研究时,我偶然发现了一个名为 WordPressToLaravel 的软件包,它可以做到这一点,但是,它只适用于 wordpress.com 托管网站。

我决定为使用自托管 WordPress 安装创建自己的系统。这种方法需要在前期做更多工作,但它允许我的网站成为一个典型的 Laravel 应用程序,我可以轻松地根据需要进行改进和扩展。

在我的内容同步完成后,我通过 Laravel Scheduler 将其设置为定期任务,因此它是完全自动化的。

我的设置也是完全根据我的需求定制的。我只想拉取帖子、类别和标签。页面和其他部分都由静态 Blade 文件或其他 API 驱动。

让我们来看看它是如何工作的。

WordPress API

WordPress 并没有自带 API;但是,社区构建了一个名为 WP Rest API 的插件,允许任何博客拥有一个 jSON API。

在我的情况下,我正在进行不需要任何身份验证的只读请求。这使得读取和获取数据变得容易,并且简化了许多代码。

这是我用来获取帖子列表的基本类

class WpApi
{
protected $url = 'http://site.com/wp-json/wp/v2/';
 
public function importPosts($page = 1)
{
$posts = collect($this->getJson($this->url . 'posts/?_embed&filter[orderby]=modified&page=' . $page));
foreach ($posts as $post) {
$this->syncPost($post);
}
}
 
protected function getJson($url)
{
$response = file_get_contents($url, false);
return json_decode( $response );
}
}

现在,当您调用 WpAPI->importPosts() 时,它将获取第一页结果。查询字符串值得一提,因为它包含一些特殊的子句。第一个是 _embed,它将嵌入所有额外的元数据,例如图像嵌入、类别和标签。下一个是按最后修改时间排序的过滤器,这样,当您在 WordPress 上编辑帖子时,它将出现在结果的第一页,这意味着它将被重新同步。

接下来,我们需要能够将帖子与我们的数据库同步。这是我的设置方式

protected function syncPost($data)
{
$found = Post::where('wp_id', $data->id)->first();
 
if (! $found) {
return $this->createPost($data);
}
 
if ($found and $found->updated_at->format("Y-m-d H:i:s") < $this->carbonDate($data->modified)->format("Y-m-d H:i:s")) {
return $this->updatePost($found, $data);
}
}
 
protected function carbonDate($date)
{
return Carbon::parse($date);
}

在此步骤中,我在自己的 Posts 表中添加了一个 wp_id 字段,这样我的本地数据库和 WordPress 之间就有一对一的关系。

接下来,我只需检查它是否存在,如果不存在,就创建它。否则,如果它自上次同步后被修改,则更新它。

createPostupdatePost 是典型的 Laravel Eloquent 插入或更新。这是创建的代码

protected function createPost($data)
{
$post = new Post();
$post->id = $data->id;
$post->wp_id = $data->id;
$post->user_id = $this->getAuthor($data->_embedded->author);
$post->title = $data->title->rendered;
$post->slug = $data->slug;
$post->featured_image = $this->featuredImage($data->_embedded);
$post->featured = ($data->sticky) ? 1 : null;
$post->excerpt = $data->excerpt->rendered;
$post->content = $data->content->rendered;
$post->format = $data->format;
$post->status = 'publish';
$post->publishes_at = $this->carbonDate($data->date);
$post->created_at = $this->carbonDate($data->date);
$post->updated_at = $this->carbonDate($data->modified);
$post->category_id = $this->getCategory($data->_embedded->{"wp:term"});
$post->save();
$this->syncTags($post, $data->_embedded->{"wp:term"});
return $post;
}

如果您仔细观察,就会发现一些特殊情况,例如作者、特色图片、类别和标签。这些来自原始查询字符串中的 _embed,同步这些数据只是做与之前相同的事情。

public function featuredImage($data)
{
if (property_exists($data, "wp:featuredmedia")) {
$data = head($data->{"wp:featuredmedia"});
if (isset($data->source_url)) {
return $data->source_url;
}
}
return null;
}
 
public function getCategory($data)
{
$category = collect($data)->collapse()->where('taxonomy', 'category')->first();
$found = Category::where('wp_id', $category->id)->first();
if ($found) {
return $found->id;
}
$cat = new Category();
$cat->id = $category->id;
$cat->wp_id = $category->id;
$cat->name = $category->name;
$cat->slug = $category->slug;
$cat->description = '';
$cat->save();
return $cat->id;
}
 
private function syncTags(Post $post, $tags)
{
$tags = collect($tags)->collapse()->where('taxonomy', 'post_tag')->pluck('name')->toArray();
if (count($tags) > 0) {
$post->setTags($tags);
}
}

对于类别,我提取了帖子分配的第一个类别,因为我只希望每个类别有一个帖子,然后在同步标签中,我使用了 Cartalyst 的标签包

创建计划任务

完成导入的最后一步是构建一个计划任务来自动拉取帖子。我通过 Artisan 创建了一个名为 Importer 的命令

php artisan make:console Importer

然后在 handle 方法中

$page = ($this->argument('page')) ? $this->argument('page') : 1;
$this->wpApi->importPosts($page);

最后,在 Console/Kernel 中将其设置为每分钟运行一次

$schedule->command('import:wordpress')
->everyMinute();

现在,每天的每分钟,它都会尝试同步数据,并创建、更新或忽略帖子。

更进一步

这是我设置此功能的基本概述,它只触及了可以做到的所有事情的表面。例如,我正在对网站内的所有 DB 查询进行大量缓存,并且在此同步过程中,如果更新了某些内容,则会清除相关的缓存。

我希望这能帮助您了解,使用 WordPress 作为后端并不难管理,同时还能为您提供许多好处,例如从移动设备创建帖子、使用媒体管理器,甚至使用 Jetpack 以 Markdown 形式编写。

Eric L. Barnes photo

Eric 是 Laravel 新闻的创建者,自 2012 年以来一直在报道 Laravel。

Cube

Laravel 时事通讯

加入 40,000 多名其他开发人员,绝不错过新的技巧、教程等。

Laravel Forge logo

Laravel Forge

轻松创建和管理您的服务器,并在几秒钟内部署您的 Laravel 应用程序。

Laravel Forge
Tinkerwell logo

Tinkerwell

Laravel 开发人员必备的代码运行器。使用 AI、自动补全和本地和生产环境的即时反馈进行调试。

Tinkerwell
No Compromises logo

绝不妥协

Joel 和 Aaron,来自“绝不妥协”播客的两名经验丰富的开发者,现在可以为您的 Laravel 项目提供服务。 ⬧ 固定费率:每月 7500 美元。 ⬧ 无需冗长的销售流程。 ⬧ 无需签订合同。 ⬧ 100%退款保证。

绝不妥协
Kirschbaum logo

Kirschbaum

提供创新和稳定性,确保您的 Web 应用程序取得成功。

Kirschbaum
Shift logo

Shift

运行旧版本的 Laravel 吗?即时自动 Laravel 升级和代码现代化,让您的应用程序保持新鲜。

Shift
Bacancy logo

Bacancy

让您的项目充满活力,只需每月 2500 美元,即可获得具有 4-6 年经验的经验丰富的 Laravel 开发人员。获得 160 小时的专职专业知识和 15 天无风险试用。立即预约电话!

Bacancy
Lucky Media logo

Lucky Media

现在就来碰碰运气 - Laravel 开发的理想选择,拥有超过十年的经验!

Lucky Media
Lunar: Laravel E-Commerce logo

Lunar:Laravel 电子商务

Laravel 的电子商务。一个开源软件包,将现代无头电子商务功能的力量带到 Laravel。

Lunar:Laravel 电子商务
LaraJobs logo

LaraJobs

官方 Laravel 工作板

LaraJobs
SaaSykit: Laravel SaaS Starter Kit logo

SaaSykit:Laravel SaaS 启动工具包

SaaSykit 是一个 Laravel SaaS 启动工具包,它包含运行现代 SaaS 所需的所有功能。支付、精美结账、管理面板、用户仪表板、身份验证、现成组件、统计信息、博客、文档等。

SaaSykit:Laravel SaaS 启动工具包
Rector logo

Rector

您的 Laravel 无缝升级合作伙伴,降低成本,加速创新,帮助企业取得成功。

Rector
MongoDB logo

MongoDB

通过将 MongoDB 和 Laravel 强力集成,增强您的 PHP 应用程序,使开发人员能够轻松高效地构建应用程序。支持事务、搜索、分析和移动用例,同时使用熟悉的 Eloquent API。了解 MongoDB 的灵活、现代数据库如何改变您的 Laravel 应用程序。

MongoDB
Maska is a Simple Zero-dependency Input Mask Library image

Maska 是一个简单的零依赖输入掩码库

阅读文章
Add Swagger UI to Your Laravel Application image

将 Swagger UI 添加到您的 Laravel 应用程序

阅读文章
Assert the Exact JSON Structure of a Response in Laravel 11.19 image

在 Laravel 11.19 中断言响应的精确 JSON 结构

阅读文章
Build SSH Apps with PHP and Laravel Prompts image

使用 PHP 和 Laravel 提示构建 SSH 应用程序

阅读文章
Building fast, fuzzy site search with Laravel and Typesense image

使用 Laravel 和 Typesense 构建快速、模糊的网站搜索

阅读文章
Add Comments to your Laravel Application with the Commenter Package image

使用 Commenter 包在 Laravel 应用程序中添加评论

阅读文章