Source code for pylj.util

from __future__ import division
import numpy as np
import webbrowser
from pylj import md, mc


[docs] class System: """Simulation system. This class is designed to store all of the information about the job that is being run. This includes the particles object, as will as sampling objects such as the temperature, pressure, etc. arrays. Parameters ---------- number_of_particles: int Number of particles to simulate. temperature: float Initial temperature of the particles, in Kelvin. box_length: float Length of a single dimension of the simulation square, in Angstrom. init_conf: string, optional The way that the particles are initially positioned. Should be one of: - 'square' - 'random' timestep_length: float (optional) Length for each Velocity-Verlet integration step, in seconds. cut_off: float (optional) The distance apart that the particles must be to consider there interaction to be negliable. constants: float, array_like (optional) The values of the constants for the forcefield used. mass: float (optional) The mass of the particles being simulated. forcefield: function (optional) The particular forcefield to be used to find the energy and forces. """ def __init__( self, number_of_particles, temperature, box_length, constants, forcefield, mass, init_conf="square", timestep_length=1e-14, cut_off=15 ): self.number_of_particles = number_of_particles self.init_temp = temperature self.constants = constants self.mass = mass self.forcefield = forcefield self.type_identifiers = None self.particle_list = None self.long_const = None self.types = None self.point_sizes = None self.setup_point_sizes() self.setup_type_identifiers() self.setup_types() if box_length <= 600: self.box_length = box_length * 1e-10 else: raise AttributeError( "With a box length of {} the particles are " "probably too small to be seen in the " "viewer. Try something (much) less than " "600.".format(box_length) ) if box_length >= 4: self.box_length = box_length * 1e-10 else: raise AttributeError( "With a box length of {} the cell is too " "small to really hold more than one " "particle.".format(box_length) ) self.timestep_length = timestep_length self.particles = None if init_conf == "square": self.square() elif init_conf == "random": self.random() else: raise NotImplementedError( "The initial configuration type {} is " "not recognised. Available options are: " "square or random".format(init_conf) ) if box_length > 30: self.cut_off = cut_off * 1e-10 else: self.cut_off = box_length / 2 * 1e-10 self.step = 0 self.time = 0.0 self.distances = np.zeros(self.number_of_pairs()) self.forces = np.zeros(self.number_of_pairs()) self.energies = np.zeros(self.number_of_pairs()) self.temperature_sample = np.array([]) self.pressure_sample = np.array([]) self.force_sample = np.array([]) self.msd_sample = np.array([]) self.energy_sample = np.array([]) self.initial_particles = np.array(self.particles) self.position_store = [0, 0] self.old_energy = 0 self.new_energy = 0 self.random_particle = 0
[docs] def number_of_pairs(self): """Calculates the number of pairwise interactions in the simulation. Returns ------- int: Number of pairwise interactions in the system. """ return int((self.number_of_particles - 1) * self.number_of_particles / 2)
[docs] def square(self): """Sets the initial positions of the particles on a square lattice. """ part_dt = particle_dt() self.particles = np.zeros(self.number_of_particles, dtype=part_dt) self.particles['types'] = self.types m = int(np.ceil(np.sqrt(self.number_of_particles))) d = self.box_length / m n = 0 for i in range(0, m): for j in range(0, m): if n < self.number_of_particles: self.particles[n]["xposition"] = (i + 0.5) * d self.particles[n]["yposition"] = (j + 0.5) * d n += 1
[docs] def random(self): """Sets the initial positions of the particles in a random arrangement. """ part_dt = particle_dt() self.particles = np.zeros(self.number_of_particles, dtype=part_dt) self.particles['types'] = self.types num_part = self.number_of_particles self.particles["xposition"] = np.random.uniform(0, self.box_length, num_part) self.particles["yposition"] = np.random.uniform(0, self.box_length, num_part)
[docs] def setup_types(self): """Sets the long constants and types arrays of the particles """ long_const = [] types = [] particle_list = [] for i in range(self.number_of_particles): # Get set of constants and index constants_type = self.constants[i%len(self.constants)] particle_type = f'{i%len(self.constants)}' # Append to lists long_const.append(constants_type) types.append(particle_type) particle = Particle(constants_type, i, self.mass, particle_type) particle.add(particle_list) self.particle_list = particle_list self.long_const = long_const self.types = types
[docs] def setup_type_identifiers(self): """Sets type-identifers arrays - legacy method now only used for plotting """ # Creates arrays to identify which particle is in which type number_of_types = len(self.constants) type_identifiers = np.zeros((number_of_types,self.number_of_particles)) i = 0 while i < self.number_of_particles: for k in range(number_of_types): if i < self.number_of_particles: type_identifiers[k][i] = 1 i+=1 self.type_identifiers = type_identifiers
[docs] def setup_point_sizes(self): """Sets point sizes for use in plotting """ point_sizes = [] for pair in self.constants: size = self.forcefield(pair).point_size point_sizes.append(size) self.point_sizes = point_sizes
[docs] def compute_force(self): """Maps to the compute_force function in either the comp (if Cython is installed) or the pairwise module and allows for a cleaner interface. """ part, dist, forces, energies = md.compute_force( self.particles, self.box_length, self.cut_off, self.constants, self.forcefield, self.mass, ) self.particles = part self.distances = dist self.forces = forces self.energies = energies
[docs] def compute_energy(self): """Maps to the compute_force function, as this also calculates energy """ self.compute_force()
#Jit tag here had to be removed
[docs] def integrate(self, method): """Maps the chosen integration method. Parameters ---------- method: method The integration method to be used, e.g. md.velocity_verlet. """ self.particles, self.distances, self.forces, self.energies = method( self.particles, self.timestep_length, self.box_length, self.cut_off, self.constants, self.forcefield, self.mass )
[docs] def md_sample(self): """Maps to the md.sample function. """ self = md.sample(self.particles, self.box_length, self.initial_particles, self)
[docs] def heat_bath(self, bath_temperature): """Maps to the heat_bath function in either the comp (if Cython is installed) or the pairwise modules. Parameters ---------- target_temperature: float The target temperature for the simulation. """ self.particles = md.heat_bath( self.particles, self.temperature_sample, bath_temperature )
[docs] def mc_sample(self): """Maps to the mc.sample function. Parameters ---------- energy: float Energy to add to the sample """ mc.sample(self.old_energy, self)
[docs] def select_random_particle(self): """Maps to the mc.select_random_particle function. """ self.random_particle, self.position_store = mc.select_random_particle( self.particles )
[docs] def new_random_position(self): """Maps to the mc.get_new_particle function. """ self.particles = mc.get_new_particle( self.particles, self.random_particle, self.box_length )
[docs] def accept(self): """Maps to the mc.accept function. """ self.old_energy = mc.accept(self.new_energy)
[docs] def reject(self): """Maps to the mc.reject function. """ self.particles = mc.reject( self.position_store, self.particles, self.random_particle )
[docs] class Particle: def __init__(self, constants, index, mass, particle_type ): self.constants = constants self.index = index self.mass = mass self.type = particle_type
[docs] def add(self, particles): particles.append(self) return particles
def __cite__(): # pragma: no cover """This function will launch the website for the JOSE publication on pylj.""" webbrowser.open("http://jose.theoj.org/papers/58daa1a1a564dc8e0f99ffcdae20eb1d") def __version__(): # pragma: no cover """This will print the number of the pylj version currently in use.""" major = 1 minor = 4 micro = 1 print("pylj-{:d}.{:d}.{:d}".format(major, minor, micro))
[docs] def particle_dt(): """Builds the data type for the particles, this consists of: - xposition and yposition - xvelocity and yvelocity - xacceleration and yacceleration - xprevious_position and yprevious_position - xforce and yforce - energy - types """ return np.dtype( [ ("xposition", np.float64), ("yposition", np.float64), ("xvelocity", np.float64), ("yvelocity", np.float64), ("xacceleration", np.float64), ("yacceleration", np.float64), ("xprevious_position", np.float64), ("yprevious_position", np.float64), ("energy", np.float64), ("xpbccount", int), ("ypbccount", int), ("types", list) ] )