2013-01-24 17:23:05 -08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @group conpherence
|
|
|
|
*/
|
|
|
|
final class ConpherenceEditor extends PhabricatorApplicationTransactionEditor {
|
|
|
|
|
2013-05-30 16:37:51 -07:00
|
|
|
const ERROR_EMPTY_PARTICIPANTS = 'error-empty-participants';
|
|
|
|
const ERROR_EMPTY_MESSAGE = 'error-empty-message';
|
|
|
|
|
|
|
|
public static function createConpherence(
|
|
|
|
PhabricatorUser $creator,
|
|
|
|
array $participant_phids,
|
|
|
|
$title,
|
|
|
|
$message,
|
|
|
|
PhabricatorContentSource $source) {
|
|
|
|
|
|
|
|
$conpherence = id(new ConpherenceThread())
|
|
|
|
->attachParticipants(array())
|
|
|
|
->attachFilePHIDs(array())
|
|
|
|
->setMessageCount(0);
|
|
|
|
$files = array();
|
|
|
|
$errors = array();
|
|
|
|
if (empty($participant_phids)) {
|
|
|
|
$errors[] = self::ERROR_EMPTY_PARTICIPANTS;
|
|
|
|
} else {
|
|
|
|
$participant_phids[] = $creator->getPHID();
|
|
|
|
$participant_phids = array_unique($participant_phids);
|
|
|
|
$conpherence->setRecentParticipantPHIDs(
|
|
|
|
array_slice($participant_phids, 0, 10));
|
|
|
|
}
|
|
|
|
|
|
|
|
if (empty($message)) {
|
|
|
|
$errors[] = self::ERROR_EMPTY_MESSAGE;
|
|
|
|
}
|
|
|
|
|
|
|
|
$file_phids =
|
|
|
|
PhabricatorMarkupEngine::extractFilePHIDsFromEmbeddedFiles(
|
|
|
|
array($message));
|
|
|
|
if ($file_phids) {
|
|
|
|
$files = id(new PhabricatorFileQuery())
|
|
|
|
->setViewer($creator)
|
|
|
|
->withPHIDs($file_phids)
|
|
|
|
->execute();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$errors) {
|
|
|
|
$xactions = array();
|
|
|
|
$xactions[] = id(new ConpherenceTransaction())
|
|
|
|
->setTransactionType(ConpherenceTransactionType::TYPE_PARTICIPANTS)
|
|
|
|
->setNewValue(array('+' => $participant_phids));
|
|
|
|
if ($files) {
|
|
|
|
$xactions[] = id(new ConpherenceTransaction())
|
|
|
|
->setTransactionType(ConpherenceTransactionType::TYPE_FILES)
|
|
|
|
->setNewValue(array('+' => mpull($files, 'getPHID')));
|
|
|
|
}
|
|
|
|
if ($title) {
|
|
|
|
$xactions[] = id(new ConpherenceTransaction())
|
|
|
|
->setTransactionType(ConpherenceTransactionType::TYPE_TITLE)
|
|
|
|
->setNewValue($title);
|
|
|
|
}
|
|
|
|
$xactions[] = id(new ConpherenceTransaction())
|
|
|
|
->setTransactionType(PhabricatorTransactions::TYPE_COMMENT)
|
|
|
|
->attachComment(
|
|
|
|
id(new ConpherenceTransactionComment())
|
|
|
|
->setContent($message)
|
|
|
|
->setConpherencePHID($conpherence->getPHID()));
|
|
|
|
|
|
|
|
id(new ConpherenceEditor())
|
|
|
|
->setContentSource($source)
|
|
|
|
->setContinueOnNoEffect(true)
|
|
|
|
->setActor($creator)
|
|
|
|
->applyTransactions($conpherence, $xactions);
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return array($errors, $conpherence);
|
|
|
|
}
|
|
|
|
|
2013-01-26 19:56:39 -08:00
|
|
|
public function generateTransactionsFromText(
|
|
|
|
ConpherenceThread $conpherence,
|
|
|
|
$text) {
|
|
|
|
|
|
|
|
$files = array();
|
|
|
|
$file_phids =
|
|
|
|
PhabricatorMarkupEngine::extractFilePHIDsFromEmbeddedFiles(
|
2013-02-19 13:33:10 -08:00
|
|
|
array($text));
|
2013-01-26 19:56:39 -08:00
|
|
|
// Since these are extracted from text, we might be re-including the
|
|
|
|
// same file -- e.g. a mock under discussion. Filter files we
|
|
|
|
// already have.
|
|
|
|
$existing_file_phids = $conpherence->getFilePHIDs();
|
|
|
|
$file_phids = array_diff($file_phids, $existing_file_phids);
|
|
|
|
if ($file_phids) {
|
|
|
|
$files = id(new PhabricatorFileQuery())
|
|
|
|
->setViewer($this->getActor())
|
|
|
|
->withPHIDs($file_phids)
|
|
|
|
->execute();
|
|
|
|
}
|
|
|
|
$xactions = array();
|
|
|
|
if ($files) {
|
|
|
|
$xactions[] = id(new ConpherenceTransaction())
|
|
|
|
->setTransactionType(ConpherenceTransactionType::TYPE_FILES)
|
|
|
|
->setNewValue(array('+' => mpull($files, 'getPHID')));
|
|
|
|
}
|
|
|
|
$xactions[] = id(new ConpherenceTransaction())
|
|
|
|
->setTransactionType(PhabricatorTransactions::TYPE_COMMENT)
|
|
|
|
->attachComment(
|
|
|
|
id(new ConpherenceTransactionComment())
|
|
|
|
->setContent($text)
|
2013-02-19 13:33:10 -08:00
|
|
|
->setConpherencePHID($conpherence->getPHID()));
|
2013-01-26 19:56:39 -08:00
|
|
|
return $xactions;
|
|
|
|
}
|
|
|
|
|
2013-01-24 17:23:05 -08:00
|
|
|
public function getTransactionTypes() {
|
|
|
|
$types = parent::getTransactionTypes();
|
|
|
|
|
|
|
|
$types[] = PhabricatorTransactions::TYPE_COMMENT;
|
|
|
|
|
|
|
|
$types[] = ConpherenceTransactionType::TYPE_TITLE;
|
|
|
|
$types[] = ConpherenceTransactionType::TYPE_PARTICIPANTS;
|
|
|
|
$types[] = ConpherenceTransactionType::TYPE_FILES;
|
|
|
|
|
|
|
|
return $types;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getCustomTransactionOldValue(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
|
|
|
|
switch ($xaction->getTransactionType()) {
|
|
|
|
case ConpherenceTransactionType::TYPE_TITLE:
|
|
|
|
return $object->getTitle();
|
|
|
|
case ConpherenceTransactionType::TYPE_PARTICIPANTS:
|
|
|
|
return $object->getParticipantPHIDs();
|
|
|
|
case ConpherenceTransactionType::TYPE_FILES:
|
|
|
|
return $object->getFilePHIDs();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getCustomTransactionNewValue(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
|
|
|
|
switch ($xaction->getTransactionType()) {
|
|
|
|
case ConpherenceTransactionType::TYPE_TITLE:
|
|
|
|
return $xaction->getNewValue();
|
|
|
|
case ConpherenceTransactionType::TYPE_PARTICIPANTS:
|
|
|
|
case ConpherenceTransactionType::TYPE_FILES:
|
|
|
|
return $this->getPHIDTransactionNewValue($xaction);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-04-04 16:57:38 -07:00
|
|
|
/**
|
|
|
|
* We really only need a read lock if we have a comment. In that case, we
|
|
|
|
* must update the messagesCount field on the conpherence and
|
|
|
|
* seenMessagesCount(s) for the participant(s).
|
|
|
|
*/
|
|
|
|
protected function shouldReadLock(
|
2013-01-24 17:23:05 -08:00
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
|
2013-04-04 16:57:38 -07:00
|
|
|
$lock = false;
|
2013-01-24 17:23:05 -08:00
|
|
|
switch ($xaction->getTransactionType()) {
|
2013-04-04 16:57:38 -07:00
|
|
|
case PhabricatorTransactions::TYPE_COMMENT:
|
|
|
|
$lock = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $lock;
|
|
|
|
}
|
|
|
|
|
2013-08-14 17:32:16 -07:00
|
|
|
/**
|
|
|
|
* We need to apply initial effects IFF the conpherence is new. We must
|
|
|
|
* save the conpherence first thing to make sure we have an id and a phid.
|
|
|
|
*/
|
|
|
|
protected function shouldApplyInitialEffects(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
array $xactions) {
|
|
|
|
|
|
|
|
return !$object->getID();
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function applyInitialEffects(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
array $xactions) {
|
|
|
|
|
|
|
|
$object->save();
|
|
|
|
}
|
|
|
|
|
2013-04-04 16:57:38 -07:00
|
|
|
protected function applyCustomInternalTransaction(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
switch ($xaction->getTransactionType()) {
|
|
|
|
case PhabricatorTransactions::TYPE_COMMENT:
|
|
|
|
$object->setMessageCount((int)$object->getMessageCount() + 1);
|
|
|
|
break;
|
2013-01-24 17:23:05 -08:00
|
|
|
case ConpherenceTransactionType::TYPE_TITLE:
|
|
|
|
$object->setTitle($xaction->getNewValue());
|
|
|
|
break;
|
|
|
|
}
|
2013-04-04 16:57:38 -07:00
|
|
|
$this->updateRecentParticipantPHIDs($object, $xaction);
|
|
|
|
}
|
|
|
|
|
|
|
|
private function updateRecentParticipantPHIDs(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
|
|
|
|
$participants = $object->getRecentParticipantPHIDs();
|
|
|
|
array_unshift($participants, $xaction->getAuthorPHID());
|
|
|
|
$participants = array_slice(array_unique($participants), 0, 10);
|
|
|
|
|
|
|
|
$object->setRecentParticipantPHIDs($participants);
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function applyCustomExternalTransaction(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
PhabricatorApplicationTransaction $xaction) {
|
|
|
|
|
|
|
|
switch ($xaction->getTransactionType()) {
|
|
|
|
case ConpherenceTransactionType::TYPE_FILES:
|
|
|
|
$editor = id(new PhabricatorEdgeEditor())
|
|
|
|
->setActor($this->getActor());
|
|
|
|
$edge_type = PhabricatorEdgeConfig::TYPE_OBJECT_HAS_FILE;
|
2013-03-08 10:40:06 -08:00
|
|
|
$old = array_fill_keys($xaction->getOldValue(), true);
|
|
|
|
$new = array_fill_keys($xaction->getNewValue(), true);
|
|
|
|
$add_edges = array_keys(array_diff_key($new, $old));
|
|
|
|
$remove_edges = array_keys(array_diff_key($old, $new));
|
|
|
|
foreach ($add_edges as $file_phid) {
|
2013-01-24 17:23:05 -08:00
|
|
|
$editor->addEdge(
|
|
|
|
$object->getPHID(),
|
|
|
|
$edge_type,
|
2013-02-19 13:33:10 -08:00
|
|
|
$file_phid);
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
2013-03-08 10:40:06 -08:00
|
|
|
foreach ($remove_edges as $file_phid) {
|
|
|
|
$editor->removeEdge(
|
|
|
|
$object->getPHID(),
|
|
|
|
$edge_type,
|
|
|
|
$file_phid);
|
|
|
|
}
|
2013-01-24 17:23:05 -08:00
|
|
|
$editor->save();
|
|
|
|
break;
|
|
|
|
case ConpherenceTransactionType::TYPE_PARTICIPANTS:
|
2013-04-02 09:32:40 -07:00
|
|
|
$participants = $object->getParticipants();
|
|
|
|
|
|
|
|
$old_map = array_fuse($xaction->getOldValue());
|
|
|
|
$new_map = array_fuse($xaction->getNewValue());
|
|
|
|
|
|
|
|
$remove = array_keys(array_diff_key($old_map, $new_map));
|
|
|
|
foreach ($remove as $phid) {
|
|
|
|
$remove_participant = $participants[$phid];
|
|
|
|
$remove_participant->delete();
|
|
|
|
unset($participants[$phid]);
|
|
|
|
}
|
|
|
|
|
|
|
|
$add = array_keys(array_diff_key($new_map, $old_map));
|
|
|
|
foreach ($add as $phid) {
|
|
|
|
if ($phid == $this->getActor()->getPHID()) {
|
2013-01-24 17:23:05 -08:00
|
|
|
$status = ConpherenceParticipationStatus::UP_TO_DATE;
|
2013-04-04 16:57:38 -07:00
|
|
|
$message_count = $object->getMessageCount();
|
2013-01-24 17:23:05 -08:00
|
|
|
} else {
|
|
|
|
$status = ConpherenceParticipationStatus::BEHIND;
|
2013-04-04 16:57:38 -07:00
|
|
|
$message_count = 0;
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
2013-04-02 09:32:40 -07:00
|
|
|
$participants[$phid] =
|
2013-03-26 13:30:35 -07:00
|
|
|
id(new ConpherenceParticipant())
|
2013-01-24 17:23:05 -08:00
|
|
|
->setConpherencePHID($object->getPHID())
|
2013-04-02 09:32:40 -07:00
|
|
|
->setParticipantPHID($phid)
|
2013-01-24 17:23:05 -08:00
|
|
|
->setParticipationStatus($status)
|
|
|
|
->setDateTouched(time())
|
|
|
|
->setBehindTransactionPHID($xaction->getPHID())
|
2013-04-04 16:57:38 -07:00
|
|
|
->setSeenMessageCount($message_count)
|
2013-01-24 17:23:05 -08:00
|
|
|
->save();
|
|
|
|
}
|
2013-03-26 13:30:35 -07:00
|
|
|
$object->attachParticipants($participants);
|
2013-01-24 17:23:05 -08:00
|
|
|
break;
|
2013-04-26 10:30:41 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function applyFinalEffects(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
array $xactions) {
|
|
|
|
|
|
|
|
// update everyone's participation status on the last xaction -only-
|
|
|
|
$xaction = end($xactions);
|
|
|
|
$xaction_phid = $xaction->getPHID();
|
|
|
|
$behind = ConpherenceParticipationStatus::BEHIND;
|
|
|
|
$up_to_date = ConpherenceParticipationStatus::UP_TO_DATE;
|
|
|
|
$participants = $object->getParticipants();
|
|
|
|
$user = $this->getActor();
|
|
|
|
$time = time();
|
|
|
|
foreach ($participants as $phid => $participant) {
|
|
|
|
if ($phid != $user->getPHID()) {
|
|
|
|
if ($participant->getParticipationStatus() != $behind) {
|
|
|
|
$participant->setBehindTransactionPHID($xaction_phid);
|
|
|
|
// decrement one as this is the message putting them behind!
|
|
|
|
$participant->setSeenMessageCount($object->getMessageCount() - 1);
|
|
|
|
}
|
|
|
|
$participant->setParticipationStatus($behind);
|
|
|
|
$participant->setDateTouched($time);
|
|
|
|
} else {
|
|
|
|
$participant->setSeenMessageCount($object->getMessageCount());
|
|
|
|
$participant->setParticipationStatus($up_to_date);
|
|
|
|
$participant->setDateTouched($time);
|
|
|
|
}
|
|
|
|
$participant->save();
|
|
|
|
}
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function mergeTransactions(
|
|
|
|
PhabricatorApplicationTransaction $u,
|
|
|
|
PhabricatorApplicationTransaction $v) {
|
|
|
|
|
|
|
|
$type = $u->getTransactionType();
|
|
|
|
switch ($type) {
|
|
|
|
case ConpherenceTransactionType::TYPE_TITLE:
|
|
|
|
return $v;
|
|
|
|
case ConpherenceTransactionType::TYPE_FILES:
|
|
|
|
case ConpherenceTransactionType::TYPE_PARTICIPANTS:
|
2013-03-28 08:34:34 -07:00
|
|
|
return $this->mergePHIDOrEdgeTransactions($u, $v);
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
return parent::mergeTransactions($u, $v);
|
|
|
|
}
|
|
|
|
|
2013-09-13 15:08:17 -07:00
|
|
|
protected function shouldSendMail(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
array $xactions) {
|
2013-01-25 16:03:54 -08:00
|
|
|
return true;
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function buildReplyHandler(PhabricatorLiskDAO $object) {
|
|
|
|
return id(new ConpherenceReplyHandler())
|
2013-01-25 16:03:54 -08:00
|
|
|
->setActor($this->getActor())
|
2013-01-24 17:23:05 -08:00
|
|
|
->setMailReceiver($object);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function buildMailTemplate(PhabricatorLiskDAO $object) {
|
|
|
|
$id = $object->getID();
|
|
|
|
$title = $object->getTitle();
|
2013-01-25 16:03:54 -08:00
|
|
|
if (!$title) {
|
|
|
|
$title = pht(
|
|
|
|
'%s sent you a message.',
|
2013-02-19 13:33:10 -08:00
|
|
|
$this->getActor()->getUserName());
|
2013-01-25 16:03:54 -08:00
|
|
|
}
|
2013-01-24 17:23:05 -08:00
|
|
|
$phid = $object->getPHID();
|
|
|
|
|
|
|
|
return id(new PhabricatorMetaMTAMail())
|
2013-01-25 16:03:54 -08:00
|
|
|
->setSubject("E{$id}: {$title}")
|
|
|
|
->addHeader('Thread-Topic', "E{$id}: {$phid}");
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
protected function getMailTo(PhabricatorLiskDAO $object) {
|
2013-04-02 09:32:40 -07:00
|
|
|
$to_phids = array();
|
2013-01-24 17:23:05 -08:00
|
|
|
$participants = $object->getParticipants();
|
2013-04-02 09:32:40 -07:00
|
|
|
if (empty($participants)) {
|
|
|
|
return $to_phids;
|
|
|
|
}
|
2013-03-26 13:30:35 -07:00
|
|
|
$preferences = id(new PhabricatorUserPreferences())
|
|
|
|
->loadAllWhere('userPHID in (%Ls)', array_keys($participants));
|
|
|
|
$preferences = mpull($preferences, null, 'getUserPHID');
|
|
|
|
foreach ($participants as $phid => $participant) {
|
|
|
|
$default = ConpherenceSettings::EMAIL_ALWAYS;
|
|
|
|
$preference = idx($preferences, $phid);
|
|
|
|
if ($preference) {
|
|
|
|
$default = $preference->getPreference(
|
|
|
|
PhabricatorUserPreferences::PREFERENCE_CONPH_NOTIFICATIONS,
|
|
|
|
ConpherenceSettings::EMAIL_ALWAYS);
|
|
|
|
}
|
|
|
|
$settings = $participant->getSettings();
|
|
|
|
$notifications = idx(
|
|
|
|
$settings,
|
|
|
|
'notifications',
|
|
|
|
$default);
|
|
|
|
if ($notifications == ConpherenceSettings::EMAIL_ALWAYS) {
|
|
|
|
$to_phids[] = $phid;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return $to_phids;
|
2013-01-24 17:23:05 -08:00
|
|
|
}
|
|
|
|
|
2013-01-25 16:03:54 -08:00
|
|
|
protected function getMailCC(PhabricatorLiskDAO $object) {
|
|
|
|
return array();
|
|
|
|
}
|
|
|
|
|
2013-01-24 17:23:05 -08:00
|
|
|
protected function buildMailBody(
|
|
|
|
PhabricatorLiskDAO $object,
|
|
|
|
array $xactions) {
|
|
|
|
|
|
|
|
$body = parent::buildMailBody($object, $xactions);
|
|
|
|
$body->addTextSection(
|
|
|
|
pht('CONPHERENCE DETAIL'),
|
|
|
|
PhabricatorEnv::getProductionURI('/conpherence/'.$object->getID().'/'));
|
|
|
|
|
|
|
|
return $body;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function getMailSubjectPrefix() {
|
|
|
|
return PhabricatorEnv::getEnvConfig('metamta.conpherence.subject-prefix');
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function supportsFeed() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
protected function supportsSearch() {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|