filteringnoticestream.php 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119
  1. <?php
  2. /**
  3. * StatusNet - the distributed open-source microblogging tool
  4. * Copyright (C) 2011, StatusNet, Inc.
  5. *
  6. * A notice stream that filters its upstream content
  7. *
  8. * PHP version 5
  9. *
  10. * This program is free software: you can redistribute it and/or modify
  11. * it under the terms of the GNU Affero General Public License as published by
  12. * the Free Software Foundation, either version 3 of the License, or
  13. * (at your option) any later version.
  14. *
  15. * This program is distributed in the hope that it will be useful,
  16. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  17. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  18. * GNU Affero General Public License for more details.
  19. *
  20. * You should have received a copy of the GNU Affero General Public License
  21. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  22. *
  23. * @category Stream
  24. * @package StatusNet
  25. * @author Evan Prodromou <evan@status.net>
  26. * @copyright 2011 StatusNet, Inc.
  27. * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPL 3.0
  28. * @link http://status.net/
  29. */
  30. if (!defined('STATUSNET')) {
  31. // This check helps protect against security problems;
  32. // your code file can't be executed directly from the web.
  33. exit(1);
  34. }
  35. /**
  36. * A class for presenting a filtered notice stream based on an upstream stream
  37. *
  38. * @category Stream
  39. * @package StatusNet
  40. * @author Evan Prodromou <evan@status.net>
  41. * @copyright 2011 StatusNet, Inc.
  42. * @license http://www.fsf.org/licensing/licenses/agpl-3.0.html AGPL 3.0
  43. * @link http://status.net/
  44. */
  45. abstract class FilteringNoticeStream extends NoticeStream
  46. {
  47. protected $upstream;
  48. function __construct(NoticeStream $upstream)
  49. {
  50. $this->upstream = $upstream;
  51. }
  52. /**
  53. * @return boolean true if we allow it, false if we deny it
  54. */
  55. abstract protected function filter(Notice $notice);
  56. function getNoticeIds($offset, $limit, $since_id, $max_id)
  57. {
  58. // "offset" is virtual; we have to get a lot
  59. $total = $offset + $limit;
  60. $filtered = array();
  61. $startAt = 0;
  62. $askFor = $total;
  63. // Keep going till we have $total notices in $notices array,
  64. // or we get nothing from upstream.
  65. $results = null;
  66. $round = 0;
  67. do {
  68. $raw = $this->upstream->getNotices($startAt, $askFor, $since_id, $max_id);
  69. $results = $raw->N;
  70. if ($results == 0) {
  71. break;
  72. }
  73. $notices = $raw->fetchAll();
  74. $this->prefill($notices);
  75. foreach ($notices as $notice) {
  76. if ($this->filter($notice)) {
  77. $filtered[] = $notice->id;
  78. if (count($filtered) >= $total) {
  79. break;
  80. }
  81. }
  82. }
  83. // XXX: make these smarter; factor hit rate into $askFor
  84. $startAt += $askFor;
  85. $hits = count($filtered);
  86. $lastAsk = $askFor;
  87. if ($hits === 0) {
  88. $askFor = max(min(2 * $askFor, NOTICES_PER_PAGE * 50), NOTICES_PER_PAGE);
  89. } else {
  90. $askFor = max(min(intval(ceil(($total - $hits)*$startAt/$hits)), NOTICES_PER_PAGE * 50), NOTICES_PER_PAGE);
  91. }
  92. $round++;
  93. } while (count($filtered) < $total && $results >= $lastAsk);
  94. return array_slice(array_values($filtered), $offset, $limit);
  95. }
  96. function prefill($notices)
  97. {
  98. return;
  99. }
  100. }