ftp.nice.ch/Attic/openStep/unix/developer/language/python/python.14.m.s.tgz#/python-14-OS-mach/Modules/pyrl.c

This is pyrl.c in view mode; [Download] [Up]

/* Copyright (c) 1996 by Lele Gaifax.  All Rights Reserved
 *
 * This file is part of Nothing (yet).
 *
 * $RCSfile: pyrl.c,v $
 * $Revision: 1.1.1.1 $
 * $Date: 1997/01/18 03:52:15 $
 *
 * Created Tue Oct  1 12:04:57 1996.
 */

/* Lele Gaifax <lele@nautilus.eclipse.it> on 28 September 1996:

   I added a completer function for the readline library, bound on
   M-Tab (aka ESC-Tab). By doing a simple parse of the current line
   it's able to expand names of builtins, locals, members, methods... */

#include "Python.h"

#ifdef WITH_READLINE

#include <stdio.h>
#include <string.h>
#include <errno.h>

#include "myproto.h"
#include "mymalloc.h"
#include "intrcheck.h"

#include <readline/readline.h>
#include <readline/history.h>
#include <setjmp.h>
#include <signal.h>


/* AIX requires this to be the first thing in the file.  */
#ifdef __GNUC__
#ifndef alloca			/* predefined by NeXT's header */
# define alloca __builtin_alloca
#endif
#else
# if HAVE_ALLOCA_H
#  include <alloca.h>
# else
#  ifdef _AIX
#pragma alloca
#  else
#   ifndef alloca /* predefined by HP cc +Olibcalls */
char *alloca ();
#   endif
#  endif
# endif
#endif

static inline char *
strdup (const char *s)
{
  char *dup = malloc (strlen (s) + 1);


  return strcpy (dup, s);
}

static jmp_buf jbuf;

/* ARGSUSED */
static RETSIGTYPE
onintr(sig)
	int sig;
{
	longjmp(jbuf, sig);
}

/* Appends to LIST the ``papabile'' symbols in DICT, that may actually
   be a dictionary, a list, or an object responding to .keys().
   Returns the number of added symbols, or 0 on errors. */
static int
append_papabile_symbols (PyObject *list,
			 PyObject *dict,
			 const char *text,
			 int len)
{
  PyObject *keys;
  int count;

  if (PyDict_Check (dict))
    {
      keys = PyDict_Keys (dict);
      if (!keys)
	return 0;
      
      if (PyList_Sort (keys) != 0)
	{
	  Py_DECREF (keys);
	  return 0;
	}
    }
  else if (PyList_Check (dict))
    {
      keys = dict;
      Py_INCREF (keys);
    }
  else
    keys = PyObject_CallMethod (dict, "keys", NULL);

  count = 0;
  if (keys)
    {
      unsigned int idx = PyList_Size (keys);
      
      while (idx--)
	{
	  PyObject *localsym = PyList_GetItem (keys, idx);
	  
	  if (strncmp (text, PyString_AsString (localsym), len) == 0)
	    {
	      PyList_Append (list, localsym);
	      count++;
	    }
	}

      Py_DECREF (keys);
    }
  else
    PyErr_Clear();
      
  return count;
}

/* If STATUS==0, collects the symbols from __main__.__dict__ and from
   __builtins__.__dict__, then build a list of those beginning with
   TEXT.
   Returns the next symbol's name if there is one, NULL otherwise.
   
   This is called from the readline's completer function. */
static char *
try_complete_local_symbol (char *text, int state)
{
  static PyObject *symbols_list = NULL;
  static int symbols_list_idx;
  
  if (!state)
    {
      PyObject *symbols;
      int len;
      
      Py_XDECREF (symbols_list);
      symbols_list = NULL;
      symbols_list_idx = 0;

      symbols_list = PyList_New(0);
      if (!symbols_list)
	return NULL;

      len = strlen (text);
      
      symbols = PyObject_GetAttrString (PyImport_AddModule ("__main__"), "__dict__");
      if (symbols)
	{
	  symbols_list_idx += append_papabile_symbols (symbols_list,
						       symbols,
						       text,
						       len);
	  Py_DECREF (symbols);
	}
      else
	PyErr_Clear();

      symbols = PyObject_GetAttrString (PyEval_GetBuiltins(), "__dict__");
      if (symbols)
	{
	  symbols_list_idx += append_papabile_symbols (symbols_list,
						       symbols,
						       text,
						       len);
	  Py_DECREF (symbols);
	}
      else
	PyErr_Clear();
    }

  if (symbols_list_idx > 0)
    {
      PyObject *localsym = PyList_GetItem (symbols_list, --symbols_list_idx);
      
      return strdup (PyString_AsString (localsym));
    }

  return NULL;
}

/* If STATUS==0, tries to understand from the entered TEXT the object
   we are writing the name, and then collects in a list the name of the
   symbols in both ``__dict__'' and ``__members__'' slots of it.

   Returns the next symbol's name if there is one, NULL otherwise.
   
   This is called from the readline's completer function. */
