Peter Wilson
○
@pwcc
○
peterwilson.cc
Burn it down: A case study in CMS replatforming
A presentation at LoopConf in February 2018 in Salt Lake City, UT, USA by Peter Wilson
 
                Peter Wilson
○
@pwcc
○
peterwilson.cc
Burn it down: A case study in CMS replatforming
 
                brisbanetimes.com.au August 27, 2017, 8:07pm
 
                brisbanetimes.com.au August 27, 2017, 8:07pm August 27, 2017, 8:08pm
 
                 
                 
                “ It was just another day in the newsroom
 
                “ It was just another day writing content
 
                humanmade.com/fairfax-media
 
                */ $must_use_plugins = [
'ffx-options/plugin.php'
,
// Filter get_option calls.
/* ... Vendor plugins snipped ... */
'ffx-helpers/plugin.php'
,
// Available during bootstrap.
'ffx-feature-flags/plugin.php' , // Available during bootstrap.
'ffx-api-content/plugin.php' ,
'ffx-api-media/plugin.php' ,
'ffx-roles-capabilities/plugin.php' ,
'ffx-article-editor/plugin.php' ,
'ffx-live-articles/plugin.php' ,
'ffx-notifications/plugin.php' ,
'ffx-shortcake/plugin.php' ,
'ffx-brightcove/plugin.php' ,
'ffx-profiles/plugin.php' ,
'ffx-unpublish/plugin.php' ,
'ffx-wire-feed/plugin.php' ,
'ffx-image-editor/plugin.php' ,
'ffx-taxonomies/plugin.php' ,
'ffx-dashboard/plugin.php' ,
'ffx-publishing/plugin.php' ,
'ffx-tooltips/plugin.php' , ];
 
                Reimagining the tech-stack
 
                The stack
 
                The stack
 
                The stack
 
                The stack
 
                The stack
 
                The stack
 
                The stack Media API Content API C loudinary
 
                Extending WordPress for enterprise
 
                Publishing and Workflow
 
                Media Library
 
                Taxonomies
 
                Tagging articles
at scale
 
                SELECT count ( * ) FROM wp_term_taxonomy WHERE taxonomy= 'ffx_tag'
 
                2 2 3 0 3
 
                 
                 
                Content API tags endpoint ! Tags manager
 
                 
                GET 50 tags Content API
 
                GET 50 tags wp_insert_term( /* Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 */ ); Content API
 
                Managing terms
makes a lot of DB queries
 
                wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] );
 
                SELECT tt.term_id, tt.term_taxonomy_id FROM wp_terms AS t
INNER JOIN wp_term_taxonomy as tt
ON tt.term_id = t.term_id WHERE t.slug = 'roxie-hart '
AND tt.parent = '105'
AND tt.taxonomy = 'ffx_tag' ORDER BY t.term_id ASC LIMIT 1
 
                SELECT tt.term_id, tt.term_taxonomy_id FROM wp_terms AS t
INNER JOIN wp_term_taxonomy as tt
ON tt.term_id = t.term_id
AND tt.parent = '105'
AND tt.taxonomy = 'ffx_tag' ORDER BY t.term_id ASC LIMIT 1
 
                SELECT tt.term_id, tt.term_taxonomy_id FROM wp_terms AS t
INNER JOIN wp_term_taxonomy as tt
ON tt.term_id = t.term_id
AND tt.parent = '105'
AND tt.taxonomy = 'ffx_tag' ORDER BY t.term_id ASC LIMIT 1
' R o x i e H a r t '
 
                SELECT
t.
*
, tt.
*
FROM
wp_terms
AS
t
INNER JOIN wp_term_taxonomy AS tt
ON t.term_id = tt.term_id WHERE tt.taxonomy IN ( 'ffx_tag' )
AND t.name IN ( 'Roxie Hart' )
AND tt.parent = '105' ORDER BY t.name ASC
 
                SELECT term_id FROM wp_terms as t WHERE t.slug = 'roxie-hart' ORDER BY t.term_id ASC LIMIT 1
 
                INSERT
INTO
wp_terms (name,slug,term_group)
VALUES
(
'Roxie Hart'
,
'roxie-hart'
,
0
)
 
                Production, Chicago
 
                . SELECT tt.term_taxonomy_id FROM wp_term_taxonomy AS tt
