#
##
## This file is part of pyFormex 2.0 (Mon Sep 14 12:29:05 CEST 2020)
## pyFormex is a tool for generating, manipulating and transforming 3D
## geometrical models by sequences of mathematical operations.
## Home page: http://pyformex.org
## Project page: http://savannah.nongnu.org/projects/pyformex/
## Copyright 2004-2020 (C) Benedict Verhegghe (benedict.verhegghe@ugent.be)
## Distributed under the GNU General Public License version 3 or later.
##
## This program is free software: you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published by
## the Free Software Foundation, either version 3 of the License, or
## (at your option) any later version.
##
## This program is distributed in the hope that it will be useful,
## but WITHOUT ANY WARRANTY; without even the implied warranty of
## MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
## GNU General Public License for more details.
##
## You should have received a copy of the GNU General Public License
## along with this program. If not, see http://www.gnu.org/licenses/.
##
"""project.py
Functions for managing a project in pyFormex.
"""
import sys
import gzip
import pickle
import pyformex as pf
from pyformex import Path
from pyformex import utils
from pyformex.track import TrackedDict
_signature_ = pf.fullVersion()
default_protocol = pickle.DEFAULT_PROTOCOL
module_relocations = {
'plugins.mesh': 'pyformex.mesh',
'plugins.surface': 'pyformex.trisurface',
'plugins.trisurface': 'pyformex.trisurface',
'pyformex.plugins.mesh': 'pyformex.mesh',
'pyformex.plugins.surface': 'pyformex.trisurface',
'pyformex.plugins.trisurface': 'pyformex.trisurface',
}
class_relocations = {
'coords.Coords': 'pyformex.coords.Coords',
'coords.BoundVectors': 'pyformex.plugins.alt.BoundVectors',
'elements.Element': 'pyformex.elements.ElementType',
'elements.Line2': 'pyformex.elements.Line2',
'elements.Tri3': 'pyformex.elements.Tri3',
'elements.Quad4': 'pyformex.elements.Quad4',
'mesh.Mesh': 'pyformex.mesh.Mesh',
'formex.Formex': 'pyformex.formex.Formex'
}
[docs]def find_class(module, name):
"""Find a class whose name or module has changed"""
pf.debug(f"I want to import {name} from {module}", pf.DEBUG.PROJECT)
clas = f"{module}.{name}"
pf.debug(f"Object is {clas}", pf.DEBUG.PROJECT)
if clas in class_relocations:
module = class_relocations[clas]
lastdot = module.rfind('.')
module, name = module[:lastdot], module[lastdot+1:]
pf.debug(f" I will try {name} from module {module} instead",
pf.DEBUG.PROJECT)
elif module in module_relocations:
module = module_relocations[module]
pf.debug(f" I will try module {module} instead", pf.DEBUG.PROJECT)
__import__(module)
mod = sys.modules[module]
clas = getattr(mod, name)
pf.debug(f"Success: Got {clas.__class__.__name__}", pf.DEBUG.PROJECT)
return clas
[docs]class Unpickler(pickle.Unpickler):
"""Customized Unpickler class"""
def __init__(self, f, try_resolve=True):
"""Initialize the Unpickler"""
pickle.Unpickler.__init__(self, f, encoding='latin1')
self.try_resolve = try_resolve
if not try_resolve:
pf.debug("NOT TRYING TO RESOLVE RELOCATIONS: "
"YOU MAY GET INTO TROUBLE", pf.DEBUG.PROJECT)
[docs] def find_class(self, module, name):
pf.debug(f"FIND MODULE {module} NAME {name}", pf.DEBUG.PROJECT)
clas = pickle.Unpickler.find_class(self, module, name)
if not clas:
clas = find_class(module, name)
return clas
[docs]class Project(TrackedDict):
"""Project: a persistent storage of pyFormex data.
A pyFormex Project is a regular Python dict that can contain named data
of any kind, and can be saved to a file to create persistence over
different pyFormex sessions.
The :class:`Project` class is used by pyFormex for the ``pyformex.PF``
global variable that collects variables exported from pyFormex scripts.
While projects are mostly handled through the pyFormex GUI, notably the
*File* menu, the user may also create and handle his own Project objects
from a script.
Because of the way pyFormex Projects are written to file,
there may be problems when trying to read a project file that was
created with another pyFormex version. Problems may occur if the
project contains data of a class whose implementation has changed,
or whose definition has been relocated. Our policy is to provide
backwards compatibility: newer versions of pyFormex will normally
read the older project formats. Saving is always done in the
newest format, and these can generally not be read back by older
program versions (unless you are prepared to do some hacking).
.. warning:: Compatibility issues.
Occasionally you may run into problems when reading back an
old project file, especially when it was created by an unreleased
(development) version of pyFormex. Because pyFormex is evolving fast,
we can not test the full compatibility with every revision
You can file a support request on the pyFormex `support tracker`_.
and we will try to add the required conversion code to
pyFormex.
The project files are mainly intended as a means to easily save lots
of data of any kind and to restore them in the same session or a later
session, to pass them to another user (with the same or later pyFormex
version), to store them over a medium time period. Occasionally opening
and saving back your project files with newer pyFormex versions may help
to avoid read-back problems over longer time.
For a problemless long time storage of Geometry type objects you may
consider to write them to a pyFormex Geometry file (.pgf) instead, since
this uses a stable ascii based format. It can however (currently) only
store obects of class Geometry or one of its subclasses.
Parameters:
- `filename`: the name of the file where the Project data will be saved.
If the file exists (and `access` is not `w`), it should be a previously
saved Project and an attempt will be made to load the data from this
file into the Project.
If this fails, an error is raised.
If the file exists and `access` is `w`, it will be overwritten,
destroying any previous contents.
If no filename is specified, a temporary file will be created when
the Project is saved for the first time. The file with not be
automatically deleted. The generated name can be retrieved from the
filename attribute.
- `access`: One of 'wr' (default), 'rw', 'w' or 'r'.
If the string contains an 'r' the data from an existing file will be
read into the dict. If the string starts with an 'r', the file should
exist. If the string contains a 'w', the data can be written back to
the file. The 'r' access mode is thus a read-only mode.
====== =============== ============ ===================
access File must exist File is read File can be written
====== =============== ============ ===================
r yes yes no
rw yes yes yes
wr no if it exists yes
w no no yes
====== =============== ============ ===================
- `convert`: if True (default), and the file is opened for reading, an
attempt is made to open old projects in a compatibility mode, doing the
necessary conversions to new data formats. If convert is set False,
only the latest format can be read and older formats will generate
an error.
- `signature`: A text that will be written in the header record of the
file. This can e.g. be used to record format version info.
- `compression`: An integer from 0 to 9: compression level. For large
data sets, compression leads to much smaller files. 0 is no compression,
9 is maximal compression. The default is 4.
- `binary`: if False and no compression is used, storage is done
in an ASCII format, allowing to edit the file. Otherwise, storage
uses a binary format. Using binary=False is deprecated.
- `data`: a dict-like object to initialize the Project contents. These data
may override values read from the file.
Example:
>>> d = dict(a=1,b=2,c=3,d=[1,2,3],e={'f':4,'g':5})
>>> P = Project()
>>> P.update(d)
>>> print(P) # doctest: +ELLIPSIS
Project name: None
access: wr mode: b gzip: 5
signature: pyFormex ...
contents: ['a', 'b', 'c', 'd', 'e']
<BLANKLINE>
>>> print(utils.dictStr(P)) # doctest: +ELLIPSIS
{'a': 1, 'b': 2, 'c': 3, 'd': [1, 2, 3], 'e': ...}
>>> with utils.TempFile() as tmp:
... P.save(filename=tmp.path, quiet=True)
... P.clear()
... print(utils.dictStr(P))
... P.load(quiet=True)
{}
>>> print(utils.dictStr(P)) # doctest: +ELLIPSIS
{'a': 1, 'b': 2, 'c': 3, 'd': [1, 2, 3], 'e': ...}
"""
# Historically there have been a few different formats of the
# written Project files (.pyf). This variable holds the latest
# version.
latest_format = 3
def __init__(self, filename=None, access='wr', convert=True,
signature=_signature_, compression=5, binary=True,
data={}, protocol=default_protocol, **kargs):
"""Create a new project."""
if 'create' in kargs:
utils.warn("warn_project_create")
if 'legacy' in kargs:
utils.warn("warn_project_legacy")
self.filename = Path(filename) if filename else None
self.access = access
self.signature = str(signature)
self.gzip = compression if compression in range(1, 10) else 0
self.mode = 'b' if binary or compression > 0 else ''
if protocol is None:
protocol = default_protocol
self.protocol = min(protocol, pickle.HIGHEST_PROTOCOL) \
if self.mode == 'b' else 0
super().__init__()
if self.filename and self.filename.exists() and 'r' in self.access:
# read existing contents
self.load(try_resolve=convert)
self.hits = 0
if data:
self.update(data)
if self.filename and self.access=='w':
# destroy existing contents
self.filename.truncate()
pf.debug(f"Initial hits = {self.hits}", pf.DEBUG.PROJECT)
def __str__(self):
return f"""\
Project name: {self.filename}
access: {self.access} mode: {self.mode} gzip: {self.gzip}
signature: {self.signature}
contents: {self.contents()}
"""
def contents(self):
return sorted(self.keys())
[docs] def save(self, filename=None, quiet=False):
"""Save the project to file."""
if filename is not None:
self.filename = Path(filename)
if self.filename is None:
raise ValueError(
"No filename specified for the Project: can not save it")
if 'w' not in self.access:
pf.debug("Not saving because Project file opened readonly",
pf.DEBUG.PROJECT)
return
if not quiet:
print(f"Project variables changed: {self.hits}")
print(
f"Saving project {self.filename} with protocol {self.protocol}, "
f"mode {self.mode} and compression {self.gzip}")
with self.filename.open('w'+self.mode) as f:
# write header
header = f"{self.header_data()}\n".encode('utf-8')
f.write(header)
f.flush()
if self.gzip:
pyf = gzip.GzipFile(mode='w'+self.mode,
compresslevel=self.gzip, fileobj=f)
pickle.dump(self, pyf, self.protocol)
pyf.close()
else:
pickle.dump(self, f, self.protocol)
self.hits = 0
[docs] def load(self, filename=None, try_resolve=True, quiet=False):
"""Load a project from file.
The loaded definitions will update the current project.
"""
if filename is not None:
self.filename = Path(filename)
if self.filename is None:
raise ValueError(
"No filename specified for the Project: can not load")
if not quiet:
print(f"Reading project file: {self.filename}")
with self.filename.open('rb') as f:
f = self.readHeader(f, quiet)
if self.format < Project.latest_format:
if not quiet:
print(f"Format looks like {self.format}")
utils.warn('warn_old_project')
pos = f.tell()
if self.gzip > 0:
if not quiet:
print("Unpickling gzip")
pyf = gzip.GzipFile(fileobj=f, mode='rb')
p = Unpickler(pyf, try_resolve).load()
pyf.close()
else:
if not quiet:
print("Unpickling clear")
f.seek(pos)
p = Unpickler(f, try_resolve).load()
self.update(p)
[docs] def convert(self, filename=None):
"""Convert an old format project file.
The project file is read, and if successful, is immediately
saved. By default, this will overwrite the original file.
If a filename is specified, the converted data are saved to
that file.
In both cases, access is set to 'wr', so the tha saved data can
be read back immediately.
"""
self.load(try_resolve=True)
print(f"GOT KEYS {list(self.keys())}")
if filename is not None:
self.filename = Path(filename)
self.access = 'w'
print(f"Will now save to {self.filename}")
self.save()
[docs] def uncompress(self, verbose=True): # noqa: C901
"""Uncompress a compressed project file.
The project file is read, and if successful, is written
back in uncompressed format. This allows to make conversions
of the data inside.
"""
if self.filename is None:
return
if verbose:
print(f"Uncompressing project file: {self.filename}")
with self.filename.open('rb') as f:
f = self.readHeader(f)
if verbose:
print(self.format, self.gzip)
if self.gzip:
try:
pyf = gzip.GzipFile(self.filename, 'r', self.gzip, f)
except Exception:
self.gzip = 0
if self.gzip:
fn = self.filename.replace('.pyf', '_uncompressed.pyf')
fu = open(fn, 'w'+self.mode)
h = self.header_data()
h['gzip'] = 0
fu.write(f"{h}\n")
while True:
x = pyf.read()
if x:
fu.write(x)
else:
break
fu.close()
if verbose:
print(f"Uncompressed {self.filename} to {fn}")
else:
utils.warn("warn_project_compression")
[docs] def delete(self):
"""Unrecoverably delete the project file."""
if self.filename:
Path(self.filename).remove()
# End