static char *
try_complete_member_symbol (char *text, int state)
{
  static PyObject *symbols_list = NULL;
  static int symbols_list_idx;
  static int prefix_len;
  
  if (!state)
    {
      int len;
      PyObject *symbols = PyImport_AddModule ("__main__");
      PyObject *list;
      const char *point, *start;
      
      Py_XDECREF (symbols_list);
      symbols_list = NULL;
      symbols_list_idx = 0;

      symbols_list = PyList_New(0);
      if (!symbols_list)
	return NULL;

      /* PyImport_AddModule() does not return a new reference! */
      Py_INCREF (symbols);
      
      start = text;
      while ((point = strchr (start, '.')))
	{
	  int symlen = point-start;
	  char *symname;

	  if (!symlen)
	    return NULL;

	  symname = alloca (symlen+1);
	  strncpy (symname, start, symlen);
	  symname[symlen] = '\0';

	  /* This is safe: if this is the first time, we INCREFed the
	     module before the loop; if not, the previous GetAttrString()
	     returned a new reference to a surely-existing-somewhere object. */
	  Py_DECREF (symbols);
	  
	  symbols = PyObject_GetAttrString (symbols, symname);
	  if (!symbols)
	    {
	      PyErr_Clear();
	      return NULL;
	    }

	  start = point+1;
	}

      prefix_len = start-text;
      len = strlen (start);

      list = PyObject_GetAttrString (symbols, "__dict__");
      if (list)
	{
	  symbols_list_idx += append_papabile_symbols (symbols_list,
						       list,
						       start,
						       len);
	  Py_DECREF (list);
	}
      else
	PyErr_Clear();

      list = PyObject_GetAttrString (symbols, "__members__");
      if (list)
	{
	  symbols_list_idx += append_papabile_symbols (symbols_list,
						       list,
						       start,
						       len);
	  Py_DECREF (list);
	}
      else
	PyErr_Clear();

      list = PyObject_GetAttrString (symbols, "__methods__");
      if (list)
	{
	  symbols_list_idx += append_papabile_symbols (symbols_list,
						       list,
						       start,
						       len);
	  Py_DECREF (list);
	}
      else
	PyErr_Clear();
    }

  if (symbols_list_idx > 0)
    {
      PyObject *localsym = PyList_GetItem (symbols_list, --symbols_list_idx);
      extern char *strdup (const char *);
      const char *ls = PyString_AsString (localsym);
      char *complete;

      complete = alloca (prefix_len + PyString_Size (localsym) + 1);
      strncpy (complete, text, prefix_len);
      strcpy (complete+prefix_len, ls);
      return strdup (complete);
    }

  return NULL;
}

static inline char **
builtin_completion (char *text, int start, int end)
{
  if (end > start && strchr (text, '.'))
    return completion_matches (text, try_complete_member_symbol);
  else
    return completion_matches (text, try_complete_local_symbol);
}

static PyObject *python_completion_in_python;

/* If the current TEXT contains a point '.', assume we are accessing a
   member of an object and attempt to complete that; otherwise try to
   complete a local symbol. */
static char **
python_completion (char *text, int start, int end)
{
  /* Do not perform standard filename completion, even if we can't complete
     the symbol. */
  rl_attempted_completion_over = 1;

  if (python_completion_in_python)
    {
      PyTupleObject *args;
      PyObject *result;
      
      args = (PyTupleObject *) PyTuple_New (3);
      if (!args)
	return NULL;

      PyTuple_SET_ITEM (args, 0, PyString_FromString (text));
      PyTuple_SET_ITEM (args, 1, PyInt_FromLong (start));
      PyTuple_SET_ITEM (args, 2, PyInt_FromLong (end));

      result = PyEval_CallObject (python_completion_in_python, (PyObject *) args);

      Py_DECREF (args);

      if (result && PyList_Check (result))
	{
	  int midx = PyList_Size (result);
	  char **matches = malloc ((midx+1) * sizeof (char *));
	  
	  if (!matches)
	    {
	      PyErr_NoMemory();
	      return NULL;
	    }

	  matches[midx] = NULL;
	  while (midx--)
	    matches[midx] = strdup (PyString_AsString (PyList_GetItem (result, midx)));

	  Py_DECREF (result);

	  return matches;
	}
      else
	{
	  if (! result)
	    {
	      PyErr_Print();

	      Py_DECREF (python_completion_in_python);
	      python_completion_in_python = NULL;
	    }
	  else
	    Py_DECREF (result);
	  return NULL;
	}
    }
  else
    return builtin_completion (text, start, end);
}