INNER JOIN wp_terms AS t
ON tt.term_id = t.term_id WHERE tt.taxonomy = 'ffx_tag'
AND t.term_id = 23170
 
                INSERT
INTO
wp_term_taxonomy
( term_id, taxonomy,
`description`, `parent`, `count`
) VALUES ( 23170 , 'ffx_tag' , '' , 105 , 0 )
 
                 
                SELECT t.term_id, tt.term_taxonomy_id FROM wp_terms t
INNER JOIN wp_term_taxonomy tt
ON ( tt.term_id = t.term_id ) WHERE t.slug = 'roxie-hart'
AND tt.parent = 105
AND tt.taxonomy = 'ffx_tag'
AND t.term_id < 23170
AND tt.term_taxonomy_id != 23170
 
                And that’s not all
 
                SELECT term_id, meta_key, meta_value FROM wp_term
SELECT meta_id FROM wp_termmeta WHERE meta_key =
INSERT INTO
wp_termmeta (term_id, meta_key,
 
                DELETE FROM
wp_options
WHERE
option_name =
'ffx_tag_children'
SELECT option_value FROM wp_options WHERE option_name = 'ffx_tag_children' LIMIT 1
all terms
in the taxonomy
to work out hierarchy
SELECT
t.term_id, tt.parent, tt.count, tt.taxonomy
FROM
wp_terms
AS
t
INNER
SELECT term_id, meta_key, meta_value FROM wp_termmeta WHERE term_id IN ( 95 ,
INSERT INTO
wp_options (option_name, option_value, autoload)
 
                Production, Chicago
 
                1 ; $page <= $total_pages ; $page ++ ) {
// Get all the tags we need to process on the current page.
$tags = get_tags_from_api( $page );
// Schedule a single event to create or update terms.
wp_schedule_single_event(
time () + ( 10 * $page * MINUTE_IN_SECONDS ),
'ffx_import_some_terms_action' ,
$tags
); }
 
                Complex sites need scheduled tasks & asynchronous processing. Meet Cavalcade: A #WordPress jobs processing solution humanmade.com /cavalcade Human Made @ humanmadeltd 5:31 AM - 30 May 2017
 
                NASA
 
                NASA
 
                 
                We were so focused on the database , we forgot about HTTP
 
                 
                wp-cron.php
 
                wp-cron.php Ta g s A P I
 
                wp-cron.php Ta g s A P I Saved as cron job
 
                wp-cron.php Ta g s A P I Saved as cron job
 
                That’s what we forgot HTTP requests time out Prairie Kittin, flic.kr/p/a4Ujpv
 
                update_option( 'cron' , [ /* ... */ ] );
 
                update_option( 'cron' , [ /* ... */ ] );
 
                update_option( 'cron' , [ /* ... */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] );
 
                update_option( 'cron' , [ /* ... */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] );
 
                update_option( 'cron' , [ /* ... */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] ); wp_insert_term(
'Roxie Hart' ,
'ffx_tags' , [
'slug'
=>
'roxie-hart'
,
'parent' => '105' , ] ); add_term_meta( 23170 , '_ffx_id' , 'h4rt' ); add_term_meta( 23170 , '_ffx_term_meta' , [ /* */ ] );
 
                We reverted the original commit.
 
                Complex sites need scheduled tasks & asynchronous processing. Meet Cavalcade: A #WordPress jobs processing solution humanmade.com /cavalcade Human Made @ humanmadeltd 5:31 AM - 30 May 2017
 
                /**
 
                 
                1 ; $page <= $total_pages ; $page ++ ) {
// Get all the tags we need to process on the current page.
$tags = get_tags_from_api( $page );
// Schedule a single event to create or update terms.
wp_schedule_single_event(
time () + ( 10 * $page * MINUTE_IN_SECONDS ),
'ffx_import_some_terms_action' ,
$tags
); }
 
                1 ; $page <= $total_pages ; $page ++ ) {
// Get all the tags we need to process on the current page.
$tags = get_tags_from_api( $page );
// Schedule a single event to create or update terms.
wp_schedule_single_event( time() , // Schedule all pages NOW!
'ffx_import_some_terms_action' ,
$tags
); }
 
                1 ; $page <= $total_pages ; $page ++ ) {
// Get all the tags we need to process on the current page.
$tags = get_tags_from_api( $page );
// Schedule a single event to create or update terms.
wp_schedule_single_event( time() , // Schedule all pages NOW!
'ffx_import_some_terms_action' ,
$tags
); }
 
                1 ; $page <= $total_pages ; $page ++ ) {
// Set the arguments needed to process the current page.
$args = [ 'page' => $page , 'qty' => 50 ];
// Schedule a single event to create or update terms.
wp_schedule_single_event(
time (), // Schedule all pages NOW!
'ffx_import_tags_cron' ,
$args
); }
 
                1 ; $page <= $total_pages ; $page ++ ) {
// Set the arguments needed to process the current page.
$args = [ 'page' => $page , 'qty' => 50 ];
// Schedule a single event to create or update terms.
wp_schedule_single_event(
time (), // Schedule all pages NOW!
 
                // Schedule a single event to create or update terms.
wp_schedule_single_event(
time (), // Schedule all pages NOW!
'ffx_import_tags_cron' ,
$args
); } // Delete the scheduling lock. delete_option( TAGS_SCHEDULING_LOCK_OPTION ); update_option( TAGS_SCHEDULED_LOCK_OPTION , time () );
 
                [11] Worker shutting down...
[11] Worker out:
[11] Worker err:
[11] Worker ret: 0
[12] Worker status: Array
(
[command] => wp cavalcade run 12
[pid] => 57
[running] =>
[signaled] =>
[stopped] =>
[exitcode] => 0
[termsig] => 0
[stopsig] => 0
)
2018-01-27T01:09:22.458320545Z
[12] Worker shutting down...
[12] Worker out:
[12] Worker err:
[12] Worker ret: 0
[14] Running wp cavalcade run 14 (ffx/taxonomies/term_importer/import_tags_cron a:1:{i:0;a:5:{s:8:"tag_type";s:4:"tags";s:10:"total_tags";i:23032;s:4:"size";i:50;s:5:"pages";i:461;s:4:"page";i:1;}})
[14] Started worker
[15] Running wp cavalcade run 15 (ffx/taxonomies/term_importer/import_tags_cron a:1:{i:0;a:5:{s:8:"tag_type";s:4:"tags";s:10:"total_tags";i:23032;s:4:"size";i:50;s:5:"pages";i:461;s:4:"page";i:2;}})
[15] Started worker
[16] Running wp cavalcade run 16 (ffx/taxonomies/term_importer/import_tags_cron a:1:{i:0;a:5:{s:8:"tag_type";s:4:"tags";s:10:"total_tags";i:23032;s:4:"size";i:50;s:5:"pages";i:461;s:4:"page";i:3;}})
[16] Started worker
[  ] Out of workers
[  ] Out of workers
[  ] Out of workers
[  ] Out of workers
[14] Worker status: Array
(
 
                GET 50 tags wp_insert_term( /* Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 */ ); Content API
 
                wp_insert_term( /* Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 */ ); GET 50 tags Content API "
 
                wp_insert_term( /* Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 / ); " GET 50 tags Content API " GET 50 tags Content API " GET 50 tags Content API " GET 50 tags Content API wp_insert_term( / Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 / ); wp_insert_term( / Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 / ); wp_insert_term( / Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 */ );
 
                wp_insert_term( /* Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 / ); " GET 50 tags Content API " GET 50 tags Content API " GET 50 tags Content API " GET 50 tags Content API wp_insert_term( / Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 / ); wp_insert_term( / Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 / ); wp_insert_term( / Roxie Hart / ); wp_insert_term( / Billy Flynn / ); wp_insert_term( / Velma Kelly / ); wp_insert_term( / x 50 */ );
 
                DELETE FROM
