2020-08-20 02:03:22 +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-05-12 11:53:20 +00:00
|
|
|
"""Helper functions to help with reproducibility of models. """
|
|
|
|
|
|
|
|
import os
|
2020-08-07 22:33:51 +00:00
|
|
|
import random
|
|
|
|
from typing import Optional
|
2020-05-12 11:53:20 +00:00
|
|
|
|
|
|
|
import numpy as np
|
|
|
|
import torch
|
|
|
|
|
|
|
|
from pytorch_lightning import _logger as log
|
|
|
|
|
|
|
|
|
|
|
|
def seed_everything(seed: Optional[int] = None) -> int:
|
2020-09-20 23:42:58 +00:00
|
|
|
"""
|
|
|
|
Function that sets seed for pseudo-random number generators in:
|
|
|
|
pytorch, numpy, python.random and sets PYTHONHASHSEED environment variable.
|
|
|
|
In addition, sets the env variable `PL_GLOBAL_SEED` which will be passed to
|
|
|
|
spawned subprocesses (e.g. ddp_spawn backend).
|
|
|
|
|
|
|
|
Args:
|
|
|
|
seed: the integer value seed for global random state in Lightning.
|
|
|
|
If `None`, will read seed from `PL_GLOBAL_SEED` env variable
|
|
|
|
or select it randomly.
|
2020-05-12 11:53:20 +00:00
|
|
|
"""
|
|
|
|
max_seed_value = np.iinfo(np.uint32).max
|
|
|
|
min_seed_value = np.iinfo(np.uint32).min
|
|
|
|
|
|
|
|
try:
|
2020-06-12 15:23:18 +00:00
|
|
|
if seed is None:
|
2020-09-20 23:42:58 +00:00
|
|
|
seed = os.environ.get("PL_GLOBAL_SEED", _select_seed_randomly(min_seed_value, max_seed_value))
|
|
|
|
seed = int(seed)
|
2020-05-12 11:53:20 +00:00
|
|
|
except (TypeError, ValueError):
|
|
|
|
seed = _select_seed_randomly(min_seed_value, max_seed_value)
|
|
|
|
|
|
|
|
if (seed > max_seed_value) or (seed < min_seed_value):
|
|
|
|
log.warning(
|
|
|
|
f"{seed} is not in bounds, \
|
|
|
|
numpy accepts from {min_seed_value} to {max_seed_value}"
|
|
|
|
)
|
|
|
|
seed = _select_seed_randomly(min_seed_value, max_seed_value)
|
|
|
|
|
|
|
|
os.environ["PYTHONHASHSEED"] = str(seed)
|
2020-09-20 23:42:58 +00:00
|
|
|
os.environ["PL_GLOBAL_SEED"] = str(seed)
|
2020-05-12 11:53:20 +00:00
|
|
|
random.seed(seed)
|
|
|
|
np.random.seed(seed)
|
|
|
|
torch.manual_seed(seed)
|
2020-08-24 13:22:05 +00:00
|
|
|
torch.cuda.manual_seed_all(seed)
|
2020-05-12 11:53:20 +00:00
|
|
|
return seed
|
|
|
|
|
|
|
|
|
|
|
|
def _select_seed_randomly(min_seed_value: int = 0, max_seed_value: int = 255) -> int:
|
|
|
|
seed = random.randint(min_seed_value, max_seed_value)
|
|
|
|
log.warning(f"No correct seed found, seed set to {seed}")
|
|
|
|
return seed
|