FlowProt / model /train.py
alibtsd's picture
Deploy FlowProt Docker Space
f34af6f verified
Raw
History Blame Contribute Delete
3.84 kB
import os
import torch
import hydra
from omegaconf import DictConfig, OmegaConf
# Pytorch lightning imports
from pytorch_lightning import LightningDataModule, LightningModule, Trainer
from pytorch_lightning.loggers.wandb import WandbLogger
from pytorch_lightning.trainer import Trainer
from pytorch_lightning.callbacks import ModelCheckpoint
from dataset.data import PdbDataModule
from models.proteinflow_wrapper import ProteinFlowModule
from utils.experiments import get_pylogger, flatten_dict
import wandb
log = get_pylogger(__name__)
torch.set_float32_matmul_precision('high')
def _configure_wandb_env(exp_cfg: DictConfig):
"""Apply optional wandb environment overrides from YAML config."""
wandb_cfg = exp_cfg.wandb
if wandb_cfg.get("offline", False):
os.environ["WANDB_MODE"] = "offline"
wandb_dir = wandb_cfg.get("dir", None)
if wandb_dir:
os.environ["WANDB_DIR"] = wandb_dir
class Experiment:
def __init__(self, *, cfg: DictConfig):
self._cfg = cfg
self._data_cfg = cfg.data
self._exp_cfg = cfg.experiment
self._datamodule: LightningDataModule = PdbDataModule(self._data_cfg)
self._model: LightningModule = ProteinFlowModule(self._cfg)
def train(self):
callbacks = []
_configure_wandb_env(self._exp_cfg)
if self._exp_cfg.debug:
log.info("Debug mode.")
logger = None
self._exp_cfg.num_devices = 1
self._data_cfg.loader.num_workers = 0
else:
logger = WandbLogger(
**self._exp_cfg.wandb,
)
# Checkpoint directory.
ckpt_dir = self._exp_cfg.checkpointer.dirpath
os.makedirs(ckpt_dir, exist_ok=True)
log.info(f"Checkpoints saved to {ckpt_dir}")
# Model checkpoints
callbacks.append(ModelCheckpoint(**self._exp_cfg.checkpointer))
# Save config
cfg_path = os.path.join(ckpt_dir, 'config.yaml')
with open(cfg_path, 'w') as f:
OmegaConf.save(config=self._cfg, f=f.name)
cfg_dict = OmegaConf.to_container(self._cfg, resolve=True)
flat_cfg = dict(flatten_dict(cfg_dict))
if isinstance(logger.experiment.config, wandb.sdk.wandb_config.Config):
logger.experiment.config.update(flat_cfg)
num_devices = int(self._exp_cfg.get("num_devices", 1))
log.info(f"Using {num_devices} device(s)")
trainer = Trainer(
**self._exp_cfg.trainer,
callbacks=callbacks,
logger=logger,
use_distributed_sampler=False,
enable_progress_bar=True,
enable_model_summary=True,
devices=num_devices,
)
trainer.fit(
model=self._model,
datamodule=self._datamodule,
ckpt_path=self._exp_cfg.warm_start
)
@hydra.main(version_base=None, config_path="./configs", config_name="base.yaml")
def main(cfg: DictConfig):
if cfg.experiment.warm_start is not None and cfg.experiment.warm_start_cfg_override:
# Loads warm start config.
warm_start_cfg_path = os.path.join(
os.path.dirname(cfg.experiment.warm_start), 'config.yaml')
warm_start_cfg = OmegaConf.load(warm_start_cfg_path)
# Warm start config may not have latest fields in the base config.
# Add these fields to the warm start config.
OmegaConf.set_struct(cfg.model, False)
OmegaConf.set_struct(warm_start_cfg.model, False)
cfg.model = OmegaConf.merge(cfg.model, warm_start_cfg.model)
OmegaConf.set_struct(cfg.model, True)
log.info(f'Loaded warm start config from {warm_start_cfg_path}')
exp = Experiment(cfg=cfg)
exp.train()
if __name__ == "__main__":
main()