wp_options
WHERE
option_name =
'ffx_tag_children'
SELECT option_value FROM wp_options WHERE option_name = 'ffx_tag_children' LIMIT 1
all terms
in the taxonomy
to work out hierarchy
SELECT
t.term_id, tt.parent, tt.count, tt.taxonomy
FROM
wp_terms
AS
t
INNER
SELECT term_id, meta_key, meta_value FROM wp_termmeta WHERE term_id IN ( 95 ,
335KB at 23K tags)
INSERT INTO
wp_options (option_name, option_value, autoload)
 
                cache
335KB at 23K tags )
wp_options 	WHERE 	option_name = 	'ffx_tag_children' 	# Get term hierarchy option (for reasons) SELECT 	option_value 	FROM 	wp_options 	WHERE 	option_name = 	'ffx_tag_children' 	# Select 	all terms	 in the taxonomy 	to work out hierarchy 	SELECT  	t.term_id, tt.parent, tt.count, tt.taxonomy 	FROM 	wp_terms 	AS 	t  	INNER  	# Warm the term meta cache for all (a bug)	 	SELECT 	term_id, meta_key, meta_value 	FROM 	wp_termmeta 	WHERE 	term_id 	IN 	(	95	# Update the term hierarchy cache (	335KB	 at 23K tags)	 	INSERT INTO 	wp_options (option_name, option_value, autoload) 
                function import_tags( $taxonomy , $tags ) {
/*
wp_suspend_cache_addition( true );
foreach ( $tags [ $taxonomy ] as $tag ) { wp_insert_term( /* $tag / ); update_term_meta( / tag meta 1 / ); update_term_meta( / tag meta 2 */ ); } }
 
                function import_tags( $taxonomy , $tags ) {
/*
wp_suspend_cache_addition( true );
foreach ( $tags [ $taxonomy ] as $tag ) { wp_insert_term( /* $tag / ); update_term_meta( / tag meta 1 / ); update_term_meta( / tag meta 2 */ ); } }
 
                Read Cache Read Cache Read Cache Read Cache Write cache Write cache Write cache Write cache
 
                 
                Improving the media library
 
                 
                 
                 
                Limited crops four by default
 
                Global crops happens on upload, calculated
 
                defined by user, not by system Unpredictable file names deh-logo.jpg
 
                Same database images and articles share a db table
 
                 
                 
                 
                 
                 
                 
                The stack
 
                 
                POST http://cms-authoring-local/wp/wp-admin/async-upload.php - action : upload-attachment
 
                add_action( 'admin_init' , 'ajax_upload_attachment' , 0 ); /**
by async-upload.php.
admin_init.
*/
function
ajax_upload_attachment() {// Only intercept the upload-attachment action.
if ( ! isset ( $_POST [ 'action' ] ) ||
'upload-attachment' !== $_POST [ 'action' ] ) {
return ;
 
                add_action( 'admin_init' , 'ajax_upload_attachment' , 0 ); /**
by async-upload.php.
admin_init.
*/
function
ajax_upload_attachment() {// Only intercept the upload-attachment action.
if ( ! isset ( $_POST [ 'action' ] ) ||
'upload-attachment' !== $_POST [ 'action' ] ) {
return ;
 
                by async-upload.php.
admin_init.
*/
function
ajax_upload_attachment() {// Only intercept the upload-attachment action.
if ( ! isset ( $_POST [ 'action' ] ) ||
'upload-attachment' !== $_POST [ 'action' ] ) {
return ; }
// Duplicate core functionality.
wp_die(); }
 
                by async-upload.php.
admin_init.
*/
function
ajax_upload_attachment() {// Only intercept the upload-attachment action.
if ( ! isset ( $_POST [ 'action' ] ) ||
'upload-attachment' !== $_POST [ 'action' ] ) {
return ; }
// Duplicate core functionality.
}
 
                {
'altText'
:
''
,
'description'
:
''
,
'caption'
:
''
,
'credit'
:
''
,
'keywords'
:
''
,
'sha1'
:
'7e51b009bf0e9097a1fd6ba339a78b6181cbde0c'
,
'source'
:
''
,
'source_system_name' : 'wordpress' ,
'fileDataURI'
:
'...'
}
 
                return ; }
// Duplicate core functionality.
$schema = Image_Schema\get_image_schema( $image , true );
wp_remote_request(
'https://api-media/v0/images' ,
[
'method'
=>
'POST'
,
'headers' => [
'content-type' => 'application/json' ,
],
'body' => $schema ,
'timeout' => 10 ,
]
); wp_delete_post( $image_id , true );
 
                wp_remote_request(
'https://api-media/v0/images' ,
[
'method'
=>
'POST'
,
'headers' => [
'content-type' => 'application/json' ,
],
'body' => $schema ,
'timeout' => 10 ,
]
); wp_delete_post( $image_id , true );
wp_die(); }
 
                 
                {
"success" : true ,
"data" : [ { "id" : 4424 /* ... / }, { "id" : 4333 / ... / }, { "id" : 4332 / ... / }, { "id" : 4330 / ... / }, { "id" : 4327 / ... / }, { "id" : 4323 / ... / }, { "id" : 4321 / ... / }, { "id" : 4317 / ... / }, { "id" : 4315 / ... / }, { "id" : 4312 / ... */ } ] } /wp-admin/admin-ajax.php?action= query-attachments
 
                {
"id"
:
4424
,
"filename"
:
"deh-logo.jpg"
,
"url"
:
"https://pwcc.cc/wp-content/uploads/2018/02/deh-logo.jpg"
,
"alt"
:
""
,
"description"
:
""
,
"caption"
:
""
,
"name"
:
"deh-logo"
,
"dateFormatted" : "February 8, 2018" ,
"mime"
:
"image/jpeg"
,
"type"
:
"image"
,
"subtype"
:
"jpeg"
,
"sizes" : {
"thumbnail" : {},
"medium" : {},
"large" : {},
"full" : {}
 
                {
"altText"
:
"Plaster casts from Dear Evan Hansen."
,
"caption"
:
"The props department creates a new plaster cast for Dear Evan Hansen each night. It"
,
"dateCreated" : "2018-02-07T23:07:57.191Z" ,
"credit"
:
"Internet"
,
"description" : "Plaster casts Evan wears in Dear Evan Hansen." ,
"id"
:
"78b576d29756c0f06ca5a1c450f4cf84bb69e8de"
,
"keywords"
:
""
,
"source"
:
"Internet"
}
 
                /**
 
                static.ffx.io /bd0d64a85c59655b815776ae46c3d14be7a6098e
 
                static.ffx.io/images/ t_resize_wp_admin /t_quality_best,f_auto/ …
 
                …/images/ $width_357,$height_201 /t_quality_best,f_auto/ …
 
                …/ $multiply_3,$zoom_0.35,$ratio_1.8,$width_357,$x_600,$y_180 / …
 
                add_image_size(
'square1x1'
,
200
,
200
,
true
);
add_image_size(
'landscape3x2'
,
300
,
200
,
true
);
add_image_size(
'landscape16x9'
,
357
,
201
,
true
);
add_image_size(
'portrait2x3'
,
200
,
300
,
true
);
 
                A complete waste of time
 
                add_filter(
'intermediate_image_sizes_advanced' ,
/* sizes generated */
'__return_empty_array'
/* [] - none */ );
 
                get_attachment( 'al4n4b3ck' );
 
                get_attachment( 'al4n4b3ck' ); is_int ( 'al4n4b3ck' );
 
                get_attachment( 'al4n4b3ck' ); is_int ( 'al4n4b3ck' );
false
 
                 
                /**
wp_get_attachment_metadata()@param
string $attachment_id Attachment ID. Default ''.
*
@param
bool   $unfiltered    True: filters are not run.
                          Default false. 
@return mixed Attachment meta field.
                          False on failure. 
/ function get_attachment_metadata( / ... */ ) { }
 
                var_dump ( wp_get_attachment_metadata() ); Array ( [ width ] => 2400 , [ height ] => 1559 , [ file ] => '2018/02/you-will-be-found.jpg' , [ sizes ] => Array ( [ square1x1 ] => Array ( width, height, file ), [ landscape3x2 ] => Array ( width, height, file ), [ landscape16x9 ] => Array ( width, height, file ), [ portrait2x3 ] => Array ( width, height, file ), [ etc ] => Array (width, height, file),
) )
 
                var_dump ( API_Media\get_attachment_metadata() ); Array ( [ width ] => 2400 , [ height ] => 1559 , [ file ] => 'bd0d64a85c59655b815776ae46c3d14be7a6098e' , [ url ] => 'http://static.ffx.io/bd0d64a85c59655b81…98e' , [ sizes ] => Array ( [ square1x1 ] => Array ( width, height, file, url ), [ landscape3x2 ] => Array ( width, height, file, url ), [ landscape16x9 ] => Array ( width, height, file, url ), [ portrait2x3 ] => Array ( width, height, file, url ), [ etc ] => Array (width, height, file, url),
) )
 
                API_Media\get_attachment_metadata( 'bd0d64...98e' ); API_Media\get_attachment( 'bd0d64...98e' ); API_Media\get_attachment_url( 'bd0d64...98e' );
 
                {
"id"
:
"bd0d64a85c59655b815776ae46c3d14be7a6098e"
,
"filename"
:
"bd0d64a85c59655b815776ae46c3d14be7a6098e"
,
"url"
:
"https://static.ffx.io/bd0d64a85c59655b815776ae46c3d14be7a6098e.jpg"
,
"alt"
:
"The Dear Evan Hansen cast perform You Will Be Found"
,
"description"
:
"Dear Evan Hansen cast perform You Will Be Found, the emotional first act climax of Dear Evan Hansen"
,
"caption"
:
"Dear Evan Hansen cast perform You Will Be Found, the emotional first act climax of Dear Evan Hansen"
,
"name"
:
"bd0d64a85c59655b815776ae46c3d14be7a6098e"
,
"dateFormatted" : "February 8, 2018 09:02am" ,
"mime"
:
"image/jpeg"
,
"type"
:
"image"
,
"subtype"
:
"jpeg"
,
"sizes" : {
"square1x1" : {},
"landscape3x2" : {},
"landscape16x9" : {},
"thumbnail" : {}
 
                Not numeric That’s new These names have changed { "id" : "bd0d64a85c59655b815776ae46c3d14be7a6098e" , "filename" : "bd0d64a85c59655b815776ae46c3d14be7a6098e" , "url" : "https://static.ffx.io/bd0d64a85c59655b815776ae46c3d14be7a6098e.jpg" "alt" : "The Dear Evan Hansen cast perform You Will Be Found" , "description" : "Dear Evan Hansen cast perform You Will Be Found, the emotional first act climax of Dear Evan Hansen" "caption" : "Dear Evan Hansen cast perform You Will Be Found, the emotional first act climax of Dear Evan Hansen" "name" : "bd0d64a85c59655b815776ae46c3d14be7a6098e" , "dateFormatted" : "February 8, 2018 09:02am" , "mime" : "image/jpeg" , "type" : "image" , "subtype" : "jpeg" , "sizes" : { “thumbnail" : {}, "landscape3x2" : {}, "landscape16x9" : {}, "thumbnail" : {}
 
                 
                 
                wp.media.view ○ Attachment ○ AttachmentCompat ○ AttachmentFilters ○ Attachments ○ AttachmentsBrowser ○ AudioDetails ○ Button ○ ButtonGroup ○ Cropper ○ DateFilter ○ EditImage ○ EditorUploader ○ Embed ○ EmbedImage ○ EmbedLink ○ EmbedUrl ○ FocusManager ○ Frame ○ Iframe ○ ImageDetails ○ Label ○ MediaDetails ○ MediaFrame ○ Menu ○ MenuItem ○ Modal ○ PriorityList ○ Router ○ RouterItem ○ Search ○ Selection ○ Settings ○ Sidebar ○ SiteIconCropper ○ SiteIconPreview ○ Spinner ○ Toolbar ○ UploaderInline ○ UploaderStatus ○ UploaderStatusError ○ UploaderWindow ○ VideoDetails
 
                Inherited from WordPress ○ 42 views ○ 6 models ○ 19 collections
 
                Inherited from WordPress ○ 42 views ○ 6 models ○ 19 collections
Underscore
Backbone $ jQuery
 
                require ( './query' );
 
                require ( './query' );
 
                require ( './query' );
 
                require ( './query' );
 
                require ( './query' );
 
                /**
wp.media.query
We're overriding this because we need to use our custom
function ( props = {} ) {
return new Attachments ( null , {
props : _ . extend (
_ . defaults ( props, { orderby : 'date' , order : 'DESC' } ), { query : true } ) } ); };
 
                /**
wp.media.query
We're overriding this because we need to use our custom
function ( props = {} ) {
return new Attachments ( null , {
props : _ . extend (
_ . defaults ( props, { orderby : 'date' , order : 'DESC' } ), { query : true } ) } ); };
 
                /**
template : function () {
const prefix = ( this . model . get ( 'type' ) === 'image' ) ? 'ffx-' : '' ;
wp
.
template
(
${ prefix } attachment
);
return template . apply ( this , arguments ); } } );
 
                /**
template : function () {
const prefix = ( this . model . get ( 'type' ) === 'image' ) ? 'ffx-' : '' ;
wp
.
template
(
${ prefix } attachment
);
return template . apply ( this , arguments ); } } );
 
                A heavy touch server side A lighter touch client side
 
                 
                 
                /**
@class
Backbone . Model . extend ( {
defaults : {
id : '' ,
url : '' ,
attachment_id : '' ,
autoCrop : true ,
fileName : '' ,
aspect : '' ,
originalWidth : 0 ,
cropWidth : 0 ,
// Attributes below should not saved.
imageUrl : '' ,
 
                 
                 
                [ img id= "f23d41a8ebf536a52eca254e1bb44d996cf21d25"
altText= "..." caption= "..." credit= "..."
source= "..." description= "..."
aspect= "1.5" cropWidth= "300" autoCrop= "false"
offsetX= "-374.3478" offsetY= "-85.9782"
zoom= "0.3062" ][/ img ]
 
                Publishing & workflow
 
                 
                 
                // We don't need this, we have
// the Publish Box of the Future™! remove_meta_box(
'submitdiv' ,
'post' ,
'side' );
 
                // T he Publish Box of the Future™! add_meta_box(
'ffx-submitdiv' , __( 'Save & Publish' , 'ffx' ),
NAMESPACE . ' \ the_loading_icon' , [ 'post' ],
'side' ,
'high' );
 
                AJAX everything avoid full page refreshes
 
                register_rest_field(
'post' ,
'ffx_writeoff' , [
'get_callback'
=>
NAMESPACE
.
'
\
get_writeoff'
,
'update_callback' => NAMESPACE . ' \ set_writeoff' ,
'schema'
=> [
'description' => __( 'Article writeoff' , 'ffx' ),
'type'
=>
'string'
,
],
]
);
 
                register_rest_field(
'post' ,
'ffx_writeoff' , [
'get_callback'
=>
NAMESPACE
.
'
\
get_writeoff'
,
'update_callback' => NAMESPACE . ' \ set_writeoff' ,
'schema'
=> [
'description' => __( 'Article writeoff' , 'ffx' ),
'type'
=>
'string'
,
],
]
);
 
                 
                 
                 
                Sixty-one custom properties
 
                register_rest_field(
'post' ,
'ffx_private' , [
'get_callback'
=>
NAMESPACE
.
'
\
get_privacy'
,
'update_callback' => NAMESPACE . ' \ set_privacy' ,
'schema'
=> [
'description' => __( 'Fairfax Visibility' , 'ffx' ),
'type'
=>
'boolean'
,
],
]
);
 
                Inherited from WordPress
Underscore
Backbone $ jQuery
 
                Inherited from WordPress
Underscore
Backbone $ jQuery % WordPress REST API client library
 
                The Post model ○ author ○ categories ○ comment_status ○ content ○ date ○ date_gmt ○ excerpt ○ featured_media ○ ffx-legal-status ○ ffx-post-state ○ ffx-sources ○ ffx-tags ○ ffx_advertisements ○ ffx_advertiser_logo ○ ffx_advertiser_name ○ ffx_article_tool ○ ffx_authors ○ ffx_bespoke_url ○ ffx_brief ○ ffx_collab_authors ○ ffx_collab_editor ○ ffx_collab_watchers ○ ffx_comments ○ ffx_comments_open ○ ffx_commercial_content_ty pe ○ ffx_correction ○ ffx_correction_text ○ ffx_format ○ ffx_identifier ○ ffx_in_numbers ○ ffx_in_numbers_title ○ ffx_index_headline ○ ffx_intro ○ ffx_label ○ ffx_last_updated ○ ffx_major_update ○ ffx_misc_update ○ ffx_off_time ○ ffx_primary_tag ○ ffx_private ○ ffx_seo_description ○ ffx_seo_news_keywords ○ ffx_seo_noindex ○ ffx_seo_title ○ ffx_slack_channel ○ ffx_sponsor ○ ffx_sports_score ○ ffx_syndication ○ ffx_talking_points ○ ffx_talking_points_live_title ○ ffx_talking_points_title ○ ffx_url ○ ffx_url_content ○ ffx_url_slug ○ ffx_why_it_matters ○ ffx_writeoff ○ format ○ id ○ meta ○ password ○ ping_status ○ slug ○ status ○ sticky ○ template ○ title
 
                wp.api.models.Post.prototype.args.ffx_private
{
required:
false
,
description:
"Fairfax Visibility."
,
type:
"boolean"
}
 
                 
                 
                Post meta of the past
 
                wp_insert_post ○ update post content % create new revision ○ update taxonomies ○ update meta data
 
                Update via REST API ○ update post content & create new revision ○ update taxonomies ○ update meta data
 
                add_filter( 'rest_pre_dispatch' , 'move_revision_callback' , 10 , 3 ); /** Move the 'wp_save_post_revision' callback for REST requests. */ function move_revision_callback( $result , $unused , $request ) {
/* SNIP: Check for post update request */
// Move default wp_save_post_revision callback.
remove_action( 'post_updated' , 'wp_save_post_revision' , 10 ); add_action( 'rest_request_after_callbacks' , 'wp_save_post_revision' );
return $result ; // Support other filters. }
 
                add_filter( 'rest_pre_dispatch' , 'move_revision_callback' , 10 , 3 ); /** Move the 'wp_save_post_revision' callback for REST requests. */ function move_revision_callback( $result , $unused , $request ) {
/* SNIP: Check for post update request */
// Move default wp_save_post_revision callback.
remove_action( 'post_updated' , 'wp_save_post_revision' , 10 ); add_action( 'rest_request_after_callbacks' , 'wp_save_post_revision' );
return $result ; // Support other filters. }
 
                add_filter( 'rest_request_after_callbacks' , 'reset_revision_cb' ); /** Reset 'wp_save_post_revision' after REST requests. */ function reset_revision_cb( $response ) {
// Move default wp_save_post_revision callback.
add_action( 'post_updated' , 'wp_save_post_revision' , 10 );
return $response ; // Support other filters. }
 
                The stack
 
                The stack
 
                Why WordPress?
 
                [ img id= "9550846e3098113e9fe16878fcbc26e23580e8ea" altText= "..." caption= "..." … ][/ img ] "body" : "<x-placeholder id='2c62…d2'></x-placeholder>" , "bodyPlaceholders" : {
"2c62…d2" : {
"type" : "image" ,
"data" : {
"id"
:
"9550846e3098113e9fe16878fcbc26e23580e8ea"
,
"caption"
:
"Lin-Manuel Miranda in Hamilton the Musical"
,
"credit"
:
"Production, Hamilton the Musical"
,
"aspect"
:
1.5
,
"cropWidth"
:
300
,
 
                [ img id= "9550846e3098113e9fe16878fcbc26e23580e8ea" altText= "..." caption= "..." … ][/ img ] "body" : "<x-placeholder id='2c62…d2'></x-placeholder>" , "bodyPlaceholders" : {
"2c62…d2" : {
"type" : "image" ,
"data" : {
"id"
:
"9550846e3098113e9fe16878fcbc26e23580e8ea"
,
"caption"
:
"Lin-Manuel Miranda in Hamilton the Musical"
,
"credit"
:
"Production, Hamilton the Musical"
,
"aspect"
:
1.5
,
"cropWidth"
:
300
,
 
                [ img id= "9550846e3098113e9fe16878fcbc26e23580e8ea" altText= "..." caption= "..." … ][/ img ] "body" : "<x-placeholder id='2c62…d2'></x-placeholder>" , "bodyPlaceholders" : {
"2c62…d2" : {
"type" : "image" ,
"data" : {
"id"
:
"9550846e3098113e9fe16878fcbc26e23580e8ea"
,
"caption"
:
"Lin-Manuel Miranda in Hamilton the Musical"
,
"credit"
:
"Production, Hamilton the Musical"
,
"aspect"
:
1.5
,
"cropWidth"
:
300
,
 
                Project retro
 
                 
                Committers 31
 
                Commits 11,737
 
                “ My job is code review and fooling myself that today I really will pick up a ticket. Me
 
                My guilty secret
 
                My guilty secret I like code review
 
                Code review
is productivity
 
                February 11, 2018 theage.com.au
 
                February 12, 2018 theage.com.au
 
                Slides and white paper
○
pwcc.cc/go/loop2018 Thank you
