readers.py 7.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281
  1. # copyright (c) 2024 PaddlePaddle Authors. All Rights Reserve.
  2. #
  3. # Licensed under the Apache License, Version 2.0 (the "License");
  4. # you may not use this file except in compliance with the License.
  5. # You may obtain a copy of the License at
  6. #
  7. # http://www.apache.org/licenses/LICENSE-2.0
  8. #
  9. # Unless required by applicable law or agreed to in writing, software
  10. # distributed under the License is distributed on an "AS IS" BASIS,
  11. # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  12. # See the License for the specific language governing permissions and
  13. # limitations under the License.
  14. import enum
  15. import itertools
  16. import cv2
  17. from PIL import Image, ImageOps
  18. import pandas as pd
  19. __all__ = ["ReaderType", "ImageReader", "VideoReader", "TSReader"]
  20. class ReaderType(enum.Enum):
  21. """ReaderType"""
  22. IMAGE = 1
  23. GENERATIVE = 2
  24. POINT_CLOUD = 3
  25. JSON = 4
  26. TS = 5
  27. class _BaseReader(object):
  28. """_BaseReader"""
  29. def __init__(self, backend, **bk_args):
  30. super().__init__()
  31. if len(bk_args) == 0:
  32. bk_args = self.get_default_backend_args()
  33. self.bk_type = backend
  34. self.bk_args = bk_args
  35. self._backend = self.get_backend()
  36. def read(self, in_path):
  37. """read file from path"""
  38. raise NotImplementedError
  39. def get_backend(self, bk_args=None):
  40. """get the backend"""
  41. if bk_args is None:
  42. bk_args = self.bk_args
  43. return self._init_backend(self.bk_type, bk_args)
  44. def set_backend(self, backend, **bk_args):
  45. self.bk_type = backend
  46. self.bk_args = bk_args
  47. self._backend = self.get_backend()
  48. def _init_backend(self, bk_type, bk_args):
  49. """init backend"""
  50. raise NotImplementedError
  51. def get_type(self):
  52. """get type"""
  53. raise NotImplementedError
  54. def get_default_backend_args(self):
  55. """get default backend arguments"""
  56. return {}
  57. class ImageReader(_BaseReader):
  58. """ImageReader"""
  59. def __init__(self, backend="opencv", **bk_args):
  60. super().__init__(backend=backend, **bk_args)
  61. def read(self, in_path):
  62. """read the image file from path"""
  63. arr = self._backend.read_file(in_path)
  64. return arr
  65. def _init_backend(self, bk_type, bk_args):
  66. """init backend"""
  67. if bk_type == "opencv":
  68. return OpenCVImageReaderBackend(**bk_args)
  69. elif bk_type == "pil" or bk_type == "pillow":
  70. return PILImageReaderBackend(**bk_args)
  71. else:
  72. raise ValueError("Unsupported backend type")
  73. def get_type(self):
  74. """get type"""
  75. return ReaderType.IMAGE
  76. class _GenerativeReader(_BaseReader):
  77. """_GenerativeReader"""
  78. def get_type(self):
  79. """get type"""
  80. return ReaderType.GENERATIVE
  81. def is_generative_reader(reader):
  82. """is_generative_reader"""
  83. return isinstance(reader, _GenerativeReader)
  84. class VideoReader(_GenerativeReader):
  85. """VideoReader"""
  86. def __init__(
  87. self,
  88. backend="opencv",
  89. st_frame_id=0,
  90. max_num_frames=None,
  91. auto_close=True,
  92. **bk_args,
  93. ):
  94. super().__init__(backend=backend, **bk_args)
  95. self.st_frame_id = st_frame_id
  96. self.max_num_frames = max_num_frames
  97. self.auto_close = auto_close
  98. def read(self, in_path):
  99. """read vide file from path"""
  100. self._backend.set_pos(self.st_frame_id)
  101. gen = self._backend.read_file(in_path)
  102. if self.num_frames is not None:
  103. gen = itertools.islice(gen, self.num_frames)
  104. yield from gen
  105. if self.auto_close:
  106. self._backend.close()
  107. def _init_backend(self, bk_type, bk_args):
  108. """init backend"""
  109. if bk_type == "opencv":
  110. return OpenCVVideoReaderBackend(**bk_args)
  111. else:
  112. raise ValueError("Unsupported backend type")
  113. class _BaseReaderBackend(object):
  114. """_BaseReaderBackend"""
  115. def read_file(self, in_path):
  116. """read file from path"""
  117. raise NotImplementedError
  118. class _ImageReaderBackend(_BaseReaderBackend):
  119. """_ImageReaderBackend"""
  120. pass
  121. class OpenCVImageReaderBackend(_ImageReaderBackend):
  122. """OpenCVImageReaderBackend"""
  123. def __init__(self, flags=cv2.IMREAD_COLOR):
  124. super().__init__()
  125. self.flags = flags
  126. def read_file(self, in_path):
  127. """read image file from path by OpenCV"""
  128. return cv2.imread(in_path, flags=self.flags)
  129. class PILImageReaderBackend(_ImageReaderBackend):
  130. """PILImageReaderBackend"""
  131. def __init__(self):
  132. super().__init__()
  133. def read_file(self, in_path):
  134. """read image file from path by PIL"""
  135. return ImageOps.exif_transpose(Image.open(in_path))
  136. class _VideoReaderBackend(_BaseReaderBackend):
  137. """_VideoReaderBackend"""
  138. def set_pos(self, pos):
  139. """set pos"""
  140. raise NotImplementedError
  141. def close(self):
  142. """close io"""
  143. raise NotImplementedError
  144. class OpenCVVideoReaderBackend(_VideoReaderBackend):
  145. """OpenCVVideoReaderBackend"""
  146. def __init__(self, **bk_args):
  147. super().__init__()
  148. self.cap_init_args = bk_args
  149. self._cap = None
  150. self._pos = 0
  151. self._max_num_frames = None
  152. def read_file(self, in_path):
  153. """read vidio file from path"""
  154. if self._cap is not None:
  155. self._cap_release()
  156. self._cap = self._cap_open(in_path)
  157. if self._pos is not None:
  158. self._cap_set_pos()
  159. return self._read_frames(self._cap)
  160. def _read_frames(self, cap):
  161. """read frames"""
  162. while True:
  163. ret, frame = cap.read()
  164. if not ret:
  165. break
  166. yield frame
  167. self._cap_release()
  168. def _cap_open(self, video_path):
  169. self._cap = cv2.VideoCapture(video_path, **self.cap_init_args)
  170. if not self._cap.isOpened():
  171. raise RuntimeError(f"Failed to open {video_path}")
  172. return self._cap
  173. def _cap_release(self):
  174. self._cap.release()
  175. def _cap_set_pos(self):
  176. self._cap.set(cv2.CAP_PROP_POS_FRAMES, self._pos)
  177. def set_pos(self, pos):
  178. self._pos = pos
  179. def close(self):
  180. if self._cap is not None:
  181. self._cap_release()
  182. self._cap = None
  183. class TSReader(_BaseReader):
  184. """TSReader"""
  185. def __init__(self, backend="pandas", **bk_args):
  186. super().__init__(backend=backend, **bk_args)
  187. def read(self, in_path):
  188. """read the image file from path"""
  189. arr = self._backend.read_file(in_path)
  190. return arr
  191. def _init_backend(self, bk_type, bk_args):
  192. """init backend"""
  193. if bk_type == "pandas":
  194. return PandasTSReaderBackend(**bk_args)
  195. else:
  196. raise ValueError("Unsupported backend type")
  197. def get_type(self):
  198. """get type"""
  199. return ReaderType.TS
  200. class _TSReaderBackend(_BaseReaderBackend):
  201. """_TSReaderBackend"""
  202. pass
  203. class PandasTSReaderBackend(_TSReaderBackend):
  204. """PandasTSReaderBackend"""
  205. def __init__(self):
  206. super().__init__()
  207. def read_file(self, in_path):
  208. """read image file from path by OpenCV"""
  209. return pd.read_csv(in_path)