# MODULE: grass.jupyter.setup
#
# AUTHOR(S): Caitlin Haedrich <caitlin DOT haedrich AT gmail>
# Vaclav Petras <wenzeslaus gmail com>
#
# PURPOSE: This module contains functions for launching a GRASS session
# in Jupyter Notebooks
#
# COPYRIGHT: (C) 2021-2022 Caitlin Haedrich, and by the GRASS Development Team
#
# This program is free software under the GNU General Public
# License (>=v2). Read the file COPYING that comes with GRASS
# for details.
"""Initialization GRASS GIS session and its finalization"""
import os
import weakref
from pathlib import Path
import grass.script as gs
def _set_notebook_defaults():
"""Set defaults appropriate for Jupyter Notebooks.
This function sets several GRASS environment variables that are
important for GRASS to run smoothly in Jupyter.
It also allows GRASS to overwrite existing maps of the same name.
"""
# We want functions to raise exceptions and see standard output of
# the modules in the notebook.
gs.set_raise_on_error(True)
gs.set_capture_stderr(True)
# Allow overwrite of existing maps
os.environ["GRASS_OVERWRITE"] = "1"
class _JupyterGlobalSession:
"""Represents a global GRASS session for Jupyter Notebooks.
Do not create objects of this class directly. Use the standalone *init* function
and an object will be returned to you, e.g.:
>>> import grass.jupyter as gj
>>> session = gj.init(...)
An object ends the session when it is destroyed or when the *finish* method is
called explicitly.
Notably, only the mapset is closed, but the libraries and GRASS modules
remain on path.
"""
def __init__(self):
self._finalizer = weakref.finalize(self, gs.setup.finish)
def switch_mapset(self, path, location=None, mapset=None):
"""Switch to a mapset provided as a name or path.
The mapset can be provided as a name, as a path,
or as database, location, and mapset.
Specifically, the *path* positional-only parameter can be either
name of a mapset in the current location or a full path to a mapset.
When location and mapset are provided using the additional parameters,
the *path* parameter is path to a database.
Raises ValueError if the mapset does not exist (e.g., when the name is
misspelled or the mapset is invalid).
"""
# The method could be a function, but this is more general (would work even for
# a non-global session).
# pylint: disable=no-self-use
# Functions needed only here.
# pylint: disable=import-outside-toplevel
from grass.grassdb.checks import (
get_mapset_invalid_reason,
is_mapset_valid,
mapset_exists,
)
from grass.grassdb.manage import resolve_mapset_path
# For only one parameter, try if it is a mapset in the current location to
# support switching only by its name.
gisenv = gs.gisenv()
if (
not location
and not mapset
and len(Path(path).parts) == 1
and mapset_exists(
path=gisenv["GISDBASE"], location=gisenv["LOCATION_NAME"], mapset=path
)
):
gs.run_command("g.gisenv", set=f"MAPSET={path}")
return
mapset_path = resolve_mapset_path(path=path, location=location, mapset=mapset)
if not is_mapset_valid(mapset_path):
raise ValueError(
_("Mapset {path} is not valid: {reason}").format(
path=mapset_path.path,
reason=get_mapset_invalid_reason(
mapset_path.directory, mapset_path.location, mapset_path.mapset
),
)
)
# This requires direct session file modification using g.gisenv because
# g.mapset locks the mapset which is not how init and finish behave.
# For code simplicity, we just change all even when only mapset is changed.
gs.run_command("g.gisenv", set=f"GISDBASE={mapset_path.directory}")
gs.run_command("g.gisenv", set=f"LOCATION_NAME={mapset_path.location}")
gs.run_command("g.gisenv", set=f"MAPSET={mapset_path.mapset}")
def finish(self):
"""Close the session, i.e., close the open mapset.
Subsequent calls to GRASS GIS modules will fail because there will be
no current (open) mapset anymore.
The finish procedure is done automatically when process finishes or the object
is destroyed.
"""
self._finalizer()
@property
def active(self):
"""True unless the session was finalized (e.g., with the *finish* function)"""
return self._finalizer.alive
# Pylint 2.12.2 identifies this a constant (although it is not), so it wants uppercase.
_global_session_handle = None # pylint: disable=invalid-name
[docs]def init(path, location=None, mapset=None, grass_path=None):
"""Initiates a GRASS session and sets GRASS environment variables.
Calling this function returns an object which represents the session.
>>> import grass.jupyter as gj
>>> session = gj.init(...)
The session is ended when `session.finish` is called or when the object is
destroyed when kernel ends or restarts. This function returns a copy of an
internally kept reference, so the return value can be safely ignored when not
needed.
The returned object can be used to switch to another mapset:
>>> session.switch_mapset("mapset_name")
Subsequent calls to the *init* function result in switching the mapset if
a session is active and result in creation of new session if it is not active.
On the other hand, if you see ``GISRC - variable not set`` after calling
a GRASS module, you know you don't have an active GRASS session.
:param str path: path to GRASS mapset or database
:param str location: name of GRASS location within the database
:param str mapset: name of mapset within location
"""
global _global_session_handle # pylint: disable=global-statement,invalid-name
if not _global_session_handle or not _global_session_handle.active:
# Create a GRASS session.
gs.setup.init(path, location=location, mapset=mapset, grass_path=grass_path)
# Set defaults for environmental variables and library.
_set_notebook_defaults()
_global_session_handle = _JupyterGlobalSession()
else:
_global_session_handle.switch_mapset(path, location=location, mapset=mapset)
return _global_session_handle