pagination.py 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. # File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details.
  2. from typing import Any, List, Generic, TypeVar, Optional, cast
  3. from typing_extensions import Protocol, override, runtime_checkable
  4. from ._base_client import BasePage, PageInfo, BaseSyncPage, BaseAsyncPage
  5. __all__ = [
  6. "SyncPage",
  7. "AsyncPage",
  8. "SyncCursorPage",
  9. "AsyncCursorPage",
  10. "SyncConversationCursorPage",
  11. "AsyncConversationCursorPage",
  12. ]
  13. _T = TypeVar("_T")
  14. @runtime_checkable
  15. class CursorPageItem(Protocol):
  16. id: Optional[str]
  17. class SyncPage(BaseSyncPage[_T], BasePage[_T], Generic[_T]):
  18. """Note: no pagination actually occurs yet, this is for forwards-compatibility."""
  19. data: List[_T]
  20. object: str
  21. @override
  22. def _get_page_items(self) -> List[_T]:
  23. data = self.data
  24. if not data:
  25. return []
  26. return data
  27. @override
  28. def next_page_info(self) -> None:
  29. """
  30. This page represents a response that isn't actually paginated at the API level
  31. so there will never be a next page.
  32. """
  33. return None
  34. class AsyncPage(BaseAsyncPage[_T], BasePage[_T], Generic[_T]):
  35. """Note: no pagination actually occurs yet, this is for forwards-compatibility."""
  36. data: List[_T]
  37. object: str
  38. @override
  39. def _get_page_items(self) -> List[_T]:
  40. data = self.data
  41. if not data:
  42. return []
  43. return data
  44. @override
  45. def next_page_info(self) -> None:
  46. """
  47. This page represents a response that isn't actually paginated at the API level
  48. so there will never be a next page.
  49. """
  50. return None
  51. class SyncCursorPage(BaseSyncPage[_T], BasePage[_T], Generic[_T]):
  52. data: List[_T]
  53. has_more: Optional[bool] = None
  54. @override
  55. def _get_page_items(self) -> List[_T]:
  56. data = self.data
  57. if not data:
  58. return []
  59. return data
  60. @override
  61. def has_next_page(self) -> bool:
  62. has_more = self.has_more
  63. if has_more is not None and has_more is False:
  64. return False
  65. return super().has_next_page()
  66. @override
  67. def next_page_info(self) -> Optional[PageInfo]:
  68. data = self.data
  69. if not data:
  70. return None
  71. item = cast(Any, data[-1])
  72. if not isinstance(item, CursorPageItem) or item.id is None:
  73. # TODO emit warning log
  74. return None
  75. return PageInfo(params={"after": item.id})
  76. class AsyncCursorPage(BaseAsyncPage[_T], BasePage[_T], Generic[_T]):
  77. data: List[_T]
  78. has_more: Optional[bool] = None
  79. @override
  80. def _get_page_items(self) -> List[_T]:
  81. data = self.data
  82. if not data:
  83. return []
  84. return data
  85. @override
  86. def has_next_page(self) -> bool:
  87. has_more = self.has_more
  88. if has_more is not None and has_more is False:
  89. return False
  90. return super().has_next_page()
  91. @override
  92. def next_page_info(self) -> Optional[PageInfo]:
  93. data = self.data
  94. if not data:
  95. return None
  96. item = cast(Any, data[-1])
  97. if not isinstance(item, CursorPageItem) or item.id is None:
  98. # TODO emit warning log
  99. return None
  100. return PageInfo(params={"after": item.id})
  101. class SyncConversationCursorPage(BaseSyncPage[_T], BasePage[_T], Generic[_T]):
  102. data: List[_T]
  103. has_more: Optional[bool] = None
  104. last_id: Optional[str] = None
  105. @override
  106. def _get_page_items(self) -> List[_T]:
  107. data = self.data
  108. if not data:
  109. return []
  110. return data
  111. @override
  112. def has_next_page(self) -> bool:
  113. has_more = self.has_more
  114. if has_more is not None and has_more is False:
  115. return False
  116. return super().has_next_page()
  117. @override
  118. def next_page_info(self) -> Optional[PageInfo]:
  119. last_id = self.last_id
  120. if not last_id:
  121. return None
  122. return PageInfo(params={"after": last_id})
  123. class AsyncConversationCursorPage(BaseAsyncPage[_T], BasePage[_T], Generic[_T]):
  124. data: List[_T]
  125. has_more: Optional[bool] = None
  126. last_id: Optional[str] = None
  127. @override
  128. def _get_page_items(self) -> List[_T]:
  129. data = self.data
  130. if not data:
  131. return []
  132. return data
  133. @override
  134. def has_next_page(self) -> bool:
  135. has_more = self.has_more
  136. if has_more is not None and has_more is False:
  137. return False
  138. return super().has_next_page()
  139. @override
  140. def next_page_info(self) -> Optional[PageInfo]:
  141. last_id = self.last_id
  142. if not last_id:
  143. return None
  144. return PageInfo(params={"after": last_id})