static char *
py_readline (prompt)
	char *prompt;
{
  int n;
  char *p;
  RETSIGTYPE (*old_inthandler)();
  int sig;
  
#ifdef NeXT
  RETSIGTYPE (*old_conthandler)();

sigcont_received:
  old_conthandler = signal (SIGCONT, onintr);
#endif
  old_inthandler = signal (SIGINT, onintr);
  if ((sig = setjmp (jbuf)))
    {
#ifdef HAVE_SIGRELSE
      /* This seems necessary on SunOS 4.1 (Rasmus Hahn) */
      sigrelse(SIGINT);
#endif
      signal(SIGINT, old_inthandler);
#ifdef NeXT
      signal (SIGCONT, old_conthandler);
      if (sig == SIGCONT)
	goto sigcont_received;
#endif
      return NULL;
    }
  
  p = readline (prompt);
  signal (SIGINT, old_inthandler);
#ifdef NeXT
  signal (SIGCONT, old_conthandler);
#endif
  
  if (p == NULL)
    {
      p = malloc(1);
      if (p != NULL)
	*p = '\0';
      return p;
    }
  n = strlen (p);
  if (n > 0)
    add_history (p);
  if ((p = realloc (p, n+2)) != NULL)
    {
      p[n] = '\n';
      p[n+1] = '\0';
    }
  return p;
}

static char pyrl_set_completer_doc[] =
"Set a new completer function. The COMPLETER argument must be a callable\n\
object, and will be called with three arguments: the TEXT entered so far,\n\
START and END show the region of TEXT that contains the word to complete.\n\
Returns the previous completer object, or None if it was the builtin one.";

static PyObject *
pyrl_set_completer (PyObject *self, PyObject *args)
{
  PyObject *new;
  
  if (PyArg_ParseTuple (args, "O;completer", &new))
    {
      if (new == Py_None || PyCallable_Check (new))
	{
	  PyObject *old = python_completion_in_python;

	  if (new != Py_None)
	    {
	      python_completion_in_python = new;
	      Py_INCREF (python_completion_in_python);
	    }
	  else
	    python_completion_in_python = NULL;
	  
	  if (!old)
	    {
	      old = Py_None;
	      Py_INCREF (old);
	    }
	  
	  return old;
	}
      else
	PyErr_BadArgument();
    }

  return NULL;
}

static char pyrl_readline_doc[] =
"Read a line of input from the user with the GNU readline facility. If the\n\
optional PROMPT is not given, sys.ps1 is used instead.";

static PyObject *
pyrl_readline (PyObject *self, PyObject *args)
{
  char *prompt = NULL;

  if (PyArg_ParseTuple (args, "|s;prompt", &prompt))
    {
      char *line;
      
      if (prompt == NULL)
	{
	  PyObject *ps1 = PySys_GetObject ("ps1");

	  prompt = PyString_AsString (ps1);
	}

      line = py_readline (prompt);
      if (line)
	{
	  PyObject *ret = PyString_FromString (line);

	  free (line);
	  return ret;
	}
      else
	{
	  PyErr_NoMemory();
	  return NULL;
	}
    }

  return NULL;
}

static char pyrl_builtin_completer_doc[] = "The builtin completer function";
static PyObject *
pyrl_builtin_completer (PyObject *self, PyObject *args)
{
  char *text;
  int start, end;

  if (PyArg_ParseTuple (args, "sii", &text, &start, &end))
    {
      char **result = builtin_completion (text, start, end);

      if (result)
	{
	  PyObject *list;
	  int ncompl;
	  
	  for (ncompl=0; result[ncompl]; ncompl++)
	    /* do nothing */;

	  list = PyList_New (ncompl);
	  if (!list)
	    return NULL;

	  while (ncompl--)
	    {
	      PyObject *str = PyString_FromString (result[ncompl]);

	      PyList_SetItem (list, ncompl, str);
	      free (result[ncompl]);
	    }
	  free (result);
	  
	  return list;
	}
      else
	{
	  Py_INCREF (Py_None);
	  return Py_None;
	}
    }
  
  return NULL;
}    
	  
extern char * (*PyOS_Readline)();

#endif /* WITH_READLINE */

static PyMethodDef pyrl_methods[] =
{
#ifdef WITH_READLINE
  { "set_completer",	(PyCFunction) pyrl_set_completer, 	METH_VARARGS,	pyrl_set_completer_doc },
  { "builtin_completer",(PyCFunction) pyrl_builtin_completer, 	METH_VARARGS,	pyrl_builtin_completer_doc },
  { "readline",		(PyCFunction) pyrl_readline,		METH_VARARGS,	pyrl_readline_doc },
#endif
  { 0, 0, 0, 0 }
};

static char pyrl_doc[] =
#ifdef WITH_READLINE
"A replacement for the builtin readline implementation that uses GNU readline.";
#else
"A placehold empty module";
#endif

void initpyrl()
{
  static int been_here = 0;

  if (!been_here)
    {
      Py_InitModule4 ("pyrl", pyrl_methods, pyrl_doc, NULL, PYTHON_API_VERSION);

      been_here++;

#ifdef WITH_READLINE
      /* Force rebind of TAB to insert-tab */
      rl_bind_key ('\t', rl_insert);
      rl_bind_key_in_map ('\t', rl_complete, emacs_meta_keymap);
      rl_readline_name = "Python";
      rl_attempted_completion_function = (CPPFunction *) python_completion;
      rl_basic_word_break_characters = " \t\"'`<>=()[]{}";
      PyOS_Readline = py_readline;
#endif
    }
}

These are the contents of the former NiCE NeXT User Group NeXTSTEP/OpenStep software archive, currently hosted by Netfuture.ch.