Source code for tecplot.session.session

import atexit
import contextlib
import datetime
import logging
import platform
import textwrap

from ..tecutil import _tecutil_connector
from ..exception import *
from .state_changed import _emit_state_changes


log = logging.getLogger(__name__)


[docs] def stop(): """Releases the |Tecplot License| and shuts down |Tecplot Engine|. This shuts down the |Tecplot Engine| and releases the |Tecplot License|. Call this function when your script is finished using |PyTecplot|. Calling this function is not required. If you do not call this function, it will be called automatically when your script exists. However, the |Tecplot License| will not be released until you call this function. This method is silently ignored when connected to a running instance of |Tecplot 360| (see `tecplot.session.connect()`). Note that stop() may only be called once during the life of a Python session. If it has already been called, subsequent calls do nothing:: >>> # Shutdown tecplot and release license >>> tecplot.session.stop() See also: `tecplot.session.acquire_license()`, `tecplot.session.release_license()`. """ _tecutil_connector.stop()
[docs] def acquire_license(): """Attempts to acquire the |Tecplot License| Call this function to attempt to acquire a |Tecplot License|. If |Tecplot Engine| is not started, this function will start the |Tecplot Engine| before attempting to acquire a license. This function can be used to re-acquire a license that was released with `tecplot.session.release_license`. If the |Tecplot Engine| is currently running, and a |Tecplot License| has already been acquired, this function has no effect. Licenses may be acquired and released any number of times during the same Python session. Raises `TecplotLicenseError` if a valid license could not be acquired. .. note:: Warning emitted when close to expiration date. A warning is emitted using Python's built-in `warnings` module if you are within 30 days of your TecPLUS subscription expiration date. The message will look like this: .. code-block:: shell $ python >>> import tecplot >>> tecplot.session.acquire_license() /path/to/tecutil_connector.py:458: UserWarning: Your Tecplot software maintenance subscription (TecPLUS) will expire in **13 days**, after which you will no longer be able to use PyTecplot. Contact sales@tecplot.com to renew your TecPLUS subscription. warn(warning_msg) These warnings can be suppressed by using the ``-W ignore`` option when invoking the python interpreter: .. code-block:: shell $ python -W ignore >>> import tecplot >>> tecplot.session.acquire_license() See also: `tecplot.session.release_license()` Example:: >>> import tecplot >>> # Do useful things >>> tecplot.session.release_license() >>> # Do time-consuming things not related to |PyTecplot| >>> tecplot.session.acquire_license() # re-acquire the license >>> # Do useful |PyTecplot| related things. """ _tecutil_connector.acquire_license()
[docs] def release_license(): """Attempts to release the |Tecplot License| Call this to release a |Tecplot License|. Normally you do not need to call this function since `tecplot.session.stop()` will call it for you when your script exists and the Python interpreter is unloaded. This function can be used to release a license so that the license is available to other instances of |Tecplot 360|. If the |Tecplot License| has already been released, this function has no effect. Licenses may be acquired and released any number of times during the same Python session. This method is silently ignored when connected to a running instance of |Tecplot 360| (see `tecplot.session.connect()`). See also: `tecplot.session.acquire_license()` Example usage:: >>> import tecplot >>> # Do useful things >>> tecplot.session.release_license() >>> # Do time-consuming things not related to |PyTecplot| >>> tecplot.session.acquire_license() # re-acquire the license >>> # Do useful |PyTecplot| related things. """ _tecutil_connector.release_license()
[docs] def start_roaming(days): """Check out a roaming license. Parameters: days (`int`): Number of days to roam. This will acquire a PyTecplot license and then attempt to set it up for roaming. The maximum number of days one may roam is typically 90. This function can be called in an interactive terminal and will affect all subsequent uses of PyTecplot on the local machine. Do not forget to `release <tecplot.session.stop_roaming>` the roaming license to the server if you are finished roaming before the expiration date. See also: `tecplot.session.stop_roaming()` Example usage where ``YYYY-MM-DD`` will be the date 10 days from when this code was executed:: >>> tecplot.session.start_roaming(10) You have successfully checked out a roaming license of PyTecplot. This will be valid for 10 days until midnight of YYY-MM-DD. >>> tecplot.session.stop_roaming() Your PyTecplot roaming license has been checked in. """ if _tecutil_connector.connected: msg = textwrap.dedent('''\ Roaming PyTecplot is not available when connected. To roam PyTecplot, first disconnect with tecplot.session.disconnect(). Roaming with Tecplot 360 is available under Help -> License Roaming. ''') raise TecplotLogicError(msg) _tecutil_connector.start_roaming(days)
[docs] def stop_roaming(force=False): """Check in (release) a roaming license. This will check in and make available to others on the network a license that you previously checked out for roaming. See also: `tecplot.session.start_roaming()` Example usage:: >>> tecplot.session.stop_roaming() Your PyTecplot roaming license has been checked in. """ if _tecutil_connector.connected: msg = textwrap.dedent('''\ Roaming PyTecplot is not available when connected. To roam PyTecplot, first disconnect with tecplot.session.disconnect(). Roaming with Tecplot 360 is available under Help -> License Roaming. ''') raise TecplotLogicError(msg) _tecutil_connector.stop_roaming(force)
[docs] def license_expiration(): """Expiration date of the current license. Returns: `datetime.date` Example usage:: >>> print(tecplot.session.license_expiration()) 1955-11-05 """ expire = _tecutil_connector.license_expiration if isinstance(expire, datetime.date): return expire
[docs] def connect(host='localhost', port=7600, timeout=10, quiet=False): """Connect this PyTecplot to a running instance of Tecplot 360. Parameters: host (`str`, optional): The host name or IP address of the machine running Tecplot 360 with the :ref:`TecUtil Server <TecUtilServer>` addon loaded and listening. (default: localhost) port (`int`, optional): The port used by the running Tecplot 360 instance. (default: 7600) timeout (`int`, optional): Number of seconds to wait before giving up. (default: 10) quiet (`bool`, optional): Suppress status messages sent to the console. Exception messages will still be presented on errors. (default: `False`) This will connect the running python script to Tecplot 360, sending requests over the network. The :ref:`TecUtil Server <TecUtilServer>` addon must be loaded and the server must be accepting requests. To turn on the server in Tecplot 360, go to the main menu, click on "Scripting -> PyTecplot Connections...", and finally check the option to "Accept connections." Make sure the same port is used in both Tecplot 360 and the python script. For more information, see :ref:`Requirements for Connecting to Tecplot 360 GUI <connections>` Example usage:: >>> tecplot.session.connect(port=7600) Connecting to Tecplot 360 TecUtil Server on: tcp://localhost:7600 Connection established. To activate the :ref:`TecUtil Server <TecUtilServer>` addon in Tecplot 360 on start-up, first create a macro file, named something like: *startTecUtilServer.mcr*, with the following content: .. code-block:: none #!MC 1410 $!EXTENDEDCOMMAND COMMANDPROCESSORID = "TecUtilServer" COMMAND = R"( AcceptRequests = Yes ListenOnAddress = localhost ListenOnPort = 7600 )" Then run Tecplot 360 from a command console with this file as one of the arguments:: > tec360 startTecUtilServer.mcr .. warning:: Adding the macro command above may cause a port binding conflict when using multiple instances of Tecplot 360. A single port can only be bound to one instance of the :ref:`TecUtil Server <TecUtilServer>`. You may still add this to the ``tecplot.add`` file which will be run everytime, but it must only be activated when Tecplot 360 is running interactively, i.e. not in batch mode. To do this, check the value of ``|INBATCHMODE|``: .. code-block:: none $!IF |INBATCHMODE| == 0 $!EXTENDEDCOMMAND COMMANDPROCESSORID = "TecUtilServer" COMMAND = R"( AcceptRequests = Yes ListenOnAddress = localhost ListenOnPort = 7600 )" $!ENDIF .. versionadded:: 2017.3 PyTecplot connections requires Tecplot 360 2017 R3 or later. """ _tecutil_connector.connect(host, port, timeout, quiet)
[docs] def connected(timeout=5): """Check if PyTecplot is connected to a running instance of Tecplot 360. This method sends a handshake message to the :ref:`TecUtil Server <TecUtilServer>` and waits for a successful reply, timing out in the number of seconds specified. Parameters: timeout (`int`, optional): Number of seconds to wait before giving up. (default: 5) .. versionadded:: 2017.3 of Tecplot 360 PyTecplot connections requires Tecplot 360 2017 R3 or later. """ if _tecutil_connector.connected: return _tecutil_connector.client.is_server_listening(timeout) else: return False
[docs] def disconnect(quit=False): """Disconnect from a running instance of Tecplot 360. Parameters: quit (`bool`, optional): Attempt to quit and close the instance of Tecplot 360 before disconnecting. .. versionadded:: 2017.3 PyTecplot connections requires Tecplot 360 2017 R3 or later. """ if connected(): if quit: _tecutil_connector.client.quit() _tecutil_connector.disconnect()
[docs] @contextlib.contextmanager def suspend(): """Suspend the Tecplot engine and graphical interface. This context may speed up several types of operations including the creation of zones, filling or alterating the underlying data and the setup of complex styles. It will put Tecplot 360 into a "suspended" state such that the engine (in batch mode) or the graphical interface (in connected mode) will not try to keep up with the operations issued from Python. Upon exit of this context, the Tecplot 360 will be notified of any data alterations that have occured and the interface will be updated accordingly. See the `state change` section for more information about how the |Tecplot 360| is updated when style or data is changed from a PyTecplot script. Example usage where `data` is some user-provided data -- see the examples under ``pytecplot/examples/working_with_datasets`` folder within the Teplot 360 installation for more information.:: fr = tp.active_frame() with tp.session.suspend(): ds = fr.create_dataset('Data', ['x', 'y', 'z']) zn = ds.add_ordered_zone('Zone', (10, 10, 10)) zn.values('x')[:] = data[0] zn.values('y')[:] = data[1] zn.values('z')[:] = data[2] fr.plot_type = tp.constant.PlotType.Cartesian3D .. versionadded:: 2018.2 The suspend context, when used in connected mode, requires Tecplot 360 2018 R2 or later to realize the full performance benefits though this will still provide some performance improvements with older versions of Tecplot 360. """ if _tecutil_connector.suspended: yield else: _tecutil_connector.suspended = True _tecutil_connector._state_changes = {} _tecutil_connector._delete_caches = [] try: if _tecutil_connector.connected: with _tecutil_connector.client.suspend_interface(): yield _emit_state_changes(_tecutil_connector._state_changes) else: yield _emit_state_changes(_tecutil_connector._state_changes) finally: for delete_cache in _tecutil_connector._delete_caches: delete_cache() del _tecutil_connector._state_changes _tecutil_connector.suspended = False
[docs] def suspend_enter(): """Free-function equivalent to entering the `suspend` context. This is an example of calling *a previously defined function* ``do_work()`` within a suspend context, using ``try/finally`` to ensure the context is properly cleared: >>> try: >>> tp.session.suspend_enter() >>> do_work() >>> finally: >>> tp.session.suspend_exit() """ if not hasattr(_tecutil_connector, 'suspend_context'): _tecutil_connector.suspend_context = suspend() _tecutil_connector.suspend_context.__enter__()
[docs] def suspend_exit(): """Free-function equivalent to exiting the `suspend` context. This must only be used following a call to `tecplot.session.suspend_enter()` and cannot be used within a `tecplot.session.suspend()` context block. """ if not hasattr(_tecutil_connector, 'suspend_context'): raise TecplotLogicError('not in suspend context') _tecutil_connector.suspend_context.__exit__(None, None, None) delattr(_tecutil_connector, 'suspend_context')
[docs] def clear_suspend(): """Break out of suspended mode when connected to Tecplot 360. Forcibly clears the suspend state of the Tecplot 360 interface. This will cause the :ref:`TecUtil Server <TecUtilServer>` to break out of a suspended state which may have been the result of a script not properly exiting from a `suspend()` context, possibly due to a segmentation fault. Example usage:: tecplot.session.connect() tecplot.session.clear_suspend() """ assert not _tecutil_connector.suspended, 'Can not clear in suspend context' assert _tecutil_connector.connected, 'Not connected to TecUtil Server' _tecutil_connector.client.clear_suspend_interface()
# Ensure the Tecplot engine is cleaned up on exit if started. atexit.register(stop)