Source code for dig.threedgraph.dataset.PygMD17

import os.path as osp
import numpy as np
from tqdm import tqdm
import torch
from sklearn.utils import shuffle

from torch_geometric.data import InMemoryDataset, download_url
from torch_geometric.data import Data, DataLoader


[docs]class MD17(InMemoryDataset): r""" A `Pytorch Geometric <https://pytorch-geometric.readthedocs.io/en/latest/index.html>`_ data interface for :obj:`MD17` dataset which is from `"Machine learning of accurate energy-conserving molecular force fields" <https://advances.sciencemag.org/content/3/5/e1603015.short>`_ paper. MD17 is a collection of eight molecular dynamics simulations for small organic molecules. Args: root (string): The dataset folder will be located at root/name. name (string): The name of dataset. Available dataset names are as follows: :obj:`aspirin`, :obj:`benzene_old`, :obj:`ethanol`, :obj:`malonaldehyde`, :obj:`naphthalene`, :obj:`salicylic`, :obj:`toluene`, :obj:`uracil`. (default: :obj:`benzene_old`) transform (callable, optional): A function/transform that takes in an :obj:`torch_geometric.data.Data` object and returns a transformed version. The data object will be transformed before every access. (default: :obj:`None`) pre_transform (callable, optional): A function/transform that takes in an :obj:`torch_geometric.data.Data` object and returns a transformed version. The data object will be transformed before being saved to disk. (default: :obj:`None`) pre_filter (callable, optional): A function that takes in an :obj:`torch_geometric.data.Data` object and returns a boolean value, indicating whether the data object should be included in the final dataset. (default: :obj:`None`) Example: -------- >>> dataset = MD17(name='aspirin') >>> split_idx = dataset.get_idx_split(len(dataset.data.y), train_size=1000, valid_size=1000, seed=42) >>> train_dataset, valid_dataset, test_dataset = dataset[split_idx['train']], dataset[split_idx['valid']], dataset[split_idx['test']] >>> train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True) >>> data = next(iter(train_loader)) >>> data Batch(batch=[672], force=[672, 3], pos=[672, 3], ptr=[33], y=[32], z=[672]) Where the attributes of the output data indicates: * :obj:`z`: The atom type. * :obj:`pos`: The 3D position for atoms. * :obj:`y`: The property (energy) for the graph (molecule). * :obj:`force`: The 3D force for atoms. * :obj:`batch`: The assignment vector which maps each node to its respective graph identifier and can help reconstructe single graphs """ def __init__(self, root = 'dataset/', name = 'benzene_old', transform = None, pre_transform = None, pre_filter = None): self.name = name self.folder = osp.join(root, self.name) self.url = 'http://quantum-machine.org/gdml/data/npz/' + self.name + '_dft.npz' super(MD17, self).__init__(self.folder, transform, pre_transform, pre_filter) self.data, self.slices = torch.load(self.processed_paths[0]) @property def raw_file_names(self): return self.name + '_dft.npz' @property def processed_file_names(self): return self.name + '_pyg.pt'
[docs] def download(self): download_url(self.url, self.raw_dir)
[docs] def process(self): data = np.load(osp.join(self.raw_dir, self.raw_file_names)) E = data['E'] F = data['F'] R = data['R'] z = data['z'] data_list = [] for i in tqdm(range(len(E))): R_i = torch.tensor(R[i],dtype=torch.float32) z_i = torch.tensor(z,dtype=torch.int64) E_i = torch.tensor(E[i],dtype=torch.float32) F_i = torch.tensor(F[i],dtype=torch.float32) data = Data(pos=R_i, z=z_i, y=E_i, force=F_i) data_list.append(data) if self.pre_filter is not None: data_list = [data for data in data_list if self.pre_filter(data)] if self.pre_transform is not None: data_list = [self.pre_transform(data) for data in data_list] data, slices = self.collate(data_list) print('Saving...') torch.save((data, slices), self.processed_paths[0])
def get_idx_split(self, data_size, train_size, valid_size, seed): ids = shuffle(range(data_size), random_state=seed) train_idx, val_idx, test_idx = torch.tensor(ids[:train_size]), torch.tensor(ids[train_size:train_size + valid_size]), torch.tensor(ids[train_size + valid_size:]) split_dict = {'train':train_idx, 'valid':val_idx, 'test':test_idx} return split_dict
if __name__ == '__main__': dataset = MD17(name='aspirin') print(dataset) print(dataset.data.z.shape) print(dataset.data.pos.shape) print(dataset.data.y.shape) print(dataset.data.force.shape) split_idx = dataset.get_idx_split(len(dataset.data.y), train_size=1000, valid_size=1000, seed=42) print(split_idx) print(dataset[split_idx['train']]) train_dataset, valid_dataset, test_dataset = dataset[split_idx['train']], dataset[split_idx['valid']], dataset[split_idx['test']] train_loader = DataLoader(train_dataset, batch_size=32, shuffle=True) data = next(iter(train_loader)) print(data)