Page Menu
Home
c4science
Search
Configure Global Search
Log In
Files
F104552277
PhabricatorNotificationBuilder.php
No One
Temporary
Actions
Download File
Edit File
Delete File
View Transforms
Subscribe
Mute Notifications
Award Token
Subscribers
None
File Metadata
Details
File Info
Storage
Attached
Created
Mon, Mar 10, 09:56
Size
4 KB
Mime Type
text/x-php
Expires
Wed, Mar 12, 09:56 (2 d)
Engine
blob
Format
Raw Data
Handle
24796066
Attached To
rPH Phabricator
PhabricatorNotificationBuilder.php
View Options
<?php
final
class
PhabricatorNotificationBuilder
extends
Phobject
{
private
$stories
;
private
$parsedStories
;
private
$user
=
null
;
public
function
__construct
(
array
$stories
)
{
assert_instances_of
(
$stories
,
'PhabricatorFeedStory'
);
$this
->
stories
=
$stories
;
}
public
function
setUser
(
$user
)
{
$this
->
user
=
$user
;
return
$this
;
}
private
function
parseStories
()
{
if
(
$this
->
parsedStories
)
{
return
$this
->
parsedStories
;
}
$stories
=
$this
->
stories
;
$stories
=
mpull
(
$stories
,
null
,
'getChronologicalKey'
);
// Aggregate notifications. Generally, we can aggregate notifications only
// by object, e.g. "a updated T123" and "b updated T123" can become
// "a and b updated T123", but we can't combine "a updated T123" and
// "a updated T234" into "a updated T123 and T234" because there would be
// nowhere sensible for the notification to link to, and no reasonable way
// to unambiguously clear it.
// Build up a map of all the possible aggregations.
$chronokey_map
=
array
();
$aggregation_map
=
array
();
$agg_types
=
array
();
foreach
(
$stories
as
$chronokey
=>
$story
)
{
$chronokey_map
[
$chronokey
]
=
$story
->
getNotificationAggregations
();
foreach
(
$chronokey_map
[
$chronokey
]
as
$key
=>
$type
)
{
$agg_types
[
$key
]
=
$type
;
$aggregation_map
[
$key
][
'keys'
][
$chronokey
]
=
true
;
}
}
// Repeatedly select the largest available aggregation until none remain.
$aggregated_stories
=
array
();
while
(
$aggregation_map
)
{
// Count the size of each aggregation, removing any which will consume
// fewer than 2 stories.
foreach
(
$aggregation_map
as
$key
=>
$dict
)
{
$size
=
count
(
$dict
[
'keys'
]);
if
(
$size
>
1
)
{
$aggregation_map
[
$key
][
'size'
]
=
$size
;
}
else
{
unset
(
$aggregation_map
[
$key
]);
}
}
// If we're out of aggregations, break out.
if
(!
$aggregation_map
)
{
break
;
}
// Select the aggregation we're going to make, and remove it from the
// map.
$aggregation_map
=
isort
(
$aggregation_map
,
'size'
);
$agg_info
=
idx
(
last
(
$aggregation_map
),
'keys'
);
$agg_key
=
last_key
(
$aggregation_map
);
unset
(
$aggregation_map
[
$agg_key
]);
// Select all the stories it aggregates, and remove them from the master
// list of stories and from all other possible aggregations.
$sub_stories
=
array
();
foreach
(
$agg_info
as
$chronokey
=>
$ignored
)
{
$sub_stories
[
$chronokey
]
=
$stories
[
$chronokey
];
unset
(
$stories
[
$chronokey
]);
foreach
(
$chronokey_map
[
$chronokey
]
as
$key
=>
$type
)
{
unset
(
$aggregation_map
[
$key
][
'keys'
][
$chronokey
]);
}
unset
(
$chronokey_map
[
$chronokey
]);
}
// Build the aggregate story.
krsort
(
$sub_stories
);
$story_class
=
$agg_types
[
$agg_key
];
$conv
=
array
(
head
(
$sub_stories
)->
getStoryData
());
$new_story
=
newv
(
$story_class
,
$conv
);
$new_story
->
setAggregateStories
(
$sub_stories
);
$aggregated_stories
[]
=
$new_story
;
}
// Combine the aggregate stories back into the list of stories.
$stories
=
array_merge
(
$stories
,
$aggregated_stories
);
$stories
=
mpull
(
$stories
,
null
,
'getChronologicalKey'
);
krsort
(
$stories
);
$this
->
parsedStories
=
$stories
;
return
$stories
;
}
public
function
buildView
()
{
$stories
=
$this
->
parseStories
();
$null_view
=
new
AphrontNullView
();
foreach
(
$stories
as
$story
)
{
try
{
$view
=
$story
->
renderView
();
}
catch
(
Exception
$ex
)
{
// TODO: Render a nice debuggable notice instead?
continue
;
}
$null_view
->
appendChild
(
$view
->
renderNotification
(
$this
->
user
));
}
return
$null_view
;
}
public
function
buildDict
()
{
$stories
=
$this
->
parseStories
();
$dict
=
array
();
foreach
(
$stories
as
$story
)
{
if
(
$story
instanceof
PhabricatorApplicationTransactionFeedStory
)
{
$dict
[]
=
array
(
'desktopReady'
=>
true
,
'title'
=>
$story
->
renderText
(),
'body'
=>
$story
->
renderTextBody
(),
'href'
=>
$story
->
getURI
(),
'icon'
=>
$story
->
getImageURI
(),
);
}
else
if
(
$story
instanceof
PhabricatorNotificationTestFeedStory
)
{
$dict
[]
=
array
(
'desktopReady'
=>
true
,
'title'
=>
pht
(
'Test Notification'
),
'body'
=>
$story
->
renderText
(),
'href'
=>
null
,
'icon'
=>
PhabricatorUser
::
getDefaultProfileImageURI
(),
);
}
else
{
$dict
[]
=
array
(
'desktopReady'
=>
false
,
'title'
=>
null
,
'body'
=>
null
,
'href'
=>
null
,
'icon'
=>
null
,
);
}
}
return
$dict
;
}
}
Event Timeline
Log In to Comment