2020-08-27 19:28:29 +00:00
|
|
|
# Copyright The PyTorch Lightning team.
|
|
|
|
#
|
|
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
# you may not use this file except in compliance with the License.
|
|
|
|
# You may obtain a copy of the License at
|
|
|
|
#
|
|
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
#
|
|
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
# See the License for the specific language governing permissions and
|
|
|
|
# limitations under the License.
|
2020-10-10 01:03:23 +00:00
|
|
|
|
2021-07-13 11:35:10 +00:00
|
|
|
from typing import Any, Iterable, Mapping, Union
|
2021-01-15 18:57:40 +00:00
|
|
|
|
2021-07-13 11:35:10 +00:00
|
|
|
import torch
|
2020-08-27 19:28:29 +00:00
|
|
|
from torch.utils.data import DataLoader, IterableDataset
|
|
|
|
|
|
|
|
from pytorch_lightning.utilities import rank_zero_warn
|
|
|
|
|
2021-07-26 11:37:35 +00:00
|
|
|
BType = Union[torch.Tensor, str, Mapping[Any, "BType"], Iterable["BType"]]
|
2021-07-13 11:35:10 +00:00
|
|
|
|
|
|
|
|
|
|
|
def extract_batch_size(batch: BType) -> int:
|
|
|
|
"""
|
|
|
|
Recursively unpack a batch to find a torch.Tensor.
|
|
|
|
|
|
|
|
Returns:
|
|
|
|
``len(tensor)`` when found, or ``1`` when it hits an empty or non iterable.
|
|
|
|
"""
|
|
|
|
if isinstance(batch, torch.Tensor):
|
|
|
|
return batch.size(0)
|
|
|
|
if isinstance(batch, str):
|
|
|
|
return len(batch)
|
|
|
|
if isinstance(batch, dict):
|
|
|
|
sample = next(iter(batch.values()), 1)
|
|
|
|
return extract_batch_size(sample)
|
|
|
|
if isinstance(batch, Iterable):
|
|
|
|
sample = next(iter(batch), 1)
|
|
|
|
return extract_batch_size(sample)
|
|
|
|
|
|
|
|
return 1
|
|
|
|
|
2020-08-27 19:28:29 +00:00
|
|
|
|
2021-08-10 06:39:00 +00:00
|
|
|
def has_iterable_dataset(dataloader: DataLoader) -> bool:
|
2021-07-26 11:37:35 +00:00
|
|
|
return hasattr(dataloader, "dataset") and isinstance(dataloader.dataset, IterableDataset)
|
2020-08-27 19:28:29 +00:00
|
|
|
|
|
|
|
|
|
|
|
def has_len(dataloader: DataLoader) -> bool:
|
2021-06-25 13:41:45 +00:00
|
|
|
"""
|
|
|
|
Checks if a given Dataloader has ``__len__`` method implemented i.e. if
|
|
|
|
it is a finite dataloader or infinite dataloader.
|
|
|
|
|
|
|
|
Raises:
|
|
|
|
ValueError:
|
|
|
|
If the length of Dataloader is 0, as it requires at least one batch
|
|
|
|
"""
|
2020-08-27 19:28:29 +00:00
|
|
|
|
|
|
|
try:
|
|
|
|
# try getting the length
|
|
|
|
if len(dataloader) == 0:
|
2021-07-26 11:37:35 +00:00
|
|
|
raise ValueError("`Dataloader` returned 0 length. Please make sure that it returns at least 1 batch")
|
2020-08-27 19:28:29 +00:00
|
|
|
has_len = True
|
|
|
|
except TypeError:
|
|
|
|
has_len = False
|
|
|
|
except NotImplementedError: # e.g. raised by torchtext if a batch_size_fn is used
|
|
|
|
has_len = False
|
|
|
|
|
2021-02-16 22:53:40 +00:00
|
|
|
if has_len and has_iterable_dataset(dataloader):
|
2020-08-27 19:28:29 +00:00
|
|
|
rank_zero_warn(
|
2021-07-26 11:37:35 +00:00
|
|
|
"Your `IterableDataset` has `__len__` defined."
|
|
|
|
" In combination with multi-process data loading (when num_workers > 1),"
|
|
|
|
" `__len__` could be inaccurate if each worker is not configured independently"
|
|
|
|
" to avoid having duplicate data."
|
2020-08-27 19:28:29 +00:00
|
|
|
)
|
|
|
|
return has_len
|
2021-01-04 19:57:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
def get_len(dataloader: DataLoader) -> Union[int, float]:
|
2021-07-26 11:37:35 +00:00
|
|
|
"""Return the length of the given DataLoader. If ``__len__`` method is not implemented, return float('inf')."""
|
2021-01-04 19:57:53 +00:00
|
|
|
|
|
|
|
if has_len(dataloader):
|
|
|
|
return len(dataloader)
|
|
|
|
|
2021-07-26 11:37:35 +00:00
|
|
|
return float("inf")
|