diff --git a/pyproject.toml b/pyproject.toml index 0ce5ca4..29f06b2 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -92,4 +92,16 @@ where = [ "src", ] +[project.entry-points.'nomad.plugin'] + +general_schema = "nomad_material_processing:schema" +solution_schema = "nomad_material_processing.solution:schema" +vd_schema = "nomad_material_processing.vapor_deposition:schema" +cvd_schema = "nomad_material_processing.vapor_deposition.cvd:schema" +pvd_schema = "nomad_material_processing.vapor_deposition.pvd:schema" +mbe_schema = "nomad_material_processing.vapor_deposition.pvd:mbe_schema" +pld_schema = "nomad_material_processing.vapor_deposition.pvd:pld_schema" +sputtering_schema = "nomad_material_processing.vapor_deposition.pvd:sputtering_schema" +thermal_schema = "nomad_material_processing.vapor_deposition.pvd:thermal_schema" + [tool.setuptools_scm] diff --git a/src/nomad_material_processing/__init__.py b/src/nomad_material_processing/__init__.py index ebd1576..73d6902 100644 --- a/src/nomad_material_processing/__init__.py +++ b/src/nomad_material_processing/__init__.py @@ -15,866 +15,18 @@ # See the License for the specific language governing permissions and # limitations under the License. # -from typing import ( - TYPE_CHECKING, -) -import numpy as np -from nomad.metainfo import ( - Package, - Quantity, - Section, - SubSection, - Datetime, - MEnum, -) -from nomad.datamodel.metainfo.basesections import ( - ElementalComposition, - SynthesisMethod, - CompositeSystem, - CompositeSystemReference, - SystemComponent, -) -from nomad.datamodel.data import ( - ArchiveSection, -) -from nomad.datamodel.metainfo.annotations import ( - ELNAnnotation, - ELNComponentEnum, -) -from nomad.datamodel.metainfo.workflow import ( - Link, -) - -if TYPE_CHECKING: - from nomad.datamodel.datamodel import ( - EntryArchive, - ) - from structlog.stdlib import ( - BoundLogger, - ) - - -m_package = Package(name='Material Processing') - - -class Geometry(ArchiveSection): - """ - Geometrical shape attributes of a system. - Sections derived from `Geometry` represent concrete geometrical shapes. - """ - - m_def = Section() - volume = Quantity( - type=float, - description='The measure of the amount of space occupied in 3D space.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - ), - unit='meter ** 3', - ) - - -class Parallelepiped(Geometry): - """ - Six-faced polyhedron with each pair of opposite faces parallel and equal in size, - characterized by rectangular sides and parallelogram faces. - """ - - m_def = Section() - height = Quantity( - type=float, - description='The z dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Height (z)', - ), - unit='meter', - ) - width = Quantity( - type=float, - description='The x dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Width (x)', - ), - unit='meter', - ) - length = Quantity( - type=float, - description='The y dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Length (y)', - ), - unit='meter', - ) - alpha = Quantity( - type=float, - description='The angle between y and z sides.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Alpha (∡ y-x-z)', - ), - unit='degree', - ) - beta = Quantity( - type=float, - description='The angle between x and z sides.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Beta (∡ x-y-z)', - ), - unit='degree', - ) - gamma = Quantity( - type=float, - description='The angle between x and y sides.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Gamma (∡ x-z-y)', - ), - unit='degree', - ) - surface_area = Quantity( - type=float, - description=""" - The product of length and width, representing the total exposed area of the - primary surface. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter ** 2', - label='Surface Area (x*y)', - ), - unit='meter ** 2', - ) - - -class SquareCuboid(Parallelepiped): - """ - A cuboid with all sides equal in length. - """ - m_def = Section() - height = Quantity( - type=float, - description='The z dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Height (z)', - ), - unit='meter', - ) - side = Quantity( - type=float, - description='The x and y dimensions of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Side (x = y)', - ), - unit='meter', - ) - surface_area = Quantity( - type=float, - description=""" - The product of length and width, representing the total exposed area of the - primary surface. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter ** 2', - label='Surface Area (x*y)', - ), - unit='meter ** 2', - ) - - -class RectangleCuboid(Parallelepiped): - """ - A rectangular cuboid is a specific type of parallelepiped - where all angles between adjacent faces are right angles, - and all faces are rectangles. - """ - m_def = Section() - height = Quantity( - type=float, - description='The z dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Height (z)', - ), - unit='meter', - ) - width = Quantity( - type=float, - description='The x dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Width (x)', - ), - unit='meter', - ) - length = Quantity( - type=float, - description='The y dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - label='Length (y)', - ), - unit='meter', - ) - surface_area = Quantity( - type=float, - description=""" - The product of length and width, representing the total exposed area of the - primary surface. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter ** 2', - label='Surface Area (x*y)', - ), - unit='meter ** 2', - ) - - -class TruncatedCone(Geometry): - """ - A cone with the top cut off parallel to the cone bottom. - """ - - m_def = Section() - height = Quantity( - type=float, - description='The z dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='nanometer', - ), - label='Height (z)', - unit='meter', - ) - lower_cap_radius = Quantity( - type=float, - description='Radius of the lower cap.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter', - ), - unit='meter', - ) - upper_cap_radius = Quantity( - type=float, - description='Radius of the upper cap.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter', - ), - unit='meter', - ) - lower_cap_surface_area = Quantity( - type=float, - description='Area of the lower cap.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter ** 2', - ), - unit='meter ** 2', - ) - upper_cap_surface_area = Quantity( - type=float, - description='Area of the upper cap.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter ** 2', - ), - unit='meter ** 2', - ) - lateral_surface_area = Quantity( - type=float, - description='Area of the lateral surface.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter ** 2', - ), - unit='meter ** 2', - ) - - -class Cylinder(Geometry): - """ - A cylinder, i.e. a prism with a circular base. - """ - - m_def = Section() - height = Quantity( - type=float, - description='The z dimension of the parallelepiped.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='nanometer', - ), - label='Height (z)', - unit='meter', - ) - radius = Quantity( - type=float, - description='Radius of the cylinder.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter', - ), - unit='meter', - ) - lower_cap_surface_area = Quantity( - type=float, - description='Area of the lower cap.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter ** 2', - ), - unit='meter ** 2', - ) - cap_surface_area = Quantity( - type=float, - description='Area of the cap.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter ** 2', - ), - unit='meter ** 2', - ) - lateral_surface_area = Quantity( - type=float, - description='Area of the lateral surface.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='millimeter ** 2', - ), - unit='meter ** 2', - ) - - -class CylinderSector(Cylinder): - central_angle = Quantity( - type=float, - description="""The angle that defines the portion of the cylinder. - This angle is taken at the center of the base circle - and extends to the arc that defines the cylindrical sector.""", - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - ), - unit='degree', - ) - - -class IrregularParallelSurfaces(Geometry): - """ - A shape that does not fit into any of the other geometry classes. - """ - - m_def = Section() - height = Quantity( - type=float, - description='The z dimension of the irregular shape.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='nanometer', - ), - label='Height (z)', - unit='meter', - ) - - -class Miscut(ArchiveSection): - """ - The miscut in a crystalline substrate refers to - the intentional deviation from a specific crystallographic orientation, - commonly expressed as the angular displacement of a crystal plane. - """ - - angle = Quantity( - type=float, - description=""" - The angular displacement from the crystallographic orientation of the substrate. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='deg', - label='Miscut Angle', - ), - unit='deg', - ) - angle_deviation = Quantity( - type=float, - description='The ± deviation in the angular displacement.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='deg', - label='± Miscut Angle Deviation', - ), - unit='deg', - ) - orientation = Quantity( - type=str, - description='The direction of the miscut in Miller index, [hkl].', - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - label='Miscut Orientation [hkl]', - ), - ) - - -class Dopant(ElementalComposition): - """ - A dopant element in a crystalline structure - is a foreign atom intentionally introduced into the crystal lattice. - """ - - doping_level = Quantity( - type=float, - description='The chemical doping level.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='1 / cm ** 3', - ), - unit='1 / m ** 3', - ) - doping_deviation = Quantity( - type=float, - description='The ± deviation in the doping level.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='1 / cm ** 3', - ), - unit='1 / m ** 3', - ) - - -class CrystalProperties(ArchiveSection): - """ - Characteristics arising from the ordered arrangement of atoms in a crystalline structure. - These properties are defined by factors such as crystal symmetry, lattice parameters, - and the specific arrangement of atoms within the crystal lattice. - """ - - -class SubstrateCrystalProperties(CrystalProperties): - """ - Crystallographic parameters such as orientation, miscut, and surface structure. - """ - bravais_lattices = Quantity( - type=MEnum( - 'Triclinic', - 'Monoclinic Simple', - 'Monoclinic Base Centered', - 'Orthorhombic Simple', - 'Orthorhombic Base Centered', - 'Orthorhombic Body Centered', - 'Orthorhombic Face Centered', - 'Tetragonal Simple', - 'Tetragonal Body Centered', - 'Cubic Simple', - 'Cubic Body Centered', - 'Cubic Face Centered', - 'Trigonal', - 'Hexagonal', - ), - description='The crystal system of the substrate.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - orientation = Quantity( - type=str, - description=""" - Alignment of crystal lattice with respect to a vector normal to the surface - specified using Miller indices. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - label='Substrate Orientation (hkl)', - ), - ) - miscut = SubSection( - section_def=Miscut, - description=""" - Section describing any miscut of the substrate with respect to the substrate - orientation. - """, - repeats=True, - ) - +from nomad.config.models.plugins import SchemaPackageEntryPoint -class ElectronicProperties(ArchiveSection): - """ - The electronic properties of a material. - """ - m_def = Section() +class GeneralSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.general import m_package - conductivity_type = Quantity( - type=MEnum( - 'P-type', - 'N-type', - ), - description='The type of semiconductor, N-type being electrons the majority carriers and P-type being holes the majority carriers.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - carrier_density = Quantity( - type=np.dtype(float), - unit='1 / cm**3', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - ), - description='Concentration of free charge carriers, electrons in the conduction band and holes in the valence band.', - ) - electrical_resistivity = Quantity( - type=float, - links=['http://fairmat-nfdi.eu/taxonomy/ElectricalResistivity'], - description='Resistance of the charges to move in the presence of an electric current.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='ohm cm', - ), - unit='ohm m', - ) + return m_package -class Substrate(CompositeSystem): - """ - A thin free standing sheet of material. Not to be confused with the substrate role - during a deposition, which can be a `Substrate` with `ThinFilm`(s) on it. - """ - - m_def = Section() - - supplier = Quantity( - type=str, - description='The supplier of the current substrate specimen.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - supplier_id = Quantity( - type=str, - description='An ID string that is unique from the supplier.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - label='Supplier ID', - ), - ) - lab_id = Quantity( - type=str, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - label='Substrate ID', - ), - ) - image = Quantity( - type=str, - description='A photograph or image of the substrate.', - a_browser={'adaptor': 'RawFileAdaptor'}, - a_eln=ELNAnnotation( - component=ELNComponentEnum.FileEditQuantity, - ), - ) - information_sheet = Quantity( - type=str, - description='Pdf files containing certificate and other documentation.', - a_browser={'adaptor': 'RawFileAdaptor'}, - a_eln=ELNAnnotation( - component='FileEditQuantity', - ), - ) - - -class CrystallineSubstrate(Substrate): - """ - The substrate defined in this class is composed of periodic arrangement of atoms - and shows typical features of a crystal structure. - """ - - m_def = Section() - - - geometry = SubSection( - section_def=Geometry, - description='Section containing the geometry of the substrate.', - ) - crystal_properties = SubSection( - section_def=SubstrateCrystalProperties, - description='Section containing the crystal properties of the substrate.', - ) - electronic_properties = SubSection( - section_def=ElectronicProperties, - description='Section containing the electronic properties of the substrate.', - ) - dopants = SubSection( - section_def=Dopant, - repeats=True, - description=""" - Repeating section containing information on any dopants in the substrate. - """, - ) - - -class ThinFilm(CompositeSystem): - """ - A thin film of material which exists as part of a stack. - """ - - m_def = Section() - - geometry = SubSection( - section_def=Geometry, - description='Section containing the geometry of the thin film.', - ) - - -class ThinFilmReference(CompositeSystemReference): - """ - Class autogenerated from yaml schema. - """ - - lab_id = Quantity( - type=str, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - label='Thin Film ID', - ), - ) - reference = Quantity( - type=ThinFilm, - a_eln=ELNAnnotation( - component=ELNComponentEnum.ReferenceEditQuantity, - label='Thin Film', - ), - ) - - -class SubstrateReference(CompositeSystemReference): - """ - A section for describing a system component and its role in a composite system. - """ - - lab_id = Quantity( - type=str, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - label='Substrate ID', - ), - ) - reference = Quantity( - type=Substrate, - a_eln=ELNAnnotation( - component=ELNComponentEnum.ReferenceEditQuantity, - label='Substrate', - ), - ) - - -class ThinFilmStack(CompositeSystem): - """ - A stack of `ThinFilm`(s). Typically deposited on a `Substrate`. - """ - - m_def = Section( - a_eln=ELNAnnotation( - hide=[ - 'components', - ], - ), - ) - layers = SubSection( - description=""" - An ordered list (starting at the substrate) of the thin films making up the - thin film stacks. - """, - section_def=ThinFilmReference, - repeats=True, - ) - substrate = SubSection( - description=""" - The substrate which the thin film layers of the thin film stack are deposited - on. - """, - section_def=SubstrateReference, - ) - - def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - """ - The normalizer for the `ThinFilmStack` class. - - Args: - archive (EntryArchive): The archive containing the section that is being - normalized. - logger (BoundLogger): A structlog logger. - """ - self.components = [] - if self.layers: - self.components = [ - SystemComponent(system=layer.reference) - for layer in self.layers - if layer.reference - ] - if self.substrate.reference: - self.components.append(SystemComponent(system=self.substrate.reference)) - super().normalize(archive, logger) - - -class ThinFilmStackReference(CompositeSystemReference): - """ - Class autogenerated from yaml schema. - """ - - m_def = Section() - lab_id = Quantity( - type=str, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - reference = Quantity( - type=ThinFilmStack, - a_eln=ELNAnnotation( - component=ELNComponentEnum.ReferenceEditQuantity, - ), - ) - - -class SampleDeposition(SynthesisMethod): - """ - The process of the settling of particles (atoms or molecules) from a solution, - suspension or vapour onto a pre-existing surface, resulting in the growth of a - new phase. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] - - Synonyms: - - deposition - """ - - m_def = Section( - links=['http://purl.obolibrary.org/obo/CHMO_0001310'], - ) - - def is_serial(self) -> bool: - """ - Method for determining if the steps are serial. Can be overwritten by sub class. - Default behavior is to return True if all steps start after the previous one. - - Returns: - bool: Whether or not the steps are serial. - """ - start_times = [] - durations = [] - for step in self.steps: - if step.start_time is None or step.duration is None: - return False - start_times.append(step.start_time.timestamp()) - durations.append(step.duration.to('s').magnitude) - start_times = np.array(start_times) - durations = np.array(durations) - end_times = start_times + durations - diffs = start_times[1:] - end_times[:-1] - if np.any(diffs < 0): - return False - return True - - def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - """ - The normalizer for the `SampleDeposition` class. - - Args: - archive (EntryArchive): The archive containing the section that is being - normalized. - logger (BoundLogger): A structlog logger. - """ - super().normalize(archive, logger) - if self.is_serial(): - tasks = [] - previous = None - for step in self.steps: - task = step.to_task() - task.outputs.append(Link(name=step.name, section=step)) - if previous is not None: - task.inputs.append(Link(name=previous.name, section=previous)) - tasks.append(task) - previous = step - archive.workflow2.tasks = tasks - - -class TimeSeries(ArchiveSection): - """ - A time series of data during a process step. - This is an abstract class and should not be used directly. - Instead, it should be derived and the the units of the `value` and `set_value` should - be specified. - - For example, a derived class could be `Temperature` with `value` in Kelvin: - ```python - class Temperature(TimeSeries): - value = TimeSeries.value.m_copy() - value.unit = "kelvin" - set_value = TimeSeries.set_value.m_copy() - set_value.unit = "kelvin" - set_value.a_eln.defaultDisplayUnit = "celsius" - ``` - """ - - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - set_value = Quantity( - type=float, - description='The set value(s) (i.e. the intended values) set.', - shape=['*'], - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - ), - ) - set_time = Quantity( - type=float, - unit='s', - description=""" - The process time when each of the set values were set. - If this is empty and only one set value is present, it is assumed that the value - was set at the start of the process step. - If two set values are present, it is assumed that a linear ramp between the two - values was set. - """, - shape=['*'], - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='s', - label='Set time', - ), - ) - value = Quantity( - type=float, - description='The observed value as a function of time.', - shape=['*'], - ) - time = Quantity( - type=float, - unit='s', - description='The process time when each of the values were recorded.', - shape=['*'], - ) - - -m_package.__init_metainfo__() +schema = GeneralSchemaPackageEntryPoint( + name='General Schema', + description="""Schema package containing basic classes used + in the nomad_material_processing plugin.""", +) diff --git a/src/nomad_material_processing/crystal_growth.py b/src/nomad_material_processing/crystal_growth.py index bd14ed3..e980ecd 100644 --- a/src/nomad_material_processing/crystal_growth.py +++ b/src/nomad_material_processing/crystal_growth.py @@ -22,7 +22,7 @@ Package, Section, ) -from nomad_material_processing import ( +from nomad_material_processing.general import ( SampleDeposition, ) @@ -38,48 +38,48 @@ class CrystalGrowth(SampleDeposition): - ''' + """ Any synthesis method used to grow crystals. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] - ''' + """ + m_def = Section( - links=[ - "http://purl.obolibrary.org/obo/CHMO_0002224" - ],) + links=['http://purl.obolibrary.org/obo/CHMO_0002224'], + ) def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - ''' + """ The normalizer for the `CrystalGrowth` class. Args: archive (EntryArchive): The archive containing the section that is being normalized. logger (BoundLogger): A structlog logger. - ''' + """ super(CrystalGrowth, self).normalize(archive, logger) class CzochralskiProcess(CrystalGrowth): - ''' + """ A method of producing large single crystals (of semiconductors or metals) by inserting a small seed crystal into a crucible filled with similar molten material, then slowly pulling the seed up from the melt while rotating it. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] - ''' + """ + m_def = Section( - links=[ - "http://purl.obolibrary.org/obo/CHMO_0002158" - ],) + links=['http://purl.obolibrary.org/obo/CHMO_0002158'], + ) def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - ''' + """ The normalizer for the `CzochralskiProcess` class. Args: archive (EntryArchive): The archive containing the section that is being normalized. logger (BoundLogger): A structlog logger. - ''' + """ super(CzochralskiProcess, self).normalize(archive, logger) diff --git a/src/nomad_material_processing/epitaxy.py b/src/nomad_material_processing/epitaxy.py index 5831e5e..1e25115 100644 --- a/src/nomad_material_processing/epitaxy.py +++ b/src/nomad_material_processing/epitaxy.py @@ -22,7 +22,7 @@ Package, Section, ) -from nomad_material_processing import ( +from nomad_material_processing.general import ( SampleDeposition, ) @@ -38,30 +38,30 @@ class Epitaxy(SampleDeposition): - ''' + """ A synthesis method which consists of depositing a monocrystalline film (from liquid or gaseous precursors) on a monocrystalline substrate. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] - ''' + """ + m_def = Section( - links=[ - "http://purl.obolibrary.org/obo/CHMO_0001336" - ],) + links=['http://purl.obolibrary.org/obo/CHMO_0001336'], + ) def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - ''' + """ The normalizer for the `Epitaxy` class. Args: archive (EntryArchive): The archive containing the section that is being normalized. logger (BoundLogger): A structlog logger. - ''' + """ super(Epitaxy, self).normalize(archive, logger) class MolecularBeamEpitaxy(Epitaxy): - ''' + """ A synthesis method which consists of depositing a monocrystalline film (from a molecular beam) on a monocrystalline substrate under high vacuum (<10^{-8} Pa). Molecular beam epitaxy is very slow, with a deposition rate of <1000 nm per hour. @@ -70,26 +70,26 @@ class MolecularBeamEpitaxy(Epitaxy): Synonyms: - MBE - molecular-beam epitaxy - ''' + """ + m_def = Section( - links=[ - "http://purl.obolibrary.org/obo/CHMO_0001341" - ],) + links=['http://purl.obolibrary.org/obo/CHMO_0001341'], + ) def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - ''' + """ The normalizer for the `MolecularBeamEpitaxy` class. Args: archive (EntryArchive): The archive containing the section that is being normalized. logger (BoundLogger): A structlog logger. - ''' + """ super(MolecularBeamEpitaxy, self).normalize(archive, logger) class VaporPhaseEpitaxy(Epitaxy): - ''' + """ A synthesis method which consists of depositing a monocrystalline film (from vapour-phase precursors) on a monocrystalline substrate. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] @@ -100,26 +100,26 @@ class VaporPhaseEpitaxy(Epitaxy): - vapor phase epitaxy - VPE - vapour phase epitaxy - ''' + """ + m_def = Section( - links=[ - "http://purl.obolibrary.org/obo/CHMO_0001346" - ],) + links=['http://purl.obolibrary.org/obo/CHMO_0001346'], + ) def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - ''' + """ The normalizer for the `VaporPhaseEpitaxy` class. Args: archive (EntryArchive): The archive containing the section that is being normalized. logger (BoundLogger): A structlog logger. - ''' + """ super(VaporPhaseEpitaxy, self).normalize(archive, logger) class MetalOrganicVaporPhaseEpitaxy(VaporPhaseEpitaxy): - ''' + """ A synthesis method which consists of depositing a monocrystalline film, from organometallic vapour-phase precursors, on a monocrystalline substrate. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] @@ -135,21 +135,21 @@ class MetalOrganicVaporPhaseEpitaxy(VaporPhaseEpitaxy): - metal organic vapor phase epitaxy - metal-organic vapour-phase epitaxy - organometallic vapour phase epitaxy - ''' + """ + m_def = Section( - links=[ - "http://purl.obolibrary.org/obo/CHMO_0001348" - ],) + links=['http://purl.obolibrary.org/obo/CHMO_0001348'], + ) def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - ''' + """ The normalizer for the `MetalOrganicVaporPhaseEpitaxy` class. Args: archive (EntryArchive): The archive containing the section that is being normalized. logger (BoundLogger): A structlog logger. - ''' + """ super(MetalOrganicVaporPhaseEpitaxy, self).normalize(archive, logger) diff --git a/src/nomad_material_processing/general.py b/src/nomad_material_processing/general.py new file mode 100644 index 0000000..b81d808 --- /dev/null +++ b/src/nomad_material_processing/general.py @@ -0,0 +1,1108 @@ +from typing import TYPE_CHECKING + +if TYPE_CHECKING: + from nomad.datamodel.datamodel import ( + EntryArchive, + ) + from structlog.stdlib import ( + BoundLogger, + ) + +from nomad.datamodel.metainfo.annotations import ELNAnnotation, ELNComponentEnum + +import numpy as np +from nomad.datamodel.data import EntryData, ArchiveSection + +from nomad.metainfo import ( + SchemaPackage, + Quantity, + SubSection, + Section, + MEnum, +) + +from nomad.datamodel.metainfo.basesections import ( + ElementalComposition, + CompositeSystem, + SynthesisMethod, + ProcessStep, + Process, + SystemComponent, + CompositeSystemReference, +) +from nomad.datamodel.metainfo.annotations import ( + ELNAnnotation, + ELNComponentEnum, +) +from nomad.datamodel.metainfo.workflow import ( + Link, +) +from nomad.metainfo import ( + SectionProxy, + Reference, +) + +from nomad.config import config + +m_package = SchemaPackage(name='Material Processing') + +configuration = config.get_plugin_entry_point('nomad_material_processing:schema') + + +class Geometry(ArchiveSection): + """ + Geometrical shape attributes of a system. + Sections derived from `Geometry` represent concrete geometrical shapes. + """ + + m_def = Section() + volume = Quantity( + type=float, + description='The measure of the amount of space occupied in 3D space.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + ), + unit='meter ** 3', + ) + + +class Parallelepiped(Geometry): + """ + Six-faced polyhedron with each pair of opposite faces parallel and equal in size, + characterized by rectangular sides and parallelogram faces. + """ + + m_def = Section() + height = Quantity( + type=float, + description='The z dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Height (z)', + ), + unit='meter', + ) + width = Quantity( + type=float, + description='The x dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Width (x)', + ), + unit='meter', + ) + length = Quantity( + type=float, + description='The y dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Length (y)', + ), + unit='meter', + ) + alpha = Quantity( + type=float, + description='The angle between y and z sides.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Alpha (∡ y-x-z)', + ), + unit='degree', + ) + beta = Quantity( + type=float, + description='The angle between x and z sides.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Beta (∡ x-y-z)', + ), + unit='degree', + ) + gamma = Quantity( + type=float, + description='The angle between x and y sides.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Gamma (∡ x-z-y)', + ), + unit='degree', + ) + surface_area = Quantity( + type=float, + description=""" + The product of length and width, representing the total exposed area of the + primary surface. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter ** 2', + label='Surface Area (x*y)', + ), + unit='meter ** 2', + ) + + +class SquareCuboid(Parallelepiped): + """ + A cuboid with all sides equal in length. + """ + + m_def = Section() + height = Quantity( + type=float, + description='The z dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Height (z)', + ), + unit='meter', + ) + side = Quantity( + type=float, + description='The x and y dimensions of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Side (x = y)', + ), + unit='meter', + ) + surface_area = Quantity( + type=float, + description=""" + The product of length and width, representing the total exposed area of the + primary surface. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter ** 2', + label='Surface Area (x*y)', + ), + unit='meter ** 2', + ) + + +class RectangleCuboid(Parallelepiped): + """ + A rectangular cuboid is a specific type of parallelepiped + where all angles between adjacent faces are right angles, + and all faces are rectangles. + """ + + m_def = Section() + height = Quantity( + type=float, + description='The z dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Height (z)', + ), + unit='meter', + ) + width = Quantity( + type=float, + description='The x dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Width (x)', + ), + unit='meter', + ) + length = Quantity( + type=float, + description='The y dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + label='Length (y)', + ), + unit='meter', + ) + surface_area = Quantity( + type=float, + description=""" + The product of length and width, representing the total exposed area of the + primary surface. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter ** 2', + label='Surface Area (x*y)', + ), + unit='meter ** 2', + ) + + +class TruncatedCone(Geometry): + """ + A cone with the top cut off parallel to the cone bottom. + """ + + m_def = Section() + height = Quantity( + type=float, + description='The z dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='nanometer', + ), + label='Height (z)', + unit='meter', + ) + lower_cap_radius = Quantity( + type=float, + description='Radius of the lower cap.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter', + ), + unit='meter', + ) + upper_cap_radius = Quantity( + type=float, + description='Radius of the upper cap.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter', + ), + unit='meter', + ) + lower_cap_surface_area = Quantity( + type=float, + description='Area of the lower cap.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter ** 2', + ), + unit='meter ** 2', + ) + upper_cap_surface_area = Quantity( + type=float, + description='Area of the upper cap.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter ** 2', + ), + unit='meter ** 2', + ) + lateral_surface_area = Quantity( + type=float, + description='Area of the lateral surface.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter ** 2', + ), + unit='meter ** 2', + ) + + +class Cylinder(Geometry): + """ + A cylinder, i.e. a prism with a circular base. + """ + + m_def = Section() + height = Quantity( + type=float, + description='The z dimension of the parallelepiped.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='nanometer', + ), + label='Height (z)', + unit='meter', + ) + radius = Quantity( + type=float, + description='Radius of the cylinder.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter', + ), + unit='meter', + ) + lower_cap_surface_area = Quantity( + type=float, + description='Area of the lower cap.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter ** 2', + ), + unit='meter ** 2', + ) + cap_surface_area = Quantity( + type=float, + description='Area of the cap.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter ** 2', + ), + unit='meter ** 2', + ) + lateral_surface_area = Quantity( + type=float, + description='Area of the lateral surface.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='millimeter ** 2', + ), + unit='meter ** 2', + ) + + +class CylinderSector(Cylinder): + central_angle = Quantity( + type=float, + description="""The angle that defines the portion of the cylinder. + This angle is taken at the center of the base circle + and extends to the arc that defines the cylindrical sector.""", + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + ), + unit='degree', + ) + + +class IrregularParallelSurfaces(Geometry): + """ + A shape that does not fit into any of the other geometry classes. + """ + + m_def = Section() + height = Quantity( + type=float, + description='The z dimension of the irregular shape.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='nanometer', + ), + label='Height (z)', + unit='meter', + ) + + +class Miscut(ArchiveSection): + """ + The miscut in a crystalline substrate refers to + the intentional deviation from a specific crystallographic orientation, + commonly expressed as the angular displacement of a crystal plane. + """ + + angle = Quantity( + type=float, + description=""" + The angular displacement from the crystallographic orientation of the substrate. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='deg', + label='Miscut Angle', + ), + unit='deg', + ) + angle_deviation = Quantity( + type=float, + description='The ± deviation in the angular displacement.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='deg', + label='± Miscut Angle Deviation', + ), + unit='deg', + ) + orientation = Quantity( + type=str, + description='The direction of the miscut in Miller index, [hkl].', + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + label='Miscut Orientation [hkl]', + ), + ) + + +class Dopant(ElementalComposition): + """ + A dopant element in a crystalline structure + is a foreign atom intentionally introduced into the crystal lattice. + """ + + doping_level = Quantity( + type=float, + description='The chemical doping level.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='1 / cm ** 3', + ), + unit='1 / m ** 3', + ) + doping_deviation = Quantity( + type=float, + description='The ± deviation in the doping level.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='1 / cm ** 3', + ), + unit='1 / m ** 3', + ) + + +class CrystalProperties(ArchiveSection): + """ + Characteristics arising from the ordered arrangement of atoms in a crystalline structure. + These properties are defined by factors such as crystal symmetry, lattice parameters, + and the specific arrangement of atoms within the crystal lattice. + """ + + +class SubstrateCrystalProperties(CrystalProperties): + """ + Crystallographic parameters such as orientation, miscut, and surface structure. + """ + + bravais_lattices = Quantity( + type=MEnum( + 'Triclinic', + 'Monoclinic Simple', + 'Monoclinic Base Centered', + 'Orthorhombic Simple', + 'Orthorhombic Base Centered', + 'Orthorhombic Body Centered', + 'Orthorhombic Face Centered', + 'Tetragonal Simple', + 'Tetragonal Body Centered', + 'Cubic Simple', + 'Cubic Body Centered', + 'Cubic Face Centered', + 'Trigonal', + 'Hexagonal', + ), + description='The crystal system of the substrate.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + orientation = Quantity( + type=str, + description=""" + Alignment of crystal lattice with respect to a vector normal to the surface + specified using Miller indices. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + label='Substrate Orientation (hkl)', + ), + ) + miscut = SubSection( + section_def=Miscut, + description=""" + Section describing any miscut of the substrate with respect to the substrate + orientation. + """, + repeats=True, + ) + + +class ElectronicProperties(ArchiveSection): + """ + The electronic properties of a material. + """ + + m_def = Section() + + conductivity_type = Quantity( + type=MEnum( + 'P-type', + 'N-type', + ), + description='The type of semiconductor, N-type being electrons the majority carriers and P-type being holes the majority carriers.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + carrier_density = Quantity( + type=np.dtype(float), + unit='1 / cm**3', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + ), + description='Concentration of free charge carriers, electrons in the conduction band and holes in the valence band.', + ) + electrical_resistivity = Quantity( + type=float, + links=['http://fairmat-nfdi.eu/taxonomy/ElectricalResistivity'], + description='Resistance of the charges to move in the presence of an electric current.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='ohm cm', + ), + unit='ohm m', + ) + + +class Substrate(CompositeSystem): + """ + A thin free standing sheet of material. Not to be confused with the substrate role + during a deposition, which can be a `Substrate` with `ThinFilm`(s) on it. + """ + + m_def = Section() + + supplier = Quantity( + type=str, + description='The supplier of the current substrate specimen.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + supplier_id = Quantity( + type=str, + description='An ID string that is unique from the supplier.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + label='Supplier ID', + ), + ) + lab_id = Quantity( + type=str, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + label='Substrate ID', + ), + ) + image = Quantity( + type=str, + description='A photograph or image of the substrate.', + a_browser={'adaptor': 'RawFileAdaptor'}, + a_eln=ELNAnnotation( + component=ELNComponentEnum.FileEditQuantity, + ), + ) + information_sheet = Quantity( + type=str, + description='Pdf files containing certificate and other documentation.', + a_browser={'adaptor': 'RawFileAdaptor'}, + a_eln=ELNAnnotation( + component='FileEditQuantity', + ), + ) + + +class CrystallineSubstrate(Substrate): + """ + The substrate defined in this class is composed of periodic arrangement of atoms + and shows typical features of a crystal structure. + """ + + m_def = Section() + + geometry = SubSection( + section_def=Geometry, + description='Section containing the geometry of the substrate.', + ) + crystal_properties = SubSection( + section_def=SubstrateCrystalProperties, + description='Section containing the crystal properties of the substrate.', + ) + electronic_properties = SubSection( + section_def=ElectronicProperties, + description='Section containing the electronic properties of the substrate.', + ) + dopants = SubSection( + section_def=Dopant, + repeats=True, + description=""" + Repeating section containing information on any dopants in the substrate. + """, + ) + + +class ThinFilm(CompositeSystem): + """ + A thin film of material which exists as part of a stack. + """ + + m_def = Section() + + geometry = SubSection( + section_def=Geometry, + description='Section containing the geometry of the thin film.', + ) + + +class ThinFilmReference(CompositeSystemReference): + """ + Class autogenerated from yaml schema. + """ + + lab_id = Quantity( + type=str, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + label='Thin Film ID', + ), + ) + reference = Quantity( + type=ThinFilm, + a_eln=ELNAnnotation( + component=ELNComponentEnum.ReferenceEditQuantity, + label='Thin Film', + ), + ) + + +class SubstrateReference(CompositeSystemReference): + """ + A section for describing a system component and its role in a composite system. + """ + + lab_id = Quantity( + type=str, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + label='Substrate ID', + ), + ) + reference = Quantity( + type=Substrate, + a_eln=ELNAnnotation( + component=ELNComponentEnum.ReferenceEditQuantity, + label='Substrate', + ), + ) + + +class ThinFilmStack(CompositeSystem): + """ + A stack of `ThinFilm`(s). Typically deposited on a `Substrate`. + """ + + m_def = Section( + a_eln=ELNAnnotation( + hide=[ + 'components', + ], + ), + ) + layers = SubSection( + description=""" + An ordered list (starting at the substrate) of the thin films making up the + thin film stacks. + """, + section_def=ThinFilmReference, + repeats=True, + ) + substrate = SubSection( + description=""" + The substrate which the thin film layers of the thin film stack are deposited + on. + """, + section_def=SubstrateReference, + ) + + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: + """ + The normalizer for the `ThinFilmStack` class. + + Args: + archive (EntryArchive): The archive containing the section that is being + normalized. + logger (BoundLogger): A structlog logger. + """ + self.components = [] + if self.layers: + self.components = [ + SystemComponent(system=layer.reference) + for layer in self.layers + if layer.reference + ] + if self.substrate.reference: + self.components.append(SystemComponent(system=self.substrate.reference)) + super().normalize(archive, logger) + + +class ThinFilmStackReference(CompositeSystemReference): + """ + Class autogenerated from yaml schema. + """ + + m_def = Section() + lab_id = Quantity( + type=str, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + reference = Quantity( + type=ThinFilmStack, + a_eln=ELNAnnotation( + component=ELNComponentEnum.ReferenceEditQuantity, + ), + ) + + +class SampleDeposition(SynthesisMethod): + """ + The process of the settling of particles (atoms or molecules) from a solution, + suspension or vapour onto a pre-existing surface, resulting in the growth of a + new phase. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] + + Synonyms: + - deposition + """ + + m_def = Section( + links=['http://purl.obolibrary.org/obo/CHMO_0001310'], + ) + + def is_serial(self) -> bool: + """ + Method for determining if the steps are serial. Can be overwritten by sub class. + Default behavior is to return True if all steps start after the previous one. + + Returns: + bool: Whether or not the steps are serial. + """ + start_times = [] + durations = [] + for step in self.steps: + if step.start_time is None or step.duration is None: + return False + start_times.append(step.start_time.timestamp()) + durations.append(step.duration.to('s').magnitude) + start_times = np.array(start_times) + durations = np.array(durations) + end_times = start_times + durations + diffs = start_times[1:] - end_times[:-1] + if np.any(diffs < 0): + return False + return True + + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: + """ + The normalizer for the `SampleDeposition` class. + + Args: + archive (EntryArchive): The archive containing the section that is being + normalized. + logger (BoundLogger): A structlog logger. + """ + super().normalize(archive, logger) + if self.is_serial(): + tasks = [] + previous = None + for step in self.steps: + task = step.to_task() + task.outputs.append(Link(name=step.name, section=step)) + if previous is not None: + task.inputs.append(Link(name=previous.name, section=previous)) + tasks.append(task) + previous = step + archive.workflow2.tasks = tasks + + +class TimeSeries(ArchiveSection): + """ + A time series of data during a process step. + This is an abstract class and should not be used directly. + Instead, it should be derived and the the units of the `value` and `set_value` should + be specified. + + For example, a derived class could be `Temperature` with `value` in Kelvin: + ```python + class Temperature(TimeSeries): + value = TimeSeries.value.m_copy() + value.unit = "kelvin" + set_value = TimeSeries.set_value.m_copy() + set_value.unit = "kelvin" + set_value.a_eln.defaultDisplayUnit = "celsius" + ``` + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + set_value = Quantity( + type=float, + description='The set value(s) (i.e. the intended values) set.', + shape=['*'], + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + ), + ) + set_time = Quantity( + type=float, + unit='s', + description=""" + The process time when each of the set values were set. + If this is empty and only one set value is present, it is assumed that the value + was set at the start of the process step. + If two set values are present, it is assumed that a linear ramp between the two + values was set. + """, + shape=['*'], + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='s', + label='Set time', + ), + ) + value = Quantity( + type=float, + description='The observed value as a function of time.', + shape=['*'], + ) + time = Quantity( + type=float, + unit='s', + description='The process time when each of the values were recorded.', + shape=['*'], + ) + + +class Recipe(ArchiveSection): + """ + A Recipe for a material processing experiment. + + This class will be subclassed for each process that needs a recipe. + + The subclass will inherit Recipe and a specific Process class. + + The only difference between the Recipe and the actual Process is that + the datetime and the input samples Entities are hidden in the Recipe. + """ + + pass + + +class EtchingStep(ProcessStep): + """ + A step of etching process. + """ + + m_def = Section() + duration = Quantity( + type=np.float64, + description='The elapsed time since the annealing process started.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' + ), + unit='second', + ) + temperature = Quantity( + type=np.float64, + description='The temperature of the etching process.', + a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'celsius'}, + unit='celsius', + ) + agitation = Quantity( + type=MEnum( + 'Magnetic Stirring', + 'Sonication', + ), + description='The agitation method used during the etching process.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + etching_reagents = SubSection(section_def=CompositeSystem, repeats=True) + + +class Etching(Process, EntryData): + """ + Selectively remove material from a surface using chemical or physical processes + to create specific patterns or structures. + """ + + m_def = Section( + a_eln=None, + links=['http://purl.obolibrary.org/obo/CHMO_0001558'], + ) + recipe = Quantity( + type=Reference(SectionProxy('EtchingRecipe')), + description=""" The recipe used for the process. If a recipe is found, + all the data is copied from the Recipe within the Process. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.ReferenceEditQuantity, + ), + ) + steps = SubSection( + description=""" + The steps of the etching process. + """, + section_def=EtchingStep, + repeats=True, + ) + + +class EtchingRecipe(Etching, Recipe, EntryData): + """ + A Recipe for an etching process. + """ + + m_def = Section( + a_eln={'hide': ['datetime', 'samples']}, + ) + + +class AnnealingStep(ProcessStep): + """ + A step of annealing process. + """ + + m_def = Section() + duration = Quantity( + type=np.float64, + description='The elapsed time since the annealing process started.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' + ), + unit='second', + ) + temperature = Quantity( + type=np.float64, + description='The temperature of the etching process.', + a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'celsius'}, + unit='celsius', + ) + + +class Annealing(Process, EntryData): + """ + Heat treatment process used to alter the material's properties, + such as reducing defects, improving crystallinity, or relieving internal stresses. + """ + + m_def = Section( + links=['http://purl.obolibrary.org/obo/CHMO_0001465'], + ) + recipe = Quantity( + type=Reference(SectionProxy('AnnealingRecipe')), + description=""" The recipe used for the process. If a recipe is found, + all the data is copied from the Recipe within the Process. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.ReferenceEditQuantity, + ), + ) + duration = Quantity( + type=np.float64, + description='The elapsed time since the annealing process started.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' + ), + unit='second', + ) + steps = SubSection( + description=""" + The steps of the annealing process. + """, + section_def=AnnealingStep, + repeats=True, + ) + + +class AnnealingRecipe(Annealing, Recipe, EntryData): + """ + A Recipe for an annealing process. + """ + + m_def = Section( + a_eln={'hide': ['datetime', 'samples']}, + ) + + +class CleaningStep(ProcessStep): + """ + A step of cleaning process. + """ + + m_def = Section() + duration = Quantity( + type=np.float64, + description='The elapsed time since the cleaning process started.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' + ), + unit='second', + ) + temperature = Quantity( + type=np.float64, + description='The temperature of the cleaning process.', + a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'celsius'}, + unit='celsius', + ) + agitation = Quantity( + type=MEnum( + 'Magnetic Stirring', + 'Sonication', + ), + description='The agitation method used during the cleaning process.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + cleaning_reagents = SubSection( + section_def=CompositeSystemReference, + ) + + +class Cleaning(Process, EntryData): + """ + Surface cleaning in thin film material science involves removing contaminants + and residues from a substrate's surface to ensure proper adhesion + and uniformity of the thin film deposition. + """ + + m_def = Section( + a_eln={'hide': ['steps']}, + ) + recipe = Quantity( + type=Reference(SectionProxy('CleaningRecipe')), + description=""" The recipe used for the process. If a recipe is found, + all the data is copied from the Recipe within the Process. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.ReferenceEditQuantity, + ), + ) + duration = Quantity( + type=np.float64, + description='The elapsed time since the annealing process started.', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' + ), + unit='second', + ) + steps = SubSection( + description=""" + The steps of the cleaning process. + """, + section_def=CleaningStep, + repeats=True, + ) + + +class CleaningRecipe(Cleaning, Recipe, EntryData): + """ + A Recipe for an cleaning process. + """ + + m_def = Section( + a_eln={'hide': ['datetime', 'samples']}, + ) + + +m_package.__init_metainfo__() diff --git a/src/nomad_material_processing/general/__init__.py b/src/nomad_material_processing/general/__init__.py deleted file mode 100644 index e69de29..0000000 diff --git a/src/nomad_material_processing/general/schema.py b/src/nomad_material_processing/general/schema.py deleted file mode 100644 index f9bb2c0..0000000 --- a/src/nomad_material_processing/general/schema.py +++ /dev/null @@ -1,271 +0,0 @@ -from typing import TYPE_CHECKING - -if TYPE_CHECKING: - from nomad.datamodel.datamodel import ( - EntryArchive, - ) - from structlog.stdlib import ( - BoundLogger, - ) - -from nomad.datamodel.metainfo.annotations import ELNAnnotation, ELNComponentEnum - -import numpy as np -from nomad.datamodel.data import EntryData, ArchiveSection - -from nomad.metainfo import ( - Quantity, - SubSection, - Section, - MEnum, -) - -from nomad.datamodel.metainfo.basesections import ( - CompositeSystem, - ProcessStep, - Process, - CompositeSystemReference, -) -from nomad.datamodel.metainfo.annotations import ( - ELNAnnotation, - ELNComponentEnum, -) - -from nomad.metainfo import ( - SectionProxy, - Reference, -) - - -class Recipe(ArchiveSection): - """ - A Recipe for a material processing experiment. - - This class will be subclassed for each process that needs a recipe. - - The subclass will inherit Recipe and a specific Process class. - - The only difference between the Recipe and the actual Process is that - the datetime and the input samples Entities are hidden in the Recipe. - """ - - pass - - -class EtchingStep(ProcessStep): - """ - A step of etching process. - """ - - m_def = Section() - duration = Quantity( - type=np.float64, - description='The elapsed time since the annealing process started.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' - ), - unit='second', - ) - temperature = Quantity( - type=np.float64, - description='The temperature of the etching process.', - a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'celsius'}, - unit='celsius', - ) - agitation = Quantity( - type=MEnum( - 'Magnetic Stirring', - 'Sonication', - ), - description='The agitation method used during the etching process.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - etching_reagents = SubSection(section_def=CompositeSystem, repeats=True) - - -class Etching(Process, EntryData): - """ - Selectively remove material from a surface using chemical or physical processes - to create specific patterns or structures. - """ - - m_def = Section( - a_eln=None, - links=['http://purl.obolibrary.org/obo/CHMO_0001558'], - ) - recipe = Quantity( - type=Reference(SectionProxy('EtchingRecipe')), - description=""" The recipe used for the process. If a recipe is found, - all the data is copied from the Recipe within the Process. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.ReferenceEditQuantity, - ), - ) - steps = SubSection( - description=""" - The steps of the etching process. - """, - section_def=EtchingStep, - repeats=True, - ) - - -class EtchingRecipe(Etching, Recipe, EntryData): - """ - A Recipe for an etching process. - """ - - m_def = Section( - a_eln={'hide': ['datetime', 'samples']}, - ) - - -class AnnealingStep(ProcessStep): - """ - A step of annealing process. - """ - - m_def = Section() - duration = Quantity( - type=np.float64, - description='The elapsed time since the annealing process started.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' - ), - unit='second', - ) - temperature = Quantity( - type=np.float64, - description='The temperature of the etching process.', - a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'celsius'}, - unit='celsius', - ) - - -class Annealing(Process, EntryData): - """ - Heat treatment process used to alter the material's properties, - such as reducing defects, improving crystallinity, or relieving internal stresses. - """ - - m_def = Section( - links=['http://purl.obolibrary.org/obo/CHMO_0001465'], - ) - recipe = Quantity( - type=Reference(SectionProxy('AnnealingRecipe')), - description=""" The recipe used for the process. If a recipe is found, - all the data is copied from the Recipe within the Process. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.ReferenceEditQuantity, - ), - ) - duration = Quantity( - type=np.float64, - description='The elapsed time since the annealing process started.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' - ), - unit='second', - ) - steps = SubSection( - description=""" - The steps of the annealing process. - """, - section_def=AnnealingStep, - repeats=True, - ) - - -class AnnealingRecipe(Annealing, Recipe, EntryData): - """ - A Recipe for an annealing process. - """ - - m_def = Section( - a_eln={'hide': ['datetime', 'samples']}, - ) - - -class CleaningStep(ProcessStep): - """ - A step of cleaning process. - """ - - m_def = Section() - duration = Quantity( - type=np.float64, - description='The elapsed time since the cleaning process started.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' - ), - unit='second', - ) - temperature = Quantity( - type=np.float64, - description='The temperature of the cleaning process.', - a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'celsius'}, - unit='celsius', - ) - agitation = Quantity( - type=MEnum( - 'Magnetic Stirring', - 'Sonication', - ), - description='The agitation method used during the cleaning process.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - cleaning_reagents = SubSection( - section_def=CompositeSystemReference, - ) - - -class Cleaning(Process, EntryData): - """ - Surface cleaning in thin film material science involves removing contaminants - and residues from a substrate's surface to ensure proper adhesion - and uniformity of the thin film deposition. - """ - - m_def = Section( - a_eln={'hide': ['steps']}, - ) - recipe = Quantity( - type=Reference(SectionProxy('CleaningRecipe')), - description=""" The recipe used for the process. If a recipe is found, - all the data is copied from the Recipe within the Process. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.ReferenceEditQuantity, - ), - ) - duration = Quantity( - type=np.float64, - description='The elapsed time since the annealing process started.', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, defaultDisplayUnit='minute' - ), - unit='second', - ) - steps = SubSection( - description=""" - The steps of the cleaning process. - """, - section_def=CleaningStep, - repeats=True, - ) - - -class CleaningRecipe(Cleaning, Recipe, EntryData): - """ - A Recipe for an cleaning process. - """ - - m_def = Section( - a_eln={'hide': ['datetime', 'samples']}, - ) diff --git a/src/nomad_material_processing/nomad_plugin.yaml b/src/nomad_material_processing/nomad_plugin.yaml deleted file mode 100644 index 61f6fce..0000000 --- a/src/nomad_material_processing/nomad_plugin.yaml +++ /dev/null @@ -1,3 +0,0 @@ -description: A plugin for NOMAD containing base sections for material processing. -name: NOMAD Material Processing -plugin_type: schema diff --git a/src/nomad_material_processing/solution/__init__.py b/src/nomad_material_processing/solution/__init__.py index 341318c..56d9976 100644 --- a/src/nomad_material_processing/solution/__init__.py +++ b/src/nomad_material_processing/solution/__init__.py @@ -1 +1,14 @@ -from nomad_material_processing.solution.schema import * +from nomad.config.models.plugins import SchemaPackageEntryPoint + + +class SolutionSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.solution.schema import m_package + + return m_package + + +schema = SolutionSchemaPackageEntryPoint( + name='Solution Schema', + description='Schema package containing classes for solution preparation.', +) diff --git a/src/nomad_material_processing/solution/nomad_plugin.yaml b/src/nomad_material_processing/solution/nomad_plugin.yaml deleted file mode 100644 index a7e2517..0000000 --- a/src/nomad_material_processing/solution/nomad_plugin.yaml +++ /dev/null @@ -1,3 +0,0 @@ -description: A plugin for NOMAD containing base sections for solutions. -name: Solutions -plugin_type: schema diff --git a/src/nomad_material_processing/solution/schema.py b/src/nomad_material_processing/solution/schema.py index de60920..11f9654 100644 --- a/src/nomad_material_processing/solution/schema.py +++ b/src/nomad_material_processing/solution/schema.py @@ -22,6 +22,7 @@ PubChemPureSubstanceSection, ) from nomad.metainfo import ( + SchemaPackage, Datetime, MEnum, Quantity, @@ -37,6 +38,14 @@ from structlog.stdlib import BoundLogger from nomad.datamodel import EntryArchive +from nomad.config import config + +m_package = SchemaPackage() + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.solution:schema' +) + class MolarConcentration(ArchiveSection): """ diff --git a/src/nomad_material_processing/vapor_deposition/__init__.py b/src/nomad_material_processing/vapor_deposition/__init__.py index db8a0aa..bccd926 100644 --- a/src/nomad_material_processing/vapor_deposition/__init__.py +++ b/src/nomad_material_processing/vapor_deposition/__init__.py @@ -15,715 +15,18 @@ # See the License for the specific language governing permissions and # limitations under the License. # -from typing import ( - TYPE_CHECKING, -) -from nomad.metainfo import ( - Package, - Section, - SubSection, - Quantity, - MEnum, -) -from nomad.datamodel.data import ( - ArchiveSection, -) -from nomad.datamodel.metainfo.annotations import ( - ELNAnnotation, - ELNComponentEnum, -) -from nomad.datamodel.metainfo.basesections import ( - Entity, - ActivityStep, - PureSubstanceSection, - Component, - CompositeSystemReference, - PubChemPureSubstanceSection, -) -from nomad.datamodel.metainfo.plot import ( - PlotSection, -) -from nomad.datamodel.metainfo.workflow import ( - Link, - Task, -) -from nomad_material_processing import ( - SampleDeposition, - ThinFilmStackReference, - ThinFilmReference, - TimeSeries, - Geometry, -) - -if TYPE_CHECKING: - from nomad.datamodel.datamodel import ( - EntryArchive, - ) - from structlog.stdlib import ( - BoundLogger, - ) - -m_package = Package(name='Vapor Deposition') - - -class InsertReduction(Entity): - """ - The reduction that sometimes is used to lodge the substrate in the substrate holder position.. - """ - - name = Quantity( - type=str, - description=""" - A short and descriptive name for this insert reduction. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - insert_id = Quantity( - type=str, - description=""" - The ID of the insert that is placed in this position to accomodate the substrate. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - material = SubSection(section_def=PubChemPureSubstanceSection, repeats=True) - inner_geometry = SubSection( - section_def=Geometry, - ) - outer_geometry = SubSection( - section_def=Geometry, - ) - - -class SubstrateHolderPosition(ArchiveSection): - """ - One casing position of the substrate holder. - """ - - name = Quantity( - type=str, - description=""" - A short name for this position. This name is used as label of the position. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - x_position = Quantity( - type=float, - unit='meter', - description=""" - The x coordinate of the substrate holder position relative to the center of the holder. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - ), - ) - y_position = Quantity( - type=float, - unit='meter', - description=""" - The y coordinate of the substrate holder position relative to the center of the holder. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - ), - ) - slot_geometry = SubSection( - section_def=Geometry, - ) - insert_reduction = SubSection( - section_def=InsertReduction, - description='Optional description of insert if used.', - ) - - -class SubstrateHolder(Entity): - """ - The holder for the substrate. - """ - - name = Quantity( - type=str, - description=""" - A short and descriptive name for this position. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - lab_id = Quantity( - type=str, - description=""" - The lab ID of the substrate holder. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.StringEditQuantity, - ), - ) - material = SubSection(section_def=PubChemPureSubstanceSection, repeats=True) - thickness = Quantity( - type=float, - unit='meter', - description=""" - The thickness of the holder to the back of the substrate. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='micrometer', - ), - ) - outer_diameter = Quantity( - type=float, - unit='meter', - description=""" - The outer diameter of the substrate holder. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit='millimeter', - ), - ) - number_of_positions = Quantity( - type=int, - description=""" - The number of positions on the holder. - """, - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - ), - ) - image = Quantity( - type=str, - description="""An image of the substrate holder.""", - a_browser={'adaptor': 'RawFileAdaptor'}, - a_eln=ELNAnnotation( - component=ELNComponentEnum.FileEditQuantity, - ), - ) - positions = SubSection( - section_def=SubstrateHolderPosition, - repeats=True, - ) - - -class FilledSubstrateHolderPosition(SubstrateHolderPosition): - """ - One casing position of the filled substrate holder. - """ - - substrate = SubSection( - section_def=CompositeSystemReference, - description=""" - The substrate that is placed in this position. - """, - ) - - -class FilledSubstrateHolder(SubstrateHolder): - """ - A substrate holder that is filled with substrate(s). - """ - - substrate_holder = SubSection( - section_def=SubstrateHolder, - ) - positions = SubSection( - section_def=FilledSubstrateHolderPosition, - repeats=True, - ) - - -class MolarFlowRate(TimeSeries): - """ - Molar flow rate is the amount of a substance which passes per unit of time. - """ - - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - measurement_type = Quantity( - type=MEnum( - 'Assumed', - 'Mass Flow Controller', - ), - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - value = TimeSeries.value.m_copy() - value.unit = 'mol/second' - set_value = TimeSeries.set_value.m_copy() - set_value.unit = 'mol/second' - - -class EvaporationSource(ArchiveSection): - pass - +from nomad.config.models.plugins import SchemaPackageEntryPoint -class VaporDepositionSource(ArchiveSection): - name = Quantity( - type=str, - description=""" - A short and descriptive name for this source. - """, - ) - material = SubSection( - section_def=Component, - description=""" - The source of the material that is being evaporated. - Example: A sputtering target, a powder in a crucible, etc. - """, - repeats=True, - ) - vapor_source = SubSection( - section_def=EvaporationSource, - description=""" - Example: A heater, a filament, a laser, a bubbler, etc. - """, - ) - vapor_molar_flow_rate = SubSection( - section_def=MolarFlowRate, - description=""" - The rate of the material being evaporated (mol/time). - """, - ) +class GeneralVdSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.general import m_package -class GrowthRate(TimeSeries): - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - measurement_type = Quantity( - type=MEnum( - 'Assumed', - 'RHEED', - 'Reflectance', - ), - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - # value = TimeSeries.value.m_copy() - # value.unit = 'meter/second' - # set_value = TimeSeries.set_value.m_copy() - # set_value.unit = 'meter/second' - # set_value.a_eln.defaultDisplayUnit = 'nm/second' - value = Quantity( - type=float, - unit='meter/second', - shape=['*'], - ) - set_value = Quantity( - type=float, - description='The set value(s) (i.e. the intended values) set.', - shape=['*'], - unit='meter/second', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - ), - ) + return m_package -class Temperature(TimeSeries): - """ - Generic Temperature monitoring - """ - - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - measurement_type = Quantity( - type=MEnum( - 'Heater thermocouple', - 'Thermocouple', - 'Pyrometer', - 'Assumed', - ), - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - # value = TimeSeries.value.m_copy() - # value.unit = 'kelvin' - # set_value = TimeSeries.set_value.m_copy() - # set_value.unit = 'kelvin' - # set_value.a_eln.defaultDisplayUnit = 'celsius' - value = Quantity( - type=float, - unit='kelvin', - shape=['*'], - ) - set_value = Quantity( - type=float, - description='The set value(s) (i.e. the intended values) set.', - shape=['*'], - unit='kelvin', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - ), - ) - - -class SampleParameters(PlotSection, ArchiveSection): - m_def = Section( - a_plotly_graph_object={ - 'label': 'Measured Temperatures', - 'index': 1, - 'dragmode': 'pan', - 'data': { - 'type': 'scattergl', - 'line': {'width': 2}, - 'marker': {'size': 2}, - 'mode': 'lines+markers', - 'name': 'Temperature', - 'x': '#temperature/time', - 'y': '#temperature/value', - }, - 'layout': { - 'title': {'text': 'Measured Temperature'}, - 'xaxis': { - 'showticklabels': True, - 'fixedrange': True, - 'ticks': '', - 'title': {'text': 'Process time [s]'}, - 'showline': True, - 'linewidth': 1, - 'linecolor': 'black', - 'mirror': True, - }, - 'yaxis': { - 'showticklabels': True, - 'fixedrange': True, - 'ticks': '', - 'title': {'text': 'Temperature [°C]'}, - 'showline': True, - 'linewidth': 1, - 'linecolor': 'black', - 'mirror': True, - }, - 'showlegend': False, - }, - 'config': { - 'displayModeBar': False, - 'scrollZoom': False, - 'responsive': False, - 'displaylogo': False, - 'dragmode': False, - }, - }, - ) - growth_rate = SubSection( - section_def=GrowthRate, - description=""" - The growth rate of the thin film (length/time). - Measured by in-situ RHEED or Reflection or assumed. - """, - ) - substrate_temperature = SubSection( - section_def=Temperature, - ) - layer = SubSection( - description=""" - The thin film that is being created during this step. - """, - section_def=ThinFilmReference, - ) - substrate = SubSection( - description=""" - The thin film stack that is being evaporated on. - """, - section_def=ThinFilmStackReference, - ) - - -class Pressure(TimeSeries): - """ - The pressure during the deposition process. - """ - - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - # value = TimeSeries.value.m_copy() - # value.unit = 'pascal' - # set_value = TimeSeries.set_value.m_copy() - # set_value.unit = 'pascal' - # set_value.a_eln.defaultDisplayUnit = 'mbar' - value = Quantity( - type=float, - unit='pascal', - shape=['*'], - ) - time = Quantity( - type=float, - unit='second', - shape=['*'], - ) - set_value = Quantity( - type=float, - unit='pascal', - shape=['*'], - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - ), - ) - set_time = Quantity( - type=float, - unit='second', - shape=['*'], - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set time', - ), - ) - - -class VolumetricFlowRate(TimeSeries): - """ - The volumetric flow rate of a gas at standard conditions, i.e. the equivalent rate - at a temperature of 0 °C (273.15 K) and a pressure of 1 atm (101325 Pa). - """ - - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - measurement_type = Quantity( - type=MEnum( - 'Mass Flow Controller', - 'Flow Meter', - 'Other', - ), - ) - # value = TimeSeries.value.m_copy() - # value.unit = 'meter ** 3 / second' - # set_value = TimeSeries.set_value.m_copy() - # set_value.unit = 'meter ** 3 / second' - # set_value.a_eln.defaultDisplayUnit = 'centimeter ** 3 / minute' - value = Quantity( - type=float, - unit='meter ** 3 / second', - shape=['*'], - ) - set_value = Quantity( - type=float, - description='The set value(s) (i.e. the intended values) set.', - shape=['*'], - unit='meter ** 3 / second', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - defaultDisplayUnit='centimeter ** 3 / minute', - ), - ) - - -class GasFlow(ArchiveSection): - """ - Section describing the flow of a gas. - """ - - m_def = Section( - a_plot=dict( - # x=['flow_rate/time', 'flow_rate/set_time'], - # y=['flow_rate/value', 'flow_rate/set_value'], - x='flow_rate/time', - y='flow_rate/value', - ), - ) - gas = SubSection( - section_def=PureSubstanceSection, - ) - flow_rate = SubSection( - section_def=VolumetricFlowRate, - ) - - -class SubstrateHeater(ArchiveSection): - pass - - -class ChamberEnvironment(ArchiveSection): - m_def = Section( - a_plot=dict( - x='pressure/time', - y='pressure/value', - ), - ) - gas_flow = SubSection( - section_def=GasFlow, - repeats=True, - ) - pressure = SubSection( - section_def=Pressure, - ) - heater = SubSection( - section_def=SubstrateHeater, - ) - - -class VaporDepositionStep(ActivityStep): - """ - A step of any vapor deposition process. - """ - - m_def = Section() - creates_new_thin_film = Quantity( - type=bool, - description=""" - Whether or not this step creates a new thin film. - """, - default=False, - a_eln=ELNAnnotation( - component='BoolEditQuantity', - ), - ) - duration = Quantity( - type=float, - unit='second', - ) - sources = SubSection( - section_def=VaporDepositionSource, - repeats=True, - ) - sample_parameters = SubSection( - section_def=SampleParameters, - repeats=True, - ) - environment = SubSection( - section_def=ChamberEnvironment, - ) - - def to_task(self) -> Task: - """ - Returns the task description of this activity step. - - Returns: - Task: The activity step as a workflow task. - """ - inputs = [] - for source in self.sources: - if source.material is not None and hasattr(source.material, 'system'): - inputs.append( - Link( - name=getattr(source.material, 'name', None), - section=getattr(source.material, 'system', None), - ) - ) - elif source.material is not None and hasattr( - source.material, 'pure_substance' - ): - inputs.append( - Link( - name=getattr(source.material, 'substance_name', None), - section=getattr(source.material, 'pure_substance', None), - ) - ) - outputs = [ - Link( - name=parameters.layer.name, - section=parameters.layer.reference, - ) - for parameters in self.sample_parameters - if parameters.layer is not None and parameters.layer.reference is not None - ] - return Task(name=self.name, inputs=inputs, outputs=outputs) - - def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - """ - The normalizer for the `VaporDepositionStep` class. - - Args: - archive (EntryArchive): The archive containing the section that is being - normalized. - logger (BoundLogger): A structlog logger. - """ - super(VaporDepositionStep, self).normalize(archive, logger) - - -class VaporDeposition(SampleDeposition): - """ - VaporDeposition is a general class that encompasses both Physical Vapor Deposition - (PVD) and Chemical Vapor Deposition (CVD). - It involves the deposition of material from a vapor phase to a solid thin film or - coating onto a substrate. - - material sources: - Both PVD and CVD involve a source material that is transformed into a vapor phase. - In PVD, the source material is physically evaporated or sputtered from a solid - target. - In CVD, gaseous precursors undergo chemical reactions to produce a solid material - on the substrate. - - substrate: - The substrate is the material onto which the thin film is deposited. - - environment: - The process typically takes place in a controlled environment. - The deposition is usually affected by the pressure in the chamber. - For some processes additional background gasses are also added. - """ - - m_def = Section( - links=[ - 'http://purl.obolibrary.org/obo/CHMO_0001314', - 'http://purl.obolibrary.org/obo/CHMO_0001356', - ], - a_plot=[ - dict( - x='steps/:/environment/pressure/time', - y='steps/:/environment/pressure/value', - ), - ], - ) - steps = SubSection( - description=""" - The steps of the deposition process. - """, - section_def=VaporDepositionStep, - repeats=True, - ) - - def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - """ - The normalizer for the `VaporDeposition` class. - - Args: - archive (EntryArchive): The archive containing the section that is being - normalized. - logger (BoundLogger): A structlog logger. - """ - super(VaporDeposition, self).normalize(archive, logger) - - -m_package.__init_metainfo__() +schema = GeneralVdSchemaPackageEntryPoint( + name='General Vapor Deposition Schema', + description="""Schema package containing basic classes used + in the vapor_deposition submodule.""", +) diff --git a/src/nomad_material_processing/vapor_deposition/cvd/__init__.py b/src/nomad_material_processing/vapor_deposition/cvd/__init__.py index b81b6ca..4110bec 100644 --- a/src/nomad_material_processing/vapor_deposition/cvd/__init__.py +++ b/src/nomad_material_processing/vapor_deposition/cvd/__init__.py @@ -1,458 +1,14 @@ -# -# Copyright The NOMAD Authors. -# -# This file is part of NOMAD. See https://nomad-lab.eu for further info. -# -# 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. -# -from typing import ( - TYPE_CHECKING, -) -from nomad.metainfo import ( - Package, - Section, - SubSection, - Quantity, - MEnum, -) - -from nomad.datamodel.metainfo.annotations import ( - ELNAnnotation, - ELNComponentEnum, -) - -from nomad.datamodel.data import ( - ArchiveSection, -) -from nomad.datamodel.metainfo.plot import PlotSection, PlotlyFigure - -from nomad.datamodel.metainfo.basesections import ( - PubChemPureSubstanceSection, - PureSubstanceComponent, -) -from nomad_material_processing import ( - TimeSeries, -) -from nomad_material_processing.vapor_deposition import ( - EvaporationSource, - VaporDepositionSource, - MolarFlowRate, - VolumetricFlowRate, - Pressure, - Temperature, - GasFlow, -) - - -if TYPE_CHECKING: - from nomad.datamodel.datamodel import ( - EntryArchive, - ) - from structlog.stdlib import ( - BoundLogger, - ) - -m_package = Package(name="Chemical Vapor Deposition") - - -class ComponentConcentration(PureSubstanceComponent): - """ - The concentration of a component in a mixed material. - """ - - theoretical_concentration = Quantity( - type=float, - description='The concentration planned for the component.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='mol / liter', - minValue=0, - ), - unit='mol / liter', - ) - effective_concentration = Quantity( - type=float, - description='The concentration calculated from the component moles and total volume.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='mol / liter', - minValue=0, - ), - unit='mol / liter', - ) - -class PushPurgeGasFlow(GasFlow): - """ - Section describing the flow of a gas. - """ - - m_def = Section( - a_plot=[dict( - # x=['flow_rate/time', 'flow_rate/set_time'], - # y=['flow_rate/value', 'flow_rate/set_value'], - x='flow_rate/time', - y='flow_rate/value', - ), - dict( - x='purge_flow_rate/time', - y='purge_flow_rate/value', - ), - ], - ) - flow_rate = SubSection( - section_def=VolumetricFlowRate, - label="push_flow_rate", - ) - purge_flow_rate = SubSection( - section_def=VolumetricFlowRate, - ) - - -class Rotation(TimeSeries): - """ - Rotation - """ - - m_def = Section(label_quantity="set_value") - - set_value = Quantity( - type=float, - description="The value scalar set for this parameter.", - a_eln=ELNAnnotation( - component="NumberEditQuantity", - defaultDisplayUnit="rpm", - ), - unit="rpm", - ) - value = Quantity( - type=float, - description="The rotation of the sample holder, or susceptor.", - # a_eln=ELNAnnotation( - # component="NumberEditQuantity", - # defaultDisplayUnit="rpm", - # ), - unit="rpm", - ) - - -class PartialVaporPressure(Pressure): - """ - The Partial Vapor Pressure (or Equilibrium Vapor Pressure), p, is the pressure exerted - by a vapor in thermodynamic equilibrium with its condensed phases (solid or liquid) - at a given temperature in a closed system. - - It can be approximately calculated by the semiempirical Antoine equation. - It is a relation between the vapor pressure and temperature of pure substances. - log10(p) = A - [B / (T + C)] - https://en.wikipedia.org/wiki/Vapor_pressure - The August-Antoine equation is a simplified version of the Antoine equation, - sometimes used to calculate Partial Vapor Pressure. - This assumes a temperature-independent heat of vaporization, i.e., C = 0. - https://en.wikipedia.org/wiki/Antoine_equation - """ - - set_value = Quantity( - type=float, - description="The value scalar set for this parameter.", - a_eln={"component": "NumberEditQuantity", "defaultDisplayUnit": "mbar"}, - unit="pascal", - ) - value = Quantity( - type=float, - unit="pascal", - shape=["*"], - ) - time = Quantity( - type=float, - unit="second", - shape=["*"], - ) - - -class BubblerMolarFlowRate(MolarFlowRate): - """ - Molar flow rate is the amount of a substance which passes per unit of time. - - The article cited below explains the equation used in MOVPE to calculate the molar flow rate. - - F_r = F_c*P_r / (P_0 - P_r) - - where: - - F_r is the molar flow rate, - F_c is the carrier gas flow rate, - P_r is the partial vapor pressure of the precursor, - P_0 is the total pressure exiting the bubbler. - - Reference: - Journal of Vacuum Science & Technology A 8, 800 (1990); doi: 10.1116/1.576921 - - """ - - value = MolarFlowRate.value.m_copy() - set_value = MolarFlowRate.set_value.m_copy() - set_value.a_eln.defaultDisplayUnit = "mol/minute" +from nomad.config.models.plugins import SchemaPackageEntryPoint -class CVDEvaporationSource(EvaporationSource): - pressure = SubSection( - section_def=Pressure, - ) - precursor_partial_pressure = SubSection( - section_def=PartialVaporPressure, - ) - temperature = SubSection( - section_def=Temperature, - ) - total_flow_rate = SubSection( - section_def=VolumetricFlowRate, - description=""" - The total flow rate exiting the source. - It can be the sum of precursor and carrier gas or only a gas, - depending on the nature of the source. - """, - ) +class CvdSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.cvd.general import m_package + return m_package -class BubblerEvaporator(CVDEvaporationSource): - """ - Delivers precursor materials to the reaction chamber. - It serves as a mechanism for introducing volatile liquid or solid precursors into the gas phase, - where they can react and deposit onto a substrate surface to form thin films or coatings. - Key components: - - Bubbler Vessel: This vessel holds the precursor material. - - Heating Element: To facilitate vaporization. - - Gas Inlet and Outlet: Gas delivery system via gas inlet and outlet ports. - - Temperature Control: Maintain the vapor pressure of the precursor at the desired level. - - Operation: - - Loading Precursor: The precursor material is loaded into the bubbler vessel - - Heating: The heating element is activated to form a vapor phase above the liquid or solid. - - Gas Flow: Carrier gas is bubbled through the precursor material. - - Transport: The precursor vapor is delivered to the reaction chamber. - The precursor undergoes decomposition or reaction on the substrate surface, - leading to thin film growth. - """ - - carrier_gas = SubSection( - section_def=PubChemPureSubstanceSection, - ) - carrier_push_flow_rate = SubSection( - section_def=VolumetricFlowRate, - description=""" - The flow through the push valve. - """, - ) - carrier_purge_flow_rate = SubSection( - section_def=VolumetricFlowRate, - description=""" - The flow through the purge valve. - """, - ) - dilution = Quantity( - type=float, - description="ONLY FOR DOPING PRECURSOR", - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit="cm ** 3 / minute", - ), - unit="cm ** 3 / minute", - ) - source = Quantity( - type=float, - description="ONLY FOR DOPING PRECURSOR", - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit="cm ** 3 / minute", - ), - unit="cm ** 3 / minute", - ) - inject = Quantity( - type=float, - description="ONLY FOR DOPING PRECURSOR", - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - defaultDisplayUnit="cm ** 3 / minute", - ), - unit="cm ** 3 / minute", - ) - - -class FlashEvaporator(CVDEvaporationSource): - """ - Flash Evaporator Unit: It typically comprises a reservoir where the metalorganic precursor, often in liquid form, is stored. - - Components: - - - Heating Mechanism. - - Carrier Gas Inlet. - - Precursor Delivery Pathway. - - Temperature Control System. - - Operation: - - - Loading of Precursor. - - Vaporization Process. - - Carrier Gas Introduction. - - Transport to Reaction Chamber. - - Temperature Regulation. - """ - - carrier_gas = SubSection( - section_def=PubChemPureSubstanceSection, - ) - carrier_push_flow_rate = SubSection( - section_def=VolumetricFlowRate, - description=""" - The flow through the push valve. - """, - ) - carrier_purge_flow_rate = SubSection( - section_def=VolumetricFlowRate, - description=""" - The flow through the purge valve. - """, - ) - pass - - -class MistEvaporator(CVDEvaporationSource): - """ - MIST-CVD source is a novel method for the deposition of thin films. - """ - - -class GasLineEvaporator(CVDEvaporationSource): - """ - In chemical vapor deposition (CVD), the gas supply plays a critical role - in providing the necessary precursor molecules for the deposition process. - - Gas lines are used to transport the precursor gases from their source to the reaction chamber. - These lines are often made of materials that are compatible with the precursor gases - and can withstand the process conditions. - They may also be heated or insulated to maintain the gases at the desired temperature - and prevent condensation or undesired reactions within the lines. - """ - - pass - - -class GasCylinderEvaporator(CVDEvaporationSource): - """ - In chemical vapor deposition (CVD), the gas supply plays a critical role - in providing the necessary precursor molecules for the deposition process. - - Contains the precursor gases under pressure. - These cylinders are connected to the CVD chamber through a system of valves, - regulators, and tubing. - The flow rate of each gas can be controlled precisely using flow meters - or mass flow controllers to achieve the desired deposition conditions. - """ - - dilution_in_cylinder = Quantity( - type=float, - description='The gas dilution in the cylinder.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - ), - ) - - effective_flow_rate = SubSection( - section_def=VolumetricFlowRate, - description=""" - Effective flow rate, to be defined better. - """, - ) - - -class CVDSource(VaporDepositionSource): - """ - A source of vapor for chemical vapor deposition (CVD) processes. - """ - valve = Quantity( - type=bool, - description='is the valve open?', - a_eln=ELNAnnotation( - component='BoolEditQuantity', - ), - ) - vapor_source = SubSection( - section_def=CVDEvaporationSource, - description=""" - Example: A heater, a filament, a laser, a bubbler, etc. - """, - ) - - -class BubblerSource(CVDSource): - vapor_source = SubSection( - section_def=BubblerEvaporator, - ) - - -class GasLineSource(CVDSource): - vapor_source = SubSection( - section_def=GasLineEvaporator, - ) - - -class GasCylinderSource(CVDSource): - vapor_source = SubSection( - section_def=GasCylinderEvaporator, - ) - - -class FlashSource(CVDSource): - vapor_source = SubSection( - section_def=FlashEvaporator, - description=""" - Example: A heater, a filament, a laser, a bubbler, etc. - """, - ) - - -class MistSource(CVDSource): - """ - Mist-CVD source is a novel method for the deposition of thin films. - """ - item = Quantity( - type=str, - description='An ID used to identify the solution.', - a_eln=ELNAnnotation( - component='StringEditQuantity', - ), - ) - stirring_time = Quantity( - type=float, - description='Solution stirring time.', - a_eln=ELNAnnotation( - component='NumberEditQuantity', - defaultDisplayUnit='minute', - ), - unit='second', - ) - description = Quantity( - type=str, - description='Some notes.', - a_eln=ELNAnnotation( - component='StringEditQuantity', - ), - ) - vapor_source = SubSection( - section_def=MistEvaporator, - ) - material = SubSection(section_def=ComponentConcentration, repeats=True) - - -m_package.__init_metainfo__() +schema = CvdSchemaPackageEntryPoint( + name='CVD Schema', + description='Schema package for general CVD techniques.', +) diff --git a/src/nomad_material_processing/vapor_deposition/cvd/general.py b/src/nomad_material_processing/vapor_deposition/cvd/general.py new file mode 100644 index 0000000..7a9ca36 --- /dev/null +++ b/src/nomad_material_processing/vapor_deposition/cvd/general.py @@ -0,0 +1,471 @@ +# +# Copyright The NOMAD Authors. +# +# This file is part of NOMAD. See https://nomad-lab.eu for further info. +# +# 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. +# +from typing import ( + TYPE_CHECKING, +) +from nomad.metainfo import ( + SchemaPackage, + Section, + SubSection, + Quantity, + MEnum, +) + +from nomad.datamodel.metainfo.annotations import ( + ELNAnnotation, + ELNComponentEnum, +) + +from nomad.datamodel.data import ( + ArchiveSection, +) +from nomad.datamodel.metainfo.plot import PlotSection, PlotlyFigure + +from nomad.datamodel.metainfo.basesections import ( + PubChemPureSubstanceSection, + PureSubstanceComponent, +) +from nomad_material_processing.general import ( + TimeSeries, +) +from nomad_material_processing.vapor_deposition.general import ( + EvaporationSource, + VaporDepositionSource, + MolarFlowRate, + VolumetricFlowRate, + Pressure, + Temperature, + GasFlow, +) + +if TYPE_CHECKING: + from nomad.datamodel.datamodel import ( + EntryArchive, + ) + from structlog.stdlib import ( + BoundLogger, + ) + +from nomad.config import config + +m_package = SchemaPackage( + aliases=[ + 'nomad_material_processing.vapor_deposition.cvd', + ] +) + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition.cvd:schema', +) + + +class ComponentConcentration(PureSubstanceComponent): + """ + The concentration of a component in a mixed material. + """ + + theoretical_concentration = Quantity( + type=float, + description='The concentration planned for the component.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='mol / liter', + minValue=0, + ), + unit='mol / liter', + ) + effective_concentration = Quantity( + type=float, + description='The concentration calculated from the component moles and total volume.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='mol / liter', + minValue=0, + ), + unit='mol / liter', + ) + + +class PushPurgeGasFlow(GasFlow): + """ + Section describing the flow of a gas. + """ + + m_def = Section( + a_plot=[ + dict( + # x=['flow_rate/time', 'flow_rate/set_time'], + # y=['flow_rate/value', 'flow_rate/set_value'], + x='flow_rate/time', + y='flow_rate/value', + ), + dict( + x='purge_flow_rate/time', + y='purge_flow_rate/value', + ), + ], + ) + flow_rate = SubSection( + section_def=VolumetricFlowRate, + label='push_flow_rate', + ) + purge_flow_rate = SubSection( + section_def=VolumetricFlowRate, + ) + + +class Rotation(TimeSeries): + """ + Rotation + """ + + m_def = Section(label_quantity='set_value') + + set_value = Quantity( + type=float, + description='The value scalar set for this parameter.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='rpm', + ), + unit='rpm', + ) + value = Quantity( + type=float, + description='The rotation of the sample holder, or susceptor.', + # a_eln=ELNAnnotation( + # component="NumberEditQuantity", + # defaultDisplayUnit="rpm", + # ), + unit='rpm', + ) + + +class PartialVaporPressure(Pressure): + """ + The Partial Vapor Pressure (or Equilibrium Vapor Pressure), p, is the pressure exerted + by a vapor in thermodynamic equilibrium with its condensed phases (solid or liquid) + at a given temperature in a closed system. + + It can be approximately calculated by the semiempirical Antoine equation. + It is a relation between the vapor pressure and temperature of pure substances. + log10(p) = A - [B / (T + C)] + https://en.wikipedia.org/wiki/Vapor_pressure + The August-Antoine equation is a simplified version of the Antoine equation, + sometimes used to calculate Partial Vapor Pressure. + This assumes a temperature-independent heat of vaporization, i.e., C = 0. + https://en.wikipedia.org/wiki/Antoine_equation + """ + + set_value = Quantity( + type=float, + description='The value scalar set for this parameter.', + a_eln={'component': 'NumberEditQuantity', 'defaultDisplayUnit': 'mbar'}, + unit='pascal', + ) + value = Quantity( + type=float, + unit='pascal', + shape=['*'], + ) + time = Quantity( + type=float, + unit='second', + shape=['*'], + ) + + +class BubblerMolarFlowRate(MolarFlowRate): + """ + Molar flow rate is the amount of a substance which passes per unit of time. + + The article cited below explains the equation used in MOVPE to calculate the molar flow rate. + + F_r = F_c*P_r / (P_0 - P_r) + + where: + + F_r is the molar flow rate, + F_c is the carrier gas flow rate, + P_r is the partial vapor pressure of the precursor, + P_0 is the total pressure exiting the bubbler. + + Reference: + Journal of Vacuum Science & Technology A 8, 800 (1990); doi: 10.1116/1.576921 + + """ + + value = MolarFlowRate.value.m_copy() + set_value = MolarFlowRate.set_value.m_copy() + set_value.a_eln.defaultDisplayUnit = 'mol/minute' + + +class CVDEvaporationSource(EvaporationSource): + pressure = SubSection( + section_def=Pressure, + ) + precursor_partial_pressure = SubSection( + section_def=PartialVaporPressure, + ) + temperature = SubSection( + section_def=Temperature, + ) + total_flow_rate = SubSection( + section_def=VolumetricFlowRate, + description=""" + The total flow rate exiting the source. + It can be the sum of precursor and carrier gas or only a gas, + depending on the nature of the source. + """, + ) + + +class BubblerEvaporator(CVDEvaporationSource): + """ + Delivers precursor materials to the reaction chamber. + It serves as a mechanism for introducing volatile liquid or solid precursors into the gas phase, + where they can react and deposit onto a substrate surface to form thin films or coatings. + + Key components: + - Bubbler Vessel: This vessel holds the precursor material. + - Heating Element: To facilitate vaporization. + - Gas Inlet and Outlet: Gas delivery system via gas inlet and outlet ports. + - Temperature Control: Maintain the vapor pressure of the precursor at the desired level. + + Operation: + - Loading Precursor: The precursor material is loaded into the bubbler vessel + - Heating: The heating element is activated to form a vapor phase above the liquid or solid. + - Gas Flow: Carrier gas is bubbled through the precursor material. + - Transport: The precursor vapor is delivered to the reaction chamber. + The precursor undergoes decomposition or reaction on the substrate surface, + leading to thin film growth. + """ + + carrier_gas = SubSection( + section_def=PubChemPureSubstanceSection, + ) + carrier_push_flow_rate = SubSection( + section_def=VolumetricFlowRate, + description=""" + The flow through the push valve. + """, + ) + carrier_purge_flow_rate = SubSection( + section_def=VolumetricFlowRate, + description=""" + The flow through the purge valve. + """, + ) + dilution = Quantity( + type=float, + description='ONLY FOR DOPING PRECURSOR', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='cm ** 3 / minute', + ), + unit='cm ** 3 / minute', + ) + source = Quantity( + type=float, + description='ONLY FOR DOPING PRECURSOR', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='cm ** 3 / minute', + ), + unit='cm ** 3 / minute', + ) + inject = Quantity( + type=float, + description='ONLY FOR DOPING PRECURSOR', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='cm ** 3 / minute', + ), + unit='cm ** 3 / minute', + ) + + +class FlashEvaporator(CVDEvaporationSource): + """ + Flash Evaporator Unit: It typically comprises a reservoir where the metalorganic precursor, often in liquid form, is stored. + + Components: + + - Heating Mechanism. + - Carrier Gas Inlet. + - Precursor Delivery Pathway. + - Temperature Control System. + + Operation: + + - Loading of Precursor. + - Vaporization Process. + - Carrier Gas Introduction. + - Transport to Reaction Chamber. + - Temperature Regulation. + """ + + carrier_gas = SubSection( + section_def=PubChemPureSubstanceSection, + ) + carrier_push_flow_rate = SubSection( + section_def=VolumetricFlowRate, + description=""" + The flow through the push valve. + """, + ) + carrier_purge_flow_rate = SubSection( + section_def=VolumetricFlowRate, + description=""" + The flow through the purge valve. + """, + ) + pass + + +class MistEvaporator(CVDEvaporationSource): + """ + MIST-CVD source is a novel method for the deposition of thin films. + """ + + +class GasLineEvaporator(CVDEvaporationSource): + """ + In chemical vapor deposition (CVD), the gas supply plays a critical role + in providing the necessary precursor molecules for the deposition process. + + Gas lines are used to transport the precursor gases from their source to the reaction chamber. + These lines are often made of materials that are compatible with the precursor gases + and can withstand the process conditions. + They may also be heated or insulated to maintain the gases at the desired temperature + and prevent condensation or undesired reactions within the lines. + """ + + pass + + +class GasCylinderEvaporator(CVDEvaporationSource): + """ + In chemical vapor deposition (CVD), the gas supply plays a critical role + in providing the necessary precursor molecules for the deposition process. + + Contains the precursor gases under pressure. + These cylinders are connected to the CVD chamber through a system of valves, + regulators, and tubing. + The flow rate of each gas can be controlled precisely using flow meters + or mass flow controllers to achieve the desired deposition conditions. + """ + + dilution_in_cylinder = Quantity( + type=float, + description='The gas dilution in the cylinder.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + ), + ) + + effective_flow_rate = SubSection( + section_def=VolumetricFlowRate, + description=""" + Effective flow rate, to be defined better. + """, + ) + + +class CVDSource(VaporDepositionSource): + """ + A source of vapor for chemical vapor deposition (CVD) processes. + """ + + valve = Quantity( + type=bool, + description='is the valve open?', + a_eln=ELNAnnotation( + component='BoolEditQuantity', + ), + ) + vapor_source = SubSection( + section_def=CVDEvaporationSource, + description=""" + Example: A heater, a filament, a laser, a bubbler, etc. + """, + ) + + +class BubblerSource(CVDSource): + vapor_source = SubSection( + section_def=BubblerEvaporator, + ) + + +class GasLineSource(CVDSource): + vapor_source = SubSection( + section_def=GasLineEvaporator, + ) + + +class GasCylinderSource(CVDSource): + vapor_source = SubSection( + section_def=GasCylinderEvaporator, + ) + + +class FlashSource(CVDSource): + vapor_source = SubSection( + section_def=FlashEvaporator, + description=""" + Example: A heater, a filament, a laser, a bubbler, etc. + """, + ) + + +class MistSource(CVDSource): + """ + Mist-CVD source is a novel method for the deposition of thin films. + """ + + item = Quantity( + type=str, + description='An ID used to identify the solution.', + a_eln=ELNAnnotation( + component='StringEditQuantity', + ), + ) + stirring_time = Quantity( + type=float, + description='Solution stirring time.', + a_eln=ELNAnnotation( + component='NumberEditQuantity', + defaultDisplayUnit='minute', + ), + unit='second', + ) + description = Quantity( + type=str, + description='Some notes.', + a_eln=ELNAnnotation( + component='StringEditQuantity', + ), + ) + vapor_source = SubSection( + section_def=MistEvaporator, + ) + material = SubSection(section_def=ComponentConcentration, repeats=True) + + +m_package.__init_metainfo__() diff --git a/src/nomad_material_processing/vapor_deposition/cvd/nomad_plugin.yaml b/src/nomad_material_processing/vapor_deposition/cvd/nomad_plugin.yaml deleted file mode 100644 index b3bc9a0..0000000 --- a/src/nomad_material_processing/vapor_deposition/cvd/nomad_plugin.yaml +++ /dev/null @@ -1,3 +0,0 @@ -description: A plugin for NOMAD containing base sections for chemical vapor deposition. -name: Chemical Vapor Deposition -plugin_type: schema diff --git a/src/nomad_material_processing/vapor_deposition/general.py b/src/nomad_material_processing/vapor_deposition/general.py new file mode 100644 index 0000000..2be3493 --- /dev/null +++ b/src/nomad_material_processing/vapor_deposition/general.py @@ -0,0 +1,735 @@ +# +# Copyright The NOMAD Authors. +# +# This file is part of NOMAD. See https://nomad-lab.eu for further info. +# +# 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. +# +from typing import ( + TYPE_CHECKING, +) +from nomad.metainfo import ( + SchemaPackage, + Section, + SubSection, + Quantity, + MEnum, +) +from nomad.datamodel.data import ( + ArchiveSection, +) +from nomad.datamodel.metainfo.annotations import ( + ELNAnnotation, + ELNComponentEnum, +) +from nomad.datamodel.metainfo.basesections import ( + Entity, + ActivityStep, + PureSubstanceSection, + Component, + CompositeSystemReference, + PubChemPureSubstanceSection, +) +from nomad.datamodel.metainfo.plot import ( + PlotSection, +) +from nomad.datamodel.metainfo.workflow import ( + Link, + Task, +) +from nomad_material_processing.general import ( + SampleDeposition, + ThinFilmStackReference, + ThinFilmReference, + TimeSeries, + Geometry, +) + +if TYPE_CHECKING: + from nomad.datamodel.datamodel import ( + EntryArchive, + ) + from structlog.stdlib import ( + BoundLogger, + ) + +from nomad.config import config + +m_package = SchemaPackage(name='Vapor Deposition') + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition:schema', +) + + +class InsertReduction(Entity): + """ + The reduction that sometimes is used to lodge the substrate in the substrate holder position.. + """ + + name = Quantity( + type=str, + description=""" + A short and descriptive name for this insert reduction. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + insert_id = Quantity( + type=str, + description=""" + The ID of the insert that is placed in this position to accomodate the substrate. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + material = SubSection(section_def=PubChemPureSubstanceSection, repeats=True) + inner_geometry = SubSection( + section_def=Geometry, + ) + outer_geometry = SubSection( + section_def=Geometry, + ) + + +class SubstrateHolderPosition(ArchiveSection): + """ + One casing position of the substrate holder. + """ + + name = Quantity( + type=str, + description=""" + A short name for this position. This name is used as label of the position. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + x_position = Quantity( + type=float, + unit='meter', + description=""" + The x coordinate of the substrate holder position relative to the center of the holder. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + ), + ) + y_position = Quantity( + type=float, + unit='meter', + description=""" + The y coordinate of the substrate holder position relative to the center of the holder. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + ), + ) + slot_geometry = SubSection( + section_def=Geometry, + ) + insert_reduction = SubSection( + section_def=InsertReduction, + description='Optional description of insert if used.', + ) + + +class SubstrateHolder(Entity): + """ + The holder for the substrate. + """ + + name = Quantity( + type=str, + description=""" + A short and descriptive name for this position. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + lab_id = Quantity( + type=str, + description=""" + The lab ID of the substrate holder. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.StringEditQuantity, + ), + ) + material = SubSection(section_def=PubChemPureSubstanceSection, repeats=True) + thickness = Quantity( + type=float, + unit='meter', + description=""" + The thickness of the holder to the back of the substrate. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='micrometer', + ), + ) + outer_diameter = Quantity( + type=float, + unit='meter', + description=""" + The outer diameter of the substrate holder. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + defaultDisplayUnit='millimeter', + ), + ) + number_of_positions = Quantity( + type=int, + description=""" + The number of positions on the holder. + """, + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + ), + ) + image = Quantity( + type=str, + description="""An image of the substrate holder.""", + a_browser={'adaptor': 'RawFileAdaptor'}, + a_eln=ELNAnnotation( + component=ELNComponentEnum.FileEditQuantity, + ), + ) + positions = SubSection( + section_def=SubstrateHolderPosition, + repeats=True, + ) + + +class FilledSubstrateHolderPosition(SubstrateHolderPosition): + """ + One casing position of the filled substrate holder. + """ + + substrate = SubSection( + section_def=CompositeSystemReference, + description=""" + The substrate that is placed in this position. + """, + ) + + +class FilledSubstrateHolder(SubstrateHolder): + """ + A substrate holder that is filled with substrate(s). + """ + + substrate_holder = SubSection( + section_def=SubstrateHolder, + ) + positions = SubSection( + section_def=FilledSubstrateHolderPosition, + repeats=True, + ) + + +class MolarFlowRate(TimeSeries): + """ + Molar flow rate is the amount of a substance which passes per unit of time. + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + measurement_type = Quantity( + type=MEnum( + 'Assumed', + 'Mass Flow Controller', + ), + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + value = TimeSeries.value.m_copy() + value.unit = 'mol/second' + set_value = TimeSeries.set_value.m_copy() + set_value.unit = 'mol/second' + + +class EvaporationSource(ArchiveSection): + pass + + +class VaporDepositionSource(ArchiveSection): + name = Quantity( + type=str, + description=""" + A short and descriptive name for this source. + """, + ) + material = SubSection( + section_def=Component, + description=""" + The source of the material that is being evaporated. + Example: A sputtering target, a powder in a crucible, etc. + """, + repeats=True, + ) + vapor_source = SubSection( + section_def=EvaporationSource, + description=""" + Example: A heater, a filament, a laser, a bubbler, etc. + """, + ) + vapor_molar_flow_rate = SubSection( + section_def=MolarFlowRate, + description=""" + The rate of the material being evaporated (mol/time). + """, + ) + + +class GrowthRate(TimeSeries): + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + measurement_type = Quantity( + type=MEnum( + 'Assumed', + 'RHEED', + 'Reflectance', + ), + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + # value = TimeSeries.value.m_copy() + # value.unit = 'meter/second' + # set_value = TimeSeries.set_value.m_copy() + # set_value.unit = 'meter/second' + # set_value.a_eln.defaultDisplayUnit = 'nm/second' + value = Quantity( + type=float, + unit='meter/second', + shape=['*'], + ) + set_value = Quantity( + type=float, + description='The set value(s) (i.e. the intended values) set.', + shape=['*'], + unit='meter/second', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + ), + ) + + +class Temperature(TimeSeries): + """ + Generic Temperature monitoring + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + measurement_type = Quantity( + type=MEnum( + 'Heater thermocouple', + 'Thermocouple', + 'Pyrometer', + 'Assumed', + ), + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + # value = TimeSeries.value.m_copy() + # value.unit = 'kelvin' + # set_value = TimeSeries.set_value.m_copy() + # set_value.unit = 'kelvin' + # set_value.a_eln.defaultDisplayUnit = 'celsius' + value = Quantity( + type=float, + unit='kelvin', + shape=['*'], + ) + set_value = Quantity( + type=float, + description='The set value(s) (i.e. the intended values) set.', + shape=['*'], + unit='kelvin', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + ), + ) + + +class SampleParameters(PlotSection, ArchiveSection): + m_def = Section( + a_plotly_graph_object={ + 'label': 'Measured Temperatures', + 'index': 1, + 'dragmode': 'pan', + 'data': { + 'type': 'scattergl', + 'line': {'width': 2}, + 'marker': {'size': 2}, + 'mode': 'lines+markers', + 'name': 'Temperature', + 'x': '#temperature/time', + 'y': '#temperature/value', + }, + 'layout': { + 'title': {'text': 'Measured Temperature'}, + 'xaxis': { + 'showticklabels': True, + 'fixedrange': True, + 'ticks': '', + 'title': {'text': 'Process time [s]'}, + 'showline': True, + 'linewidth': 1, + 'linecolor': 'black', + 'mirror': True, + }, + 'yaxis': { + 'showticklabels': True, + 'fixedrange': True, + 'ticks': '', + 'title': {'text': 'Temperature [°C]'}, + 'showline': True, + 'linewidth': 1, + 'linecolor': 'black', + 'mirror': True, + }, + 'showlegend': False, + }, + 'config': { + 'displayModeBar': False, + 'scrollZoom': False, + 'responsive': False, + 'displaylogo': False, + 'dragmode': False, + }, + }, + ) + growth_rate = SubSection( + section_def=GrowthRate, + description=""" + The growth rate of the thin film (length/time). + Measured by in-situ RHEED or Reflection or assumed. + """, + ) + substrate_temperature = SubSection( + section_def=Temperature, + ) + layer = SubSection( + description=""" + The thin film that is being created during this step. + """, + section_def=ThinFilmReference, + ) + substrate = SubSection( + description=""" + The thin film stack that is being evaporated on. + """, + section_def=ThinFilmStackReference, + ) + + +class Pressure(TimeSeries): + """ + The pressure during the deposition process. + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + # value = TimeSeries.value.m_copy() + # value.unit = 'pascal' + # set_value = TimeSeries.set_value.m_copy() + # set_value.unit = 'pascal' + # set_value.a_eln.defaultDisplayUnit = 'mbar' + value = Quantity( + type=float, + unit='pascal', + shape=['*'], + ) + time = Quantity( + type=float, + unit='second', + shape=['*'], + ) + set_value = Quantity( + type=float, + unit='pascal', + shape=['*'], + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + ), + ) + set_time = Quantity( + type=float, + unit='second', + shape=['*'], + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set time', + ), + ) + + +class VolumetricFlowRate(TimeSeries): + """ + The volumetric flow rate of a gas at standard conditions, i.e. the equivalent rate + at a temperature of 0 °C (273.15 K) and a pressure of 1 atm (101325 Pa). + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + measurement_type = Quantity( + type=MEnum( + 'Mass Flow Controller', + 'Flow Meter', + 'Other', + ), + ) + # value = TimeSeries.value.m_copy() + # value.unit = 'meter ** 3 / second' + # set_value = TimeSeries.set_value.m_copy() + # set_value.unit = 'meter ** 3 / second' + # set_value.a_eln.defaultDisplayUnit = 'centimeter ** 3 / minute' + value = Quantity( + type=float, + unit='meter ** 3 / second', + shape=['*'], + ) + set_value = Quantity( + type=float, + description='The set value(s) (i.e. the intended values) set.', + shape=['*'], + unit='meter ** 3 / second', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + defaultDisplayUnit='centimeter ** 3 / minute', + ), + ) + + +class GasFlow(ArchiveSection): + """ + Section describing the flow of a gas. + """ + + m_def = Section( + a_plot=dict( + # x=['flow_rate/time', 'flow_rate/set_time'], + # y=['flow_rate/value', 'flow_rate/set_value'], + x='flow_rate/time', + y='flow_rate/value', + ), + ) + gas = SubSection( + section_def=PureSubstanceSection, + ) + flow_rate = SubSection( + section_def=VolumetricFlowRate, + ) + + +class SubstrateHeater(ArchiveSection): + pass + + +class ChamberEnvironment(ArchiveSection): + m_def = Section( + a_plot=dict( + x='pressure/time', + y='pressure/value', + ), + ) + gas_flow = SubSection( + section_def=GasFlow, + repeats=True, + ) + pressure = SubSection( + section_def=Pressure, + ) + heater = SubSection( + section_def=SubstrateHeater, + ) + + +class VaporDepositionStep(ActivityStep): + """ + A step of any vapor deposition process. + """ + + m_def = Section() + creates_new_thin_film = Quantity( + type=bool, + description=""" + Whether or not this step creates a new thin film. + """, + default=False, + a_eln=ELNAnnotation( + component='BoolEditQuantity', + ), + ) + duration = Quantity( + type=float, + unit='second', + ) + sources = SubSection( + section_def=VaporDepositionSource, + repeats=True, + ) + sample_parameters = SubSection( + section_def=SampleParameters, + repeats=True, + ) + environment = SubSection( + section_def=ChamberEnvironment, + ) + + def to_task(self) -> Task: + """ + Returns the task description of this activity step. + + Returns: + Task: The activity step as a workflow task. + """ + inputs = [] + for source in self.sources: + if source.material is not None and hasattr(source.material, 'system'): + inputs.append( + Link( + name=getattr(source.material, 'name', None), + section=getattr(source.material, 'system', None), + ) + ) + elif source.material is not None and hasattr( + source.material, 'pure_substance' + ): + inputs.append( + Link( + name=getattr(source.material, 'substance_name', None), + section=getattr(source.material, 'pure_substance', None), + ) + ) + outputs = [ + Link( + name=parameters.layer.name, + section=parameters.layer.reference, + ) + for parameters in self.sample_parameters + if parameters.layer is not None and parameters.layer.reference is not None + ] + return Task(name=self.name, inputs=inputs, outputs=outputs) + + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: + """ + The normalizer for the `VaporDepositionStep` class. + + Args: + archive (EntryArchive): The archive containing the section that is being + normalized. + logger (BoundLogger): A structlog logger. + """ + super(VaporDepositionStep, self).normalize(archive, logger) + + +class VaporDeposition(SampleDeposition): + """ + VaporDeposition is a general class that encompasses both Physical Vapor Deposition + (PVD) and Chemical Vapor Deposition (CVD). + It involves the deposition of material from a vapor phase to a solid thin film or + coating onto a substrate. + - material sources: + Both PVD and CVD involve a source material that is transformed into a vapor phase. + In PVD, the source material is physically evaporated or sputtered from a solid + target. + In CVD, gaseous precursors undergo chemical reactions to produce a solid material + on the substrate. + - substrate: + The substrate is the material onto which the thin film is deposited. + - environment: + The process typically takes place in a controlled environment. + The deposition is usually affected by the pressure in the chamber. + For some processes additional background gasses are also added. + """ + + m_def = Section( + links=[ + 'http://purl.obolibrary.org/obo/CHMO_0001314', + 'http://purl.obolibrary.org/obo/CHMO_0001356', + ], + a_plot=[ + dict( + x='steps/:/environment/pressure/time', + y='steps/:/environment/pressure/value', + ), + ], + ) + steps = SubSection( + description=""" + The steps of the deposition process. + """, + section_def=VaporDepositionStep, + repeats=True, + ) + + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: + """ + The normalizer for the `VaporDeposition` class. + + Args: + archive (EntryArchive): The archive containing the section that is being + normalized. + logger (BoundLogger): A structlog logger. + """ + super(VaporDeposition, self).normalize(archive, logger) + + +m_package.__init_metainfo__() diff --git a/src/nomad_material_processing/vapor_deposition/nomad_plugin.yaml b/src/nomad_material_processing/vapor_deposition/nomad_plugin.yaml deleted file mode 100644 index 5695f9d..0000000 --- a/src/nomad_material_processing/vapor_deposition/nomad_plugin.yaml +++ /dev/null @@ -1,3 +0,0 @@ -description: A plugin for NOMAD containing base sections for vapor deposition. -name: Vapor Deposition -plugin_type: schema diff --git a/src/nomad_material_processing/vapor_deposition/pvd/__init__.py b/src/nomad_material_processing/vapor_deposition/pvd/__init__.py index 74391ed..64e5e09 100644 --- a/src/nomad_material_processing/vapor_deposition/pvd/__init__.py +++ b/src/nomad_material_processing/vapor_deposition/pvd/__init__.py @@ -15,257 +15,74 @@ # See the License for the specific language governing permissions and # limitations under the License. # -from typing import ( - TYPE_CHECKING, -) -from nomad.metainfo import ( - Package, - Section, - SubSection, - Quantity, - MEnum, -) -from nomad.datamodel.metainfo.annotations import ( - ELNAnnotation, - ELNComponentEnum, -) - -from nomad_material_processing import ( - TimeSeries, -) -from nomad_material_processing.vapor_deposition import ( - EvaporationSource, - VaporDepositionSource, - SampleParameters, - VaporDepositionStep, - VaporDeposition, -) - -if TYPE_CHECKING: - from nomad.datamodel.datamodel import ( - EntryArchive, - ) - from structlog.stdlib import ( - BoundLogger, - ) - -m_package = Package(name='Physical Vapor Deposition') +from nomad.config.models.plugins import SchemaPackageEntryPoint -class SourcePower(TimeSeries): - """ - The power supplied to the source (watt). - """ +class GeneralPvdSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.pvd.general import m_package - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) + return m_package - value = Quantity( - type=float, - unit='watt', - shape=['*'], - ) - set_value = Quantity( - type=float, - description='The set value(s) (i.e. the intended values) set.', - shape=['*'], - unit='watt', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - defaultDisplayUnit='watt', - ), - ) - - -class PVDEvaporationSource(EvaporationSource): - m_def = Section( - a_plot=dict( - x='power/time', - y='power/value', - ), - ) - power = SubSection( - section_def=SourcePower, - ) +schema = GeneralPvdSchemaPackageEntryPoint( + name='General PVD Schema', + description="""Schema package containing basic classes used + in the vapor_deposition submodule.""", +) -class ImpingingFlux(TimeSeries): - """ - The impinging flux of the material onto the substrate (mol/area/time). - """ - m_def = Section( - a_plot=dict( - # x=['time', 'set_time'], - # y=['value', 'set_value'], - x='time', - y='value', - ), - ) - measurement_type = Quantity( - type=MEnum( - 'Assumed', - 'Quartz Crystal Microbalance', - ), - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) +class MbeSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.pvd.mbe import m_package - value = Quantity( - type=float, - unit='mol/meter ** 2/second', - shape=['*'], - ) - set_value = Quantity( - type=float, - description='The set value(s) (i.e. the intended values) set.', - shape=['*'], - unit='mol/meter ** 2/second', - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - label='Set value', - defaultDisplayUnit='mol/meter ** 2/second', - ), - ) + return m_package -class PVDSource(VaporDepositionSource): - m_def = Section( - a_plot=[ - dict( - x=[ - 'vapor_source/power/time', - 'impinging_flux/:/time', - ], - y=[ - 'vapor_source/power/value', - 'impinging_flux/:/value', - ], - ), - ], - ) - vapor_source = SubSection( - section_def=PVDEvaporationSource, - description=""" - Example: A heater, a filament, a laser, etc. - """, - ) - impinging_flux = SubSection( - section_def=ImpingingFlux, - description=""" - The deposition rate of the material onto the substrate (mol/area/time). - """, - repeats=True, - ) +mbe_schema = MbeSchemaPackageEntryPoint( + name='Mbe Schema', + description="""Schema package containing basic classes used + in the MBE submodule.""", +) -class PVDSampleParameters(SampleParameters): - heater = Quantity( - description=""" - What is the substrate heated by. - """, - type=MEnum( - 'No heating', - 'Halogen lamp', - 'Filament', - 'Resistive element', - 'CO2 laser', - ), - a_eln=ELNAnnotation( - component=ELNComponentEnum.EnumEditQuantity, - ), - ) - distance_to_source = Quantity( - type=float, - unit='meter', - description=""" - The distance between the substrate and all the sources. - In the case of multiple sources, the distances are listed in the same order as the - sources are listed in the parent `VaporDepositionStep` section. - """, - shape=['*'], - a_eln=ELNAnnotation( - component=ELNComponentEnum.NumberEditQuantity, - ), - ) +class PldSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.pvd.pld import m_package + return m_package -class PVDStep(VaporDepositionStep): - """ - A step of any physical vapor deposition process. - """ - sources = SubSection( - section_def=PVDSource, - repeats=True, - ) - sample_parameters = SubSection( - section_def=PVDSampleParameters, - repeats=True, - ) +pld_schema = PldSchemaPackageEntryPoint( + name='Pld Schema', + description="""Schema package containing basic classes used + in the PLD submodule.""", +) - def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - """ - The normalizer for the `PVDStep` class. - Args: - archive (EntryArchive): The archive containing the section that is being - normalized. - logger (BoundLogger): A structlog logger. - """ - super(PVDStep, self).normalize(archive, logger) +class SputteringSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.pvd.sputtering import m_package + return m_package -class PhysicalVaporDeposition(VaporDeposition): - """ - A synthesis technique where vaporized molecules or atoms condense on a surface, - forming a thin layer. The process is purely physical; no chemical reaction occurs - at the surface. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] - Synonyms: - - PVD - - physical vapor deposition - """ +sputtering_schema = SputteringSchemaPackageEntryPoint( + name='Sputtering Schema', + description="""Schema package containing basic classes used + in the sputtering submodule.""", +) - m_def = Section( - links=['http://purl.obolibrary.org/obo/CHMO_0001356'], - a_plot=[ - dict( - x='steps/:/environment/pressure/time', - y='steps/:/environment/pressure/value', - ), - dict( - x='steps/:/source/:/vapor_source/power/time', - y='steps/:/source/:/vapor_source/power/value', - ), - ], - ) - steps = SubSection( - description=""" - The steps of the deposition process. - """, - section_def=PVDStep, - repeats=True, - ) - def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: - """ - The normalizer for the `PhysicalVaporDeposition` class. +class ThermalSchemaPackageEntryPoint(SchemaPackageEntryPoint): + def load(self): + from nomad_material_processing.vapor_deposition.pvd.thermal import m_package - Args: - archive (EntryArchive): The archive containing the section that is being - normalized. - logger (BoundLogger): A structlog logger. - """ - super(PhysicalVaporDeposition, self).normalize(archive, logger) + return m_package -m_package.__init_metainfo__() +thermal_schema = ThermalSchemaPackageEntryPoint( + name='Thermal Schema', + description="""Schema package containing basic classes used + in the thermal submodule.""", +) diff --git a/src/nomad_material_processing/vapor_deposition/pvd/general.py b/src/nomad_material_processing/vapor_deposition/pvd/general.py new file mode 100644 index 0000000..4cc3e16 --- /dev/null +++ b/src/nomad_material_processing/vapor_deposition/pvd/general.py @@ -0,0 +1,282 @@ +# +# Copyright The NOMAD Authors. +# +# This file is part of NOMAD. See https://nomad-lab.eu for further info. +# +# 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. +# +from typing import ( + TYPE_CHECKING, +) +from nomad.metainfo import ( + SchemaPackage, + Section, + SubSection, + Quantity, + MEnum, +) +from nomad.datamodel.metainfo.annotations import ( + ELNAnnotation, + ELNComponentEnum, +) + +from nomad_material_processing.general import ( + TimeSeries, +) +from nomad_material_processing.vapor_deposition.general import ( + EvaporationSource, + VaporDepositionSource, + SampleParameters, + VaporDepositionStep, + VaporDeposition, +) + +if TYPE_CHECKING: + from nomad.datamodel.datamodel import ( + EntryArchive, + ) + from structlog.stdlib import ( + BoundLogger, + ) + +from nomad.config import config + +m_package = SchemaPackage( + name='Physical Vapor Deposition', + aliases=[ + 'nomad_material_processing.vapor_deposition.pvd', + ], +) + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition.pvd:schema', +) + + +class SourcePower(TimeSeries): + """ + The power supplied to the source (watt). + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + + value = Quantity( + type=float, + unit='watt', + shape=['*'], + ) + set_value = Quantity( + type=float, + description='The set value(s) (i.e. the intended values) set.', + shape=['*'], + unit='watt', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + defaultDisplayUnit='watt', + ), + ) + + +class PVDEvaporationSource(EvaporationSource): + m_def = Section( + a_plot=dict( + x='power/time', + y='power/value', + ), + ) + power = SubSection( + section_def=SourcePower, + ) + + +class ImpingingFlux(TimeSeries): + """ + The impinging flux of the material onto the substrate (mol/area/time). + """ + + m_def = Section( + a_plot=dict( + # x=['time', 'set_time'], + # y=['value', 'set_value'], + x='time', + y='value', + ), + ) + measurement_type = Quantity( + type=MEnum( + 'Assumed', + 'Quartz Crystal Microbalance', + ), + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + + value = Quantity( + type=float, + unit='mol/meter ** 2/second', + shape=['*'], + ) + set_value = Quantity( + type=float, + description='The set value(s) (i.e. the intended values) set.', + shape=['*'], + unit='mol/meter ** 2/second', + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + label='Set value', + defaultDisplayUnit='mol/meter ** 2/second', + ), + ) + + +class PVDSource(VaporDepositionSource): + m_def = Section( + a_plot=[ + dict( + x=[ + 'vapor_source/power/time', + 'impinging_flux/:/time', + ], + y=[ + 'vapor_source/power/value', + 'impinging_flux/:/value', + ], + ), + ], + ) + vapor_source = SubSection( + section_def=PVDEvaporationSource, + description=""" + Example: A heater, a filament, a laser, etc. + """, + ) + impinging_flux = SubSection( + section_def=ImpingingFlux, + description=""" + The deposition rate of the material onto the substrate (mol/area/time). + """, + repeats=True, + ) + + +class PVDSampleParameters(SampleParameters): + heater = Quantity( + description=""" + What is the substrate heated by. + """, + type=MEnum( + 'No heating', + 'Halogen lamp', + 'Filament', + 'Resistive element', + 'CO2 laser', + ), + a_eln=ELNAnnotation( + component=ELNComponentEnum.EnumEditQuantity, + ), + ) + distance_to_source = Quantity( + type=float, + unit='meter', + description=""" + The distance between the substrate and all the sources. + In the case of multiple sources, the distances are listed in the same order as the + sources are listed in the parent `VaporDepositionStep` section. + """, + shape=['*'], + a_eln=ELNAnnotation( + component=ELNComponentEnum.NumberEditQuantity, + ), + ) + + +class PVDStep(VaporDepositionStep): + """ + A step of any physical vapor deposition process. + """ + + sources = SubSection( + section_def=PVDSource, + repeats=True, + ) + sample_parameters = SubSection( + section_def=PVDSampleParameters, + repeats=True, + ) + + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: + """ + The normalizer for the `PVDStep` class. + + Args: + archive (EntryArchive): The archive containing the section that is being + normalized. + logger (BoundLogger): A structlog logger. + """ + super(PVDStep, self).normalize(archive, logger) + + +class PhysicalVaporDeposition(VaporDeposition): + """ + A synthesis technique where vaporized molecules or atoms condense on a surface, + forming a thin layer. The process is purely physical; no chemical reaction occurs + at the surface. [database_cross_reference: https://orcid.org/0000-0002-0640-0422] + + Synonyms: + - PVD + - physical vapor deposition + """ + + m_def = Section( + links=['http://purl.obolibrary.org/obo/CHMO_0001356'], + a_plot=[ + dict( + x='steps/:/environment/pressure/time', + y='steps/:/environment/pressure/value', + ), + dict( + x='steps/:/source/:/vapor_source/power/time', + y='steps/:/source/:/vapor_source/power/value', + ), + ], + ) + steps = SubSection( + description=""" + The steps of the deposition process. + """, + section_def=PVDStep, + repeats=True, + ) + + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: + """ + The normalizer for the `PhysicalVaporDeposition` class. + + Args: + archive (EntryArchive): The archive containing the section that is being + normalized. + logger (BoundLogger): A structlog logger. + """ + super(PhysicalVaporDeposition, self).normalize(archive, logger) + + +m_package.__init_metainfo__() diff --git a/src/nomad_material_processing/vapor_deposition/pvd/mbe.py b/src/nomad_material_processing/vapor_deposition/pvd/mbe.py index e093351..d688616 100644 --- a/src/nomad_material_processing/vapor_deposition/pvd/mbe.py +++ b/src/nomad_material_processing/vapor_deposition/pvd/mbe.py @@ -17,14 +17,22 @@ # from nomad.metainfo import ( - Package, + SchemaPackage, Quantity, Section, ) -from nomad_material_processing.vapor_deposition.pvd import ( +from nomad_material_processing.vapor_deposition.pvd.general import ( PhysicalVaporDeposition, ) +from nomad.config import config + +m_package = SchemaPackage(name='Molecular Beam Epitaxy') + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition.pvd:mbe_schema', +) + class MolecularBeamEpitaxy(PhysicalVaporDeposition): """ diff --git a/src/nomad_material_processing/vapor_deposition/pvd/nomad_plugin.yaml b/src/nomad_material_processing/vapor_deposition/pvd/nomad_plugin.yaml deleted file mode 100644 index 2e4393f..0000000 --- a/src/nomad_material_processing/vapor_deposition/pvd/nomad_plugin.yaml +++ /dev/null @@ -1,3 +0,0 @@ -description: A plugin for NOMAD containing base sections for physical vapor deposition. -name: Physical Vapor Deposition -plugin_type: schema diff --git a/src/nomad_material_processing/vapor_deposition/pvd/pld.py b/src/nomad_material_processing/vapor_deposition/pvd/pld.py index d895abf..a149fd0 100644 --- a/src/nomad_material_processing/vapor_deposition/pvd/pld.py +++ b/src/nomad_material_processing/vapor_deposition/pvd/pld.py @@ -19,7 +19,7 @@ TYPE_CHECKING, ) from nomad.metainfo import ( - Package, + SchemaPackage, Section, SubSection, Quantity, @@ -34,7 +34,7 @@ ReadableIdentifiers, ) -from nomad_material_processing.vapor_deposition.pvd import ( +from nomad_material_processing.vapor_deposition.pvd.general import ( PVDEvaporationSource, PVDSource, PVDStep, @@ -49,7 +49,13 @@ BoundLogger, ) -m_package = Package(name='Pulsed Laser Deposition') +from nomad.config import config + +m_package = SchemaPackage(name='Pulsed Laser Deposition') + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition.pvd:pld_schema', +) class PLDTarget(CompositeSystem): diff --git a/src/nomad_material_processing/vapor_deposition/pvd/sputtering.py b/src/nomad_material_processing/vapor_deposition/pvd/sputtering.py index 5af3c94..5633560 100644 --- a/src/nomad_material_processing/vapor_deposition/pvd/sputtering.py +++ b/src/nomad_material_processing/vapor_deposition/pvd/sputtering.py @@ -19,12 +19,12 @@ TYPE_CHECKING, ) from nomad.metainfo import ( - Package, + SchemaPackage, Section, Quantity, ) -from nomad_material_processing.vapor_deposition.pvd import ( +from nomad_material_processing.vapor_deposition.pvd.general import ( PhysicalVaporDeposition, ) @@ -36,7 +36,13 @@ BoundLogger, ) -m_package = Package(name="Sputter Deposition") +from nomad.config import config + +m_package = SchemaPackage(name='Sputter Deposition') + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition.pvd:sputtering_schema', +) class SputterDeposition(PhysicalVaporDeposition): @@ -52,14 +58,14 @@ class SputterDeposition(PhysicalVaporDeposition): """ m_def = Section( - links=["http://purl.obolibrary.org/obo/CHMO_0001364"], + links=['http://purl.obolibrary.org/obo/CHMO_0001364'], ) method = Quantity( type=str, - default="Sputter Deposition", + default='Sputter Deposition', ) - def normalize(self, archive: "EntryArchive", logger: "BoundLogger") -> None: + def normalize(self, archive: 'EntryArchive', logger: 'BoundLogger') -> None: """ The normalizer for the `SputterDeposition` class. diff --git a/src/nomad_material_processing/vapor_deposition/pvd/thermal.py b/src/nomad_material_processing/vapor_deposition/pvd/thermal.py index cfbebeb..06a7786 100644 --- a/src/nomad_material_processing/vapor_deposition/pvd/thermal.py +++ b/src/nomad_material_processing/vapor_deposition/pvd/thermal.py @@ -19,16 +19,16 @@ TYPE_CHECKING, ) from nomad.metainfo import ( - Package, + SchemaPackage, Section, SubSection, Quantity, ) -from nomad_material_processing import ( +from nomad_material_processing.general import ( TimeSeries, ) -from nomad_material_processing.vapor_deposition.pvd import ( +from nomad_material_processing.vapor_deposition.pvd.general import ( PVDEvaporationSource, PVDSource, PVDStep, @@ -48,7 +48,13 @@ BoundLogger, ) -m_package = Package(name='Thermal Evaporation') +from nomad.config import config + +m_package = SchemaPackage(name='Thermal Evaporation') + +configuration = config.get_plugin_entry_point( + 'nomad_material_processing.vapor_deposition.pvd:thermal_schema', +) class ThermalEvaporationHeaterTemperature(TimeSeries): diff --git a/tests/solution/test_solution_schema.py b/tests/solution/test_solution_schema.py index 91e40f4..ae90308 100644 --- a/tests/solution/test_solution_schema.py +++ b/tests/solution/test_solution_schema.py @@ -4,7 +4,7 @@ from nomad.units import ureg from nomad.datamodel.metainfo.basesections import PubChemPureSubstanceSection -from nomad_material_processing.solution import ( +from nomad_material_processing.solution.schema import ( Solution, SolutionComponent, SolutionComponentReference,