Merge branch 'main' into event_sources
Some checks failed
PHP Code Checker / PHP Code Checker (pull_request) Failing after 48s
PHPUnit / PHPUnit – PHP 7.4 (pull_request) Failing after 56s
PHPUnit / PHPUnit – PHP 8.0 (pull_request) Failing after 57s
PHPUnit / PHPUnit – PHP 8.1 (pull_request) Failing after 51s
PHPUnit / PHPUnit – PHP 8.2 (pull_request) Failing after 53s
PHPUnit / PHPUnit – PHP 8.3 (pull_request) Failing after 50s
PHPUnit / PHPUnit – PHP 8.4 (pull_request) Failing after 51s
Some checks failed
PHP Code Checker / PHP Code Checker (pull_request) Failing after 48s
PHPUnit / PHPUnit – PHP 7.4 (pull_request) Failing after 56s
PHPUnit / PHPUnit – PHP 8.0 (pull_request) Failing after 57s
PHPUnit / PHPUnit – PHP 8.1 (pull_request) Failing after 51s
PHPUnit / PHPUnit – PHP 8.2 (pull_request) Failing after 53s
PHPUnit / PHPUnit – PHP 8.3 (pull_request) Failing after 50s
PHPUnit / PHPUnit – PHP 8.4 (pull_request) Failing after 51s
This commit is contained in:
commit
e494aba31e
13 changed files with 25 additions and 524 deletions
|
@ -115,11 +115,6 @@ jobs:
|
||||||
env:
|
env:
|
||||||
PHP_VERSION: ${{ matrix.php-version }}
|
PHP_VERSION: ${{ matrix.php-version }}
|
||||||
|
|
||||||
- name: Run Integration tests for EventPrime
|
|
||||||
run: cd /workspace/Event-Federation/wordpress-event-bridge-for-activitypub/ && ./vendor/bin/phpunit --filter=eventprime
|
|
||||||
env:
|
|
||||||
PHP_VERSION: ${{ matrix.php-version }}
|
|
||||||
|
|
||||||
- name: Run Integration tests for Event Organiser
|
- name: Run Integration tests for Event Organiser
|
||||||
run: cd /workspace/Event-Federation/wordpress-event-bridge-for-activitypub/ && ./vendor/bin/phpunit --filter=event_organiser
|
run: cd /workspace/Event-Federation/wordpress-event-bridge-for-activitypub/ && ./vendor/bin/phpunit --filter=event_organiser
|
||||||
env:
|
env:
|
||||||
|
|
|
@ -5,13 +5,7 @@ All notable changes to this project will be documented in this file.
|
||||||
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
|
The format is based on [Keep a Changelog](https://keepachangelog.com/en/1.1.0/),
|
||||||
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
|
||||||
|
|
||||||
## Unreleased
|
## [0.3.3] - 2024-12-19
|
||||||
|
|
||||||
### Added
|
|
||||||
|
|
||||||
* Add custom summary via shortcodes
|
|
||||||
|
|
||||||
## [0.3.2] - 2024-12-12
|
|
||||||
|
|
||||||
* Initial release on WordPress.org
|
* Initial release on WordPress.org
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
**Tags:** events, fediverse, activitypub, calendar
|
**Tags:** events, fediverse, activitypub, calendar
|
||||||
**Requires at least:** 6.5
|
**Requires at least:** 6.5
|
||||||
**Tested up to:** 6.7
|
**Tested up to:** 6.7
|
||||||
**Stable tag:** 0.3.2
|
**Stable tag:** 0.3.3
|
||||||
**Requires PHP:** 7.4
|
**Requires PHP:** 7.4
|
||||||
**License:** AGPL-3.0-or-later
|
**License:** AGPL-3.0-or-later
|
||||||
**License URI:** https://www.gnu.org/licenses/agpl-3.0.html
|
**License URI:** https://www.gnu.org/licenses/agpl-3.0.html
|
||||||
|
@ -63,7 +63,6 @@ This plugin depends on the [ActivityPub plugin](https://wordpress.org/plugins/ac
|
||||||
* [Eventin](https://de.wordpress.org/plugins/wp-event-solution/)
|
* [Eventin](https://de.wordpress.org/plugins/wp-event-solution/)
|
||||||
* [Modern Events Calendar Lite](https://webnus.net/modern-events-calendar/)
|
* [Modern Events Calendar Lite](https://webnus.net/modern-events-calendar/)
|
||||||
* [GatherPress](https://gatherpress.org/)
|
* [GatherPress](https://gatherpress.org/)
|
||||||
* [EventPrime – Events Calendar, Bookings and Tickets](https://wordpress.org/plugins/eventprime-event-calendar-management/)
|
|
||||||
* [Event Organiser](https://wordpress.org/plugins/event-organiser/)
|
* [Event Organiser](https://wordpress.org/plugins/event-organiser/)
|
||||||
|
|
||||||
## Configuration ##
|
## Configuration ##
|
||||||
|
@ -102,7 +101,7 @@ We're always interested in your feedback. Feel free to reach out to us via [E-Ma
|
||||||
|
|
||||||
## Changelog ##
|
## Changelog ##
|
||||||
|
|
||||||
### [0.3.2] 2024-12-12 ###
|
### [0.3.3] 2024-12-19 ###
|
||||||
|
|
||||||
* Initial release on https://wordpress.org/
|
* Initial release on https://wordpress.org/
|
||||||
|
|
||||||
|
|
|
@ -3,7 +3,7 @@
|
||||||
* Plugin Name: Event Bridge for ActivityPub
|
* Plugin Name: Event Bridge for ActivityPub
|
||||||
* Description: Integrating popular event plugins with the ActivityPub plugin.
|
* Description: Integrating popular event plugins with the ActivityPub plugin.
|
||||||
* Plugin URI: https://event-federation.eu/
|
* Plugin URI: https://event-federation.eu/
|
||||||
* Version: 0.3.2.7
|
* Version: 0.3.3
|
||||||
* Author: André Menrath
|
* Author: André Menrath
|
||||||
* Author URI: https://graz.social/@linos
|
* Author URI: https://graz.social/@linos
|
||||||
* Text Domain: event-bridge-for-activitypub
|
* Text Domain: event-bridge-for-activitypub
|
||||||
|
|
|
@ -1,78 +0,0 @@
|
||||||
<?php
|
|
||||||
/**
|
|
||||||
* ActivityPub Transformer for the plugin EventPrime.
|
|
||||||
*
|
|
||||||
* @package Event_Bridge_For_ActivityPub
|
|
||||||
* @license AGPL-3.0-or-later
|
|
||||||
*/
|
|
||||||
|
|
||||||
namespace Event_Bridge_For_ActivityPub\ActivityPub\Transformer;
|
|
||||||
|
|
||||||
// Exit if accessed directly.
|
|
||||||
defined( 'ABSPATH' ) || exit; // @codeCoverageIgnore
|
|
||||||
|
|
||||||
use Activitypub\Activity\Extended_Object\Place;
|
|
||||||
use Event_Bridge_For_ActivityPub\ActivityPub\Transformer\Event;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* ActivityPub Transformer for VS Event
|
|
||||||
*
|
|
||||||
* @since 1.0.0
|
|
||||||
*/
|
|
||||||
final class EventPrime extends Event {
|
|
||||||
/**
|
|
||||||
* Get the end time from the event object.
|
|
||||||
*/
|
|
||||||
public function get_end_time(): ?string {
|
|
||||||
$timestamp = get_post_meta( $this->wp_object->ID, 'em_end_date', true );
|
|
||||||
if ( $timestamp ) {
|
|
||||||
return \gmdate( 'Y-m-d\TH:i:s\Z', $timestamp );
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the end time from the event object.
|
|
||||||
*/
|
|
||||||
public function get_start_time(): string {
|
|
||||||
$timestamp = get_post_meta( $this->wp_object->ID, 'em_start_date', true );
|
|
||||||
if ( $timestamp ) {
|
|
||||||
return \gmdate( 'Y-m-d\TH:i:s\Z', $timestamp );
|
|
||||||
} else {
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get location from the event object.
|
|
||||||
*/
|
|
||||||
public function get_location(): ?Place {
|
|
||||||
$venue_term_id = get_post_meta( $this->wp_object->ID, 'em_venue', true );
|
|
||||||
if ( ! $venue_term_id ) {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
$venue = wp_get_post_terms( $this->wp_object->ID, 'em_venue' );
|
|
||||||
|
|
||||||
if ( empty( $venue ) ) {
|
|
||||||
return null;
|
|
||||||
} else {
|
|
||||||
$venue = $venue[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
$place = new Place();
|
|
||||||
|
|
||||||
$place->set_name( $venue->name );
|
|
||||||
$place->set_content( $venue->description );
|
|
||||||
|
|
||||||
$address = get_term_meta( $venue->term_id, 'em_address', true );
|
|
||||||
$display_address = get_term_meta( $venue->term_id, 'em_display_address_on_frontend', true );
|
|
||||||
|
|
||||||
if ( $address && $display_address ) {
|
|
||||||
$place->set_address( get_term_meta( $venue->term_id, 'em_address', true ) );
|
|
||||||
}
|
|
||||||
|
|
||||||
return $place;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -29,7 +29,7 @@ class General_Admin_Notices {
|
||||||
*/
|
*/
|
||||||
const ACTIVITYPUB_PLUGIN_URL = 'https://wordpress.org/plugins/activitypub';
|
const ACTIVITYPUB_PLUGIN_URL = 'https://wordpress.org/plugins/activitypub';
|
||||||
|
|
||||||
const EVENT_BRIDGE_FOR_ACTIVITYPUB_SUPPORTED_EVENT_PLUGINS_URL = 'https://code.event-federation.eu/Event-Federation/wordpress-event-bridge-for-activitypub#events-plugin-that-will-be-supported-at-first';
|
const EVENT_BRIDGE_FOR_ACTIVITYPUB_SUPPORTED_EVENT_PLUGINS_URL = 'https://code.event-federation.eu/Event-Federation/wordpress-event-bridge-for-activitypub#supported-event-plugins';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Allowed HTML for admin notices.
|
* Allowed HTML for admin notices.
|
||||||
|
@ -38,8 +38,9 @@ class General_Admin_Notices {
|
||||||
*/
|
*/
|
||||||
const ALLOWED_HTML = array(
|
const ALLOWED_HTML = array(
|
||||||
'a' => array(
|
'a' => array(
|
||||||
'href' => true,
|
'href' => true,
|
||||||
'title' => true,
|
'title' => true,
|
||||||
|
'target' => true,
|
||||||
),
|
),
|
||||||
'br',
|
'br',
|
||||||
'i',
|
'i',
|
||||||
|
@ -90,11 +91,11 @@ class General_Admin_Notices {
|
||||||
/* translators: 1: An URL to the list of supported event plugins. */
|
/* translators: 1: An URL to the list of supported event plugins. */
|
||||||
_x(
|
_x(
|
||||||
'The Plugin <i>Event Bridge for ActivityPub</i> is of no use, because you do not have installed and activated a supported Event Plugin.
|
'The Plugin <i>Event Bridge for ActivityPub</i> is of no use, because you do not have installed and activated a supported Event Plugin.
|
||||||
<br> For a list of supported Event Plugins see <a href="%1$s">here</a>.',
|
<br> For a list of supported Event Plugins see <a href="%1$s" target="_blank">here</a>.',
|
||||||
'admin notice',
|
'admin notice',
|
||||||
'event-bridge-for-activitypub'
|
'event-bridge-for-activitypub'
|
||||||
),
|
),
|
||||||
esc_html( self::EVENT_BRIDGE_FOR_ACTIVITYPUB_SUPPORTED_EVENT_PLUGINS_URL )
|
esc_url( self::EVENT_BRIDGE_FOR_ACTIVITYPUB_SUPPORTED_EVENT_PLUGINS_URL )
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -116,7 +116,10 @@ class Health_Check {
|
||||||
if ( ! $event_post_type ) {
|
if ( ! $event_post_type ) {
|
||||||
$active_event_plugins = Setup::get_instance()->get_active_event_plugins();
|
$active_event_plugins = Setup::get_instance()->get_active_event_plugins();
|
||||||
$active_event_plugin = reset( $active_event_plugins );
|
$active_event_plugin = reset( $active_event_plugins );
|
||||||
$event_post_type = $active_event_plugin->get_post_type();
|
if ( ! $active_event_plugin ) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
$event_post_type = $active_event_plugin->get_post_type();
|
||||||
}
|
}
|
||||||
|
|
||||||
$args = array(
|
$args = array(
|
||||||
|
|
|
@ -126,7 +126,6 @@ class Setup {
|
||||||
\Event_Bridge_For_ActivityPub\Integrations\WP_Event_Manager::class,
|
\Event_Bridge_For_ActivityPub\Integrations\WP_Event_Manager::class,
|
||||||
\Event_Bridge_For_ActivityPub\Integrations\Eventin::class,
|
\Event_Bridge_For_ActivityPub\Integrations\Eventin::class,
|
||||||
\Event_Bridge_For_ActivityPub\Integrations\Modern_Events_Calendar_Lite::class,
|
\Event_Bridge_For_ActivityPub\Integrations\Modern_Events_Calendar_Lite::class,
|
||||||
\Event_Bridge_For_ActivityPub\Integrations\EventPrime::class,
|
|
||||||
\Event_Bridge_For_ActivityPub\Integrations\Event_Organiser::class,
|
\Event_Bridge_For_ActivityPub\Integrations\Event_Organiser::class,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
|
|
@ -1,230 +0,0 @@
|
||||||
<?php
|
|
||||||
/**
|
|
||||||
* EventPrime – Events Calendar, Bookings and Tickets.
|
|
||||||
*
|
|
||||||
* Defines all the necessary meta information and methods for the integration of the
|
|
||||||
* WordPress plugin "EventPrime – Events Calendar, Bookings and Tickets".
|
|
||||||
*
|
|
||||||
* @link https://wordpress.org/plugins/eventprime-event-calendar-management/
|
|
||||||
* @package Event_Bridge_For_ActivityPub
|
|
||||||
* @since 1.0.0
|
|
||||||
*/
|
|
||||||
|
|
||||||
namespace Event_Bridge_For_ActivityPub\Integrations;
|
|
||||||
|
|
||||||
use Event_Bridge_For_ActivityPub\ActivityPub\Transformer\EventPrime as EventPrime_Transformer;
|
|
||||||
|
|
||||||
use Activitypub\Signature;
|
|
||||||
use Eventprime_Basic_Functions;
|
|
||||||
|
|
||||||
// Exit if accessed directly.
|
|
||||||
defined( 'ABSPATH' ) || exit; // @codeCoverageIgnore
|
|
||||||
|
|
||||||
/**
|
|
||||||
* EventPrime – Events Calendar, Bookings and Tickets.
|
|
||||||
*
|
|
||||||
* Defines all the necessary meta information and methods for the integration of the
|
|
||||||
* WordPress plugin "EventPrime – Events Calendar, Bookings and Tickets".
|
|
||||||
*
|
|
||||||
* @since 1.0.0
|
|
||||||
*/
|
|
||||||
final class EventPrime extends Event_Plugin_Integration {
|
|
||||||
/**
|
|
||||||
* Add filter for the template inclusion.
|
|
||||||
*/
|
|
||||||
public function __construct() {
|
|
||||||
\add_filter( 'template_include', array( self::class, 'render_activitypub_template' ), 100 );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the full plugin file.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public static function get_relative_plugin_file(): string {
|
|
||||||
return 'eventprime-event-calendar-management/event-prime.php';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the event post type of the plugin.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public static function get_post_type(): string {
|
|
||||||
return 'em_event';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the IDs of the admin pages of the plugin.
|
|
||||||
*
|
|
||||||
* @return array The settings page urls.
|
|
||||||
*/
|
|
||||||
public static function get_settings_pages(): array {
|
|
||||||
return array( 'ep-settings' );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the ActivityPub transformer for a EventPrime event post.
|
|
||||||
*
|
|
||||||
* @param WP_Post $post The WordPress post object of the Event.
|
|
||||||
* @return EventPrime_Transformer
|
|
||||||
*/
|
|
||||||
public static function get_activitypub_event_transformer( $post ): EventPrime_Transformer {
|
|
||||||
return new EventPrime_Transformer( $post, self::get_event_category_taxonomy() );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Returns the taxonomy used for the plugin's event categories.
|
|
||||||
*
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public static function get_event_category_taxonomy(): string {
|
|
||||||
return 'em_event_type';
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine whether the current request is an EventPrime ActivityPub request.
|
|
||||||
*
|
|
||||||
* Forked from https://github.com/Automattic/wordpress-activitypub/blob/trunk/includes/functions.php
|
|
||||||
* the function is_activitypub_request.
|
|
||||||
*/
|
|
||||||
private static function is_eventprime_activitypub_request() {
|
|
||||||
global $wp_query;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* ActivityPub requests are currently only made for
|
|
||||||
* author archives, singular posts, and the homepage.
|
|
||||||
*/
|
|
||||||
if ( ! \is_author() && ! \is_singular() && ! \is_home() && ! defined( '\REST_REQUEST' ) ) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if the current post type supports ActivityPub.
|
|
||||||
if ( \is_singular() ) {
|
|
||||||
$queried_object = \get_queried_object();
|
|
||||||
|
|
||||||
if ( ! $queried_object instanceof \WP_Post ) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( '[em_event]' !== $queried_object->post_content && '[em_events]' !== $queried_object->post_content ) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if header already sent.
|
|
||||||
if ( ! \headers_sent() && ACTIVITYPUB_SEND_VARY_HEADER ) {
|
|
||||||
// Send Vary header for Accept header.
|
|
||||||
\header( 'Vary: Accept' );
|
|
||||||
}
|
|
||||||
|
|
||||||
// One can trigger an ActivityPub request by adding ?activitypub to the URL.
|
|
||||||
if ( isset( $wp_query->query_vars['activitypub'] ) ) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* The other (more common) option to make an ActivityPub request
|
|
||||||
* is to send an Accept header.
|
|
||||||
*/
|
|
||||||
if ( isset( $_SERVER['HTTP_ACCEPT'] ) ) {
|
|
||||||
$accept = sanitize_text_field( wp_unslash( $_SERVER['HTTP_ACCEPT'] ) );
|
|
||||||
|
|
||||||
/*
|
|
||||||
* $accept can be a single value, or a comma separated list of values.
|
|
||||||
* We want to support both scenarios,
|
|
||||||
* and return true when the header includes at least one of the following:
|
|
||||||
* - application/activity+json
|
|
||||||
* - application/ld+json
|
|
||||||
* - application/json
|
|
||||||
*/
|
|
||||||
if ( preg_match( '/(application\/(ld\+json|activity\+json|json))/i', $accept ) ) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Extract the post id of the event for an EventPrime event query.
|
|
||||||
*
|
|
||||||
* @return bool|int The post ID if an event could be identified, false otherwise.
|
|
||||||
*/
|
|
||||||
private static function get_eventprime_post_id() {
|
|
||||||
$event = get_query_var( 'event' );
|
|
||||||
if ( ! $event ) {
|
|
||||||
if ( ! empty( filter_input( INPUT_GET, 'event', FILTER_SANITIZE_FULL_SPECIAL_CHARS ) ) ) {
|
|
||||||
$event = rtrim( filter_input( INPUT_GET, 'event', FILTER_SANITIZE_FULL_SPECIAL_CHARS ), '/\\' );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( $event ) {
|
|
||||||
$ep_basic_functions = new Eventprime_Basic_Functions();
|
|
||||||
return $ep_basic_functions->ep_get_id_by_slug( $event, 'em_event' );
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add the ActivityPub template for EventPrime.
|
|
||||||
*
|
|
||||||
* @param string $template The path to the template object.
|
|
||||||
* @return string The new path to the JSON template.
|
|
||||||
*/
|
|
||||||
public static function render_activitypub_template( $template ) {
|
|
||||||
if ( defined( 'REST_REQUEST' ) && REST_REQUEST ) {
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if the request is a page with (solely) the eventprime shortcode in it.
|
|
||||||
if ( ! self::is_eventprime_activitypub_request() ) {
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( ! \is_singular() ) {
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
|
|
||||||
$post_id = self::get_eventprime_post_id();
|
|
||||||
|
|
||||||
if ( $post_id ) {
|
|
||||||
$preview = \get_query_var( 'preview' );
|
|
||||||
if ( $preview ) {
|
|
||||||
$activitypub_template = ACTIVITYPUB_PLUGIN_DIR . '/templates/post-preview.php';
|
|
||||||
} else {
|
|
||||||
$activitypub_template = ACTIVITYPUB_PLUGIN_DIR . '/templates/post-json.php';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Check if the request is authorized.
|
|
||||||
*
|
|
||||||
* @see https://www.w3.org/wiki/SocialCG/ActivityPub/Primer/Authentication_Authorization#Authorized_fetch
|
|
||||||
* @see https://swicg.github.io/activitypub-http-signature/#authorized-fetch
|
|
||||||
*/
|
|
||||||
if ( $activitypub_template && defined( 'ACTIVITYPUB_AUTHORIZED_FETCH' ) && constant( 'ACTIVITYPUB_AUTHORIZED_FETCH' ) ) {
|
|
||||||
$verification = Signature::verify_http_signature( $_SERVER );
|
|
||||||
if ( \is_wp_error( $verification ) ) {
|
|
||||||
header( 'HTTP/1.1 401 Unauthorized' );
|
|
||||||
|
|
||||||
// Fallback as template_loader can't return http headers.
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ( $activitypub_template ) {
|
|
||||||
global $post;
|
|
||||||
|
|
||||||
$post = get_post( $post_id ); // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited
|
|
||||||
|
|
||||||
// Ensure WordPress functions use the new post data.
|
|
||||||
setup_postdata( $post );
|
|
||||||
// Return the default ActivityPub template.
|
|
||||||
return $activitypub_template;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $template;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -3,7 +3,7 @@ Contributors: andremenrath
|
||||||
Tags: events, fediverse, activitypub, calendar
|
Tags: events, fediverse, activitypub, calendar
|
||||||
Requires at least: 6.5
|
Requires at least: 6.5
|
||||||
Tested up to: 6.7
|
Tested up to: 6.7
|
||||||
Stable tag: 0.3.2
|
Stable tag: 0.3.3
|
||||||
Requires PHP: 7.4
|
Requires PHP: 7.4
|
||||||
License: AGPL-3.0-or-later
|
License: AGPL-3.0-or-later
|
||||||
License URI: https://www.gnu.org/licenses/agpl-3.0.html
|
License URI: https://www.gnu.org/licenses/agpl-3.0.html
|
||||||
|
@ -57,7 +57,6 @@ This plugin depends on the [ActivityPub plugin](https://wordpress.org/plugins/ac
|
||||||
* [Eventin](https://de.wordpress.org/plugins/wp-event-solution/)
|
* [Eventin](https://de.wordpress.org/plugins/wp-event-solution/)
|
||||||
* [Modern Events Calendar Lite](https://webnus.net/modern-events-calendar/)
|
* [Modern Events Calendar Lite](https://webnus.net/modern-events-calendar/)
|
||||||
* [GatherPress](https://gatherpress.org/)
|
* [GatherPress](https://gatherpress.org/)
|
||||||
* [EventPrime – Events Calendar, Bookings and Tickets](https://wordpress.org/plugins/eventprime-event-calendar-management/)
|
|
||||||
* [Event Organiser](https://wordpress.org/plugins/event-organiser/)
|
* [Event Organiser](https://wordpress.org/plugins/event-organiser/)
|
||||||
|
|
||||||
== Configuration ==
|
== Configuration ==
|
||||||
|
@ -96,6 +95,6 @@ We're always interested in your feedback. Feel free to reach out to us via [E-Ma
|
||||||
|
|
||||||
== Changelog ==
|
== Changelog ==
|
||||||
|
|
||||||
= [0.3.2] 2024-12-12 =
|
= [0.3.3] 2024-12-19 =
|
||||||
|
|
||||||
* Initial release on https://wordpress.org/
|
* Initial release on https://wordpress.org/
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
defined( 'ABSPATH' ) || exit; // @codeCoverageIgnore
|
defined( 'ABSPATH' ) || exit; // @codeCoverageIgnore
|
||||||
|
|
||||||
use Event_Bridge_For_ActivityPub\Setup;
|
use Event_Bridge_For_ActivityPub\Setup;
|
||||||
|
use Event_Bridge_For_ActivityPub\Admin\General_Admin_Notices;
|
||||||
use Event_Bridge_For_ActivityPub\Admin\Settings_Page;
|
use Event_Bridge_For_ActivityPub\Admin\Settings_Page;
|
||||||
use Event_Bridge_For_ActivityPub\Admin\Health_Check;
|
use Event_Bridge_For_ActivityPub\Admin\Health_Check;
|
||||||
|
|
||||||
|
@ -43,6 +44,12 @@ WP_Filesystem();
|
||||||
<div class="box">
|
<div class="box">
|
||||||
<h2><?php \esc_html_e( 'Status', 'event-bridge-for-activitypub' ); ?></h2>
|
<h2><?php \esc_html_e( 'Status', 'event-bridge-for-activitypub' ); ?></h2>
|
||||||
<p><?php \esc_html_e( 'The Event Bridge for ActivityPub detected the following (activated) event plugins:', 'event-bridge-for-activitypub' ); ?></p>
|
<p><?php \esc_html_e( 'The Event Bridge for ActivityPub detected the following (activated) event plugins:', 'event-bridge-for-activitypub' ); ?></p>
|
||||||
|
<?php
|
||||||
|
if ( empty( $active_event_plugins ) ) {
|
||||||
|
$notice = General_Admin_Notices::get_admin_notice_no_supported_event_plugin_active();
|
||||||
|
}
|
||||||
|
echo '<p>⚠' . \wp_kses( $notice, General_Admin_Notices::ALLOWED_HTML ) . '</p>';
|
||||||
|
?>
|
||||||
<?php foreach ( $active_event_plugins as $active_event_plugin ) { ?>
|
<?php foreach ( $active_event_plugins as $active_event_plugin ) { ?>
|
||||||
<h3><?php echo esc_html( $active_event_plugin->get_plugin_name() ); ?>:</h3>
|
<h3><?php echo esc_html( $active_event_plugin->get_plugin_name() ); ?>:</h3>
|
||||||
<ul class="event-bridge-for-activitypub-list">
|
<ul class="event-bridge-for-activitypub-list">
|
||||||
|
|
|
@ -1,188 +0,0 @@
|
||||||
<?php
|
|
||||||
/**
|
|
||||||
* Class SampleTest
|
|
||||||
*
|
|
||||||
* @package Event_Bridge_For_ActivityPub
|
|
||||||
*/
|
|
||||||
|
|
||||||
namespace Event_Bridge_For_ActivityPub\Tests\ActivityPub\Transformer;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Sample test case.
|
|
||||||
*/
|
|
||||||
class Test_EventPrime extends \WP_UnitTestCase {
|
|
||||||
/**
|
|
||||||
* Mockup venues of certain complexity.
|
|
||||||
*
|
|
||||||
* @var array
|
|
||||||
*/
|
|
||||||
private $mockup_venue = array();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Mockup events for tests.
|
|
||||||
*
|
|
||||||
* @var array
|
|
||||||
*/
|
|
||||||
protected $mockup_events = array();
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Override the setup function, so that tests don't run if the Events Calendar is not active.
|
|
||||||
*/
|
|
||||||
public function set_up() {
|
|
||||||
parent::set_up();
|
|
||||||
|
|
||||||
if ( ! class_exists( '\Eventprime_Basic_Functions' ) ) {
|
|
||||||
self::markTestSkipped( 'The EventPrime Calendar management plugin is not active.' );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure that ActivityPub support is enabled for The Events Calendar.
|
|
||||||
$aeb = \Event_Bridge_For_ActivityPub\Setup::get_instance();
|
|
||||||
$aeb->activate_activitypub_support_for_active_event_plugins();
|
|
||||||
|
|
||||||
// Delete all posts afterwards.
|
|
||||||
_delete_all_posts();
|
|
||||||
|
|
||||||
$this->setup_mockup_data();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Setup mockup events.
|
|
||||||
*/
|
|
||||||
private function setup_mockup_data() {
|
|
||||||
$this->mockup_events = array(
|
|
||||||
'minimal_event' => array(
|
|
||||||
'name' => 'EventPrime Event title',
|
|
||||||
'description' => 'EventPrime event description',
|
|
||||||
'status' => 'Publish',
|
|
||||||
'em_event_type' => '',
|
|
||||||
'em_venue' => '',
|
|
||||||
'em_organizer' => '',
|
|
||||||
'em_performer' => '',
|
|
||||||
'em_start_date' => strtotime( '+10 days 15:00:00' ),
|
|
||||||
'em_end_date' => strtotime( '+10 days 16:00:00' ),
|
|
||||||
'em_enable_booking' => 'bookings_off',
|
|
||||||
'em_ticket_price' => 0,
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
$this->mockup_venue = array(
|
|
||||||
'name' => 'Test Venue',
|
|
||||||
'address' => 'Fediverse-street 1337, 1234 Fediverse-town',
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test that the right transformer gets applied.
|
|
||||||
*/
|
|
||||||
public function test_the_events_calendar_transformer_class() {
|
|
||||||
// We only test for one event plugin being active at the same time,
|
|
||||||
// even though we support multiple onces in theory.
|
|
||||||
// But testing all combinations is beyond scope.
|
|
||||||
$active_event_plugins = \Event_Bridge_For_ActivityPub\Setup::get_instance()->get_active_event_plugins();
|
|
||||||
$this->assertEquals( 1, count( $active_event_plugins ) );
|
|
||||||
|
|
||||||
// Enable ActivityPub support for the event plugin.
|
|
||||||
$this->assertContains( 'em_event', get_option( 'activitypub_support_post_types' ) );
|
|
||||||
|
|
||||||
// Create an EventPrime Event without content.
|
|
||||||
$ep_functions = new Eventprime_Basic_Functions();
|
|
||||||
|
|
||||||
$post_id = $ep_functions->insert_event_post_data( $this->mockup_events['minimal_event'] );
|
|
||||||
|
|
||||||
$wp_object = get_post( $post_id );
|
|
||||||
|
|
||||||
// Call the transformer Factory.
|
|
||||||
$transformer = \Activitypub\Transformer\Factory::get_transformer( $wp_object );
|
|
||||||
|
|
||||||
// Check that we got the right transformer.
|
|
||||||
$this->assertInstanceOf( \Event_Bridge_For_ActivityPub\ActivityPub\Transformer\EventPrime::class, $transformer );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test transformation of minimal event.
|
|
||||||
*/
|
|
||||||
public function test_transformation_of_minimal_event() {
|
|
||||||
// Create an EventPrime Event without content.
|
|
||||||
$ep_functions = new Eventprime_Basic_Functions();
|
|
||||||
|
|
||||||
$post_id = $ep_functions->insert_event_post_data( $this->mockup_events['minimal_event'] );
|
|
||||||
|
|
||||||
$wp_object = get_post( $post_id );
|
|
||||||
|
|
||||||
// Call the transformer Factory.
|
|
||||||
$event_array = \Activitypub\Transformer\Factory::get_transformer( $wp_object )->to_object()->to_array();
|
|
||||||
|
|
||||||
// Check that the event ActivityStreams representation contains everything as expected.
|
|
||||||
$this->assertEquals( 'Event', $event_array['type'] );
|
|
||||||
$this->assertEquals( 'EventPrime Event title', $event_array['name'] );
|
|
||||||
$this->assertEquals( 'EventPrime event description', wp_strip_all_tags( $event_array['content'] ) );
|
|
||||||
$this->assertEquals( gmdate( 'Y-m-d', strtotime( '+10 days 15:00:00' ) ) . 'T15:00:00Z', $event_array['startTime'] );
|
|
||||||
$this->assertEquals( gmdate( 'Y-m-d', strtotime( '+10 days 16:00:00' ) ) . 'T16:00:00Z', $event_array['endTime'] );
|
|
||||||
$this->assertTrue( $event_array['commentsEnabled'] );
|
|
||||||
$this->assertEquals( 'allow_all', $event_array['repliesModerationOption'] );
|
|
||||||
$this->assertEquals( 'external', $event_array['joinMode'] );
|
|
||||||
$this->assertArrayNotHasKey( 'location', $event_array );
|
|
||||||
$this->assertEquals( 'MEETING', $event_array['category'] );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test transformation of minimal event.
|
|
||||||
*/
|
|
||||||
public function test_transformation_of_minimal_event_with_venue() {
|
|
||||||
// Create an EventPrime Event without content.
|
|
||||||
$ep_functions = new Eventprime_Basic_Functions();
|
|
||||||
|
|
||||||
$venue_term_id = wp_insert_term( $this->mockup_venue['name'], 'em_venue' )['term_id'];
|
|
||||||
add_term_meta( $venue_term_id, 'em_address', $this->mockup_venue['address'], true );
|
|
||||||
add_term_meta( $venue_term_id, 'em_display_address_on_frontend', true, true );
|
|
||||||
|
|
||||||
$event_data = $this->mockup_events['minimal_event'];
|
|
||||||
$event_data['em_venue'] = $venue_term_id;
|
|
||||||
|
|
||||||
$post_id = $ep_functions->insert_event_post_data( $event_data );
|
|
||||||
|
|
||||||
$wp_object = get_post( $post_id );
|
|
||||||
|
|
||||||
// Call the transformer Factory.
|
|
||||||
$event_array = \Activitypub\Transformer\Factory::get_transformer( $wp_object )->to_object()->to_array();
|
|
||||||
|
|
||||||
// Check that the event ActivityStreams representation contains everything as expected.
|
|
||||||
$this->assertEquals( 'Event', $event_array['type'] );
|
|
||||||
$this->assertEquals( 'EventPrime Event title', $event_array['name'] );
|
|
||||||
$this->assertEquals( 'EventPrime event description', wp_strip_all_tags( $event_array['content'] ) );
|
|
||||||
$this->assertEquals( gmdate( 'Y-m-d', strtotime( '+10 days 15:00:00' ) ) . 'T15:00:00Z', $event_array['startTime'] );
|
|
||||||
$this->assertEquals( gmdate( 'Y-m-d', strtotime( '+10 days 16:00:00' ) ) . 'T16:00:00Z', $event_array['endTime'] );
|
|
||||||
$this->assertTrue( $event_array['commentsEnabled'] );
|
|
||||||
$this->assertEquals( 'allow_all', $event_array['repliesModerationOption'] );
|
|
||||||
$this->assertEquals( 'external', $event_array['joinMode'] );
|
|
||||||
$this->assertEquals( $this->mockup_venue['name'], $event_array['location']['name'] );
|
|
||||||
$this->assertEquals( $this->mockup_venue['address'], $event_array['location']['address'] );
|
|
||||||
|
|
||||||
$this->assertEquals( 'MEETING', $event_array['category'] );
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Test transformation of minimal event with venue which has a hidden address.
|
|
||||||
*/
|
|
||||||
public function test_transformation_of_minimal_event_with_venue_with_hidden_address() {
|
|
||||||
// Create an EventPrime Event without content.
|
|
||||||
$ep_functions = new Eventprime_Basic_Functions();
|
|
||||||
|
|
||||||
$venue_term_id = wp_insert_term( $this->mockup_venue['name'], 'em_venue' )['term_id'];
|
|
||||||
add_term_meta( $venue_term_id, 'em_address', $this->mockup_venue['address'], true );
|
|
||||||
add_term_meta( $venue_term_id, 'em_display_address_on_frontend', false, true );
|
|
||||||
|
|
||||||
$event_data = $this->mockup_events['minimal_event'];
|
|
||||||
$event_data['em_venue'] = $venue_term_id;
|
|
||||||
|
|
||||||
$post_id = $ep_functions->insert_event_post_data( $event_data );
|
|
||||||
|
|
||||||
$wp_object = get_post( $post_id );
|
|
||||||
|
|
||||||
// Call the transformer Factory.
|
|
||||||
$event_array = \Activitypub\Transformer\Factory::get_transformer( $wp_object )->to_object()->to_array();
|
|
||||||
|
|
||||||
// Check that the event ActivityStreams representation contains everything as expected.
|
|
||||||
$this->assertArrayNotHasKey( 'address', $event_array['location'] );
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -58,7 +58,7 @@ class Test_GatherPress extends \WP_UnitTestCase {
|
||||||
$params = array(
|
$params = array(
|
||||||
'datetime_start' => '+10 days 15:00:00',
|
'datetime_start' => '+10 days 15:00:00',
|
||||||
'datetime_end' => '+10 days 16:00:00',
|
'datetime_end' => '+10 days 16:00:00',
|
||||||
'timezone' => 'America/New_York',
|
'timezone' => \wp_timezone_string(),
|
||||||
);
|
);
|
||||||
|
|
||||||
$event->save_datetimes( $params );
|
$event->save_datetimes( $params );
|
||||||
|
@ -87,7 +87,7 @@ class Test_GatherPress extends \WP_UnitTestCase {
|
||||||
$params = array(
|
$params = array(
|
||||||
'datetime_start' => '+10 days 15:00:00',
|
'datetime_start' => '+10 days 15:00:00',
|
||||||
'datetime_end' => '+10 days 16:00:00',
|
'datetime_end' => '+10 days 16:00:00',
|
||||||
'timezone' => 'America/New_York',
|
'timezone' => \wp_timezone_string(),
|
||||||
);
|
);
|
||||||
$event->save_datetimes( $params );
|
$event->save_datetimes( $params );
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue