filteringnoticestream.php 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116
  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($upstream)
  49. {
  50. $this->upstream = $upstream;
  51. }
  52. abstract function filter($notice);
  53. function getNoticeIds($offset, $limit, $since_id, $max_id)
  54. {
  55. // "offset" is virtual; we have to get a lot
  56. $total = $offset + $limit;
  57. $filtered = array();
  58. $startAt = 0;
  59. $askFor = $total;
  60. // Keep going till we have $total notices in $notices array,
  61. // or we get nothing from upstream.
  62. $results = null;
  63. $round = 0;
  64. do {
  65. $raw = $this->upstream->getNotices($startAt, $askFor, $since_id, $max_id);
  66. $results = $raw->N;
  67. if ($results == 0) {
  68. break;
  69. }
  70. $notices = $raw->fetchAll();
  71. $this->prefill($notices);
  72. foreach ($notices as $notice) {
  73. if ($this->filter($notice)) {
  74. $filtered[] = $notice->id;
  75. if (count($filtered) >= $total) {
  76. break;
  77. }
  78. }
  79. }
  80. // XXX: make these smarter; factor hit rate into $askFor
  81. $startAt += $askFor;
  82. $hits = count($filtered);
  83. $lastAsk = $askFor;
  84. if ($hits === 0) {
  85. $askFor = max(min(2 * $askFor, NOTICES_PER_PAGE * 50), NOTICES_PER_PAGE);
  86. } else {
  87. $askFor = max(min(intval(ceil(($total - $hits)*$startAt/$hits)), NOTICES_PER_PAGE * 50), NOTICES_PER_PAGE);
  88. }
  89. $round++;
  90. } while (count($filtered) < $total && $results >= $lastAsk);
  91. return array_slice(array_values($filtered), $offset, $limit);
  92. }
  93. function prefill($notices)
  94. {
  95. return;
  96. }
  97. }