diff options
Diffstat (limited to 'Python/_warnings.c')
| -rw-r--r-- | Python/_warnings.c | 93 | 
1 files changed, 80 insertions, 13 deletions
| diff --git a/Python/_warnings.c b/Python/_warnings.c index 6dff0a2..978bad1 100644 --- a/Python/_warnings.c +++ b/Python/_warnings.c @@ -513,6 +513,64 @@ warn_explicit(PyObject *category, PyObject *message,      return result;  /* Py_None or NULL. */  } +static int +is_internal_frame(PyFrameObject *frame) +{ +    static PyObject *importlib_string = NULL; +    static PyObject *bootstrap_string = NULL; +    PyObject *filename; +    int contains; + +    if (importlib_string == NULL) { +        importlib_string = PyUnicode_FromString("importlib"); +        if (importlib_string == NULL) { +            return 0; +        } + +        bootstrap_string = PyUnicode_FromString("_bootstrap"); +        if (bootstrap_string == NULL) { +            Py_DECREF(importlib_string); +            return 0; +        } +        Py_INCREF(importlib_string); +        Py_INCREF(bootstrap_string); +    } + +    if (frame == NULL || frame->f_code == NULL || +            frame->f_code->co_filename == NULL) { +        return 0; +    } +    filename = frame->f_code->co_filename; +    if (!PyUnicode_Check(filename)) { +        return 0; +    } +    contains = PyUnicode_Contains(filename, importlib_string); +    if (contains < 0) { +        return 0; +    } +    else if (contains > 0) { +        contains = PyUnicode_Contains(filename, bootstrap_string); +        if (contains < 0) { +            return 0; +        } +        else if (contains > 0) { +            return 1; +        } +    } + +    return 0; +} + +static PyFrameObject * +next_external_frame(PyFrameObject *frame) +{ +    do { +        frame = frame->f_back; +    } while (frame != NULL && is_internal_frame(frame)); + +    return frame; +} +  /* filename, module, and registry are new refs, globals is borrowed */  /* Returns 0 on error (no new refs), 1 on success */  static int @@ -523,8 +581,18 @@ setup_context(Py_ssize_t stack_level, PyObject **filename, int *lineno,      /* Setup globals and lineno. */      PyFrameObject *f = PyThreadState_GET()->frame; -    while (--stack_level > 0 && f != NULL) -        f = f->f_back; +    // Stack level comparisons to Python code is off by one as there is no +    // warnings-related stack level to avoid. +    if (stack_level <= 0 || is_internal_frame(f)) { +        while (--stack_level > 0 && f != NULL) { +            f = f->f_back; +        } +    } +    else { +        while (--stack_level > 0 && f != NULL) { +            f = next_external_frame(f); +        } +    }      if (f == NULL) {          globals = PyThreadState_Get()->interp->sysdict; @@ -580,13 +648,12 @@ setup_context(Py_ssize_t stack_level, PyObject **filename, int *lineno,          data = PyUnicode_DATA(*filename);  #define ascii_lower(c) ((c <= 127) ? Py_TOLOWER(c) : 0) -        /* if filename.lower().endswith((".pyc", ".pyo")): */ +        /* if filename.lower().endswith(".pyc"): */          if (len >= 4 &&              PyUnicode_READ(kind, data, len-4) == '.' &&              ascii_lower(PyUnicode_READ(kind, data, len-3)) == 'p' &&              ascii_lower(PyUnicode_READ(kind, data, len-2)) == 'y' && -            (ascii_lower(PyUnicode_READ(kind, data, len-1)) == 'c' || -                ascii_lower(PyUnicode_READ(kind, data, len-1)) == 'o')) +            ascii_lower(PyUnicode_READ(kind, data, len-1)) == 'c')          {              *filename = PyUnicode_Substring(*filename, 0,                                              PyUnicode_GET_LENGTH(*filename)-1); @@ -613,8 +680,7 @@ setup_context(Py_ssize_t stack_level, PyObject **filename, int *lineno,                      goto handle_error;                  }                  else if (!is_true) { -                    Py_DECREF(*filename); -                    *filename = PyUnicode_FromString("__main__"); +                    Py_SETREF(*filename, PyUnicode_FromString("__main__"));                      if (*filename == NULL)                          goto handle_error;                  } @@ -654,16 +720,17 @@ get_category(PyObject *message, PyObject *category)      if (rc == 1)          category = (PyObject*)message->ob_type; -    else if (category == NULL) +    else if (category == NULL || category == Py_None)          category = PyExc_UserWarning;      /* Validate category. */      rc = PyObject_IsSubclass(category, PyExc_Warning); -    if (rc == -1) -        return NULL; -    if (rc == 0) { -        PyErr_SetString(PyExc_ValueError, -                        "category is not a subclass of Warning"); +    /* category is not a subclass of PyExc_Warning or +       PyObject_IsSubclass raised an error */ +    if (rc == -1 || rc == 0) { +        PyErr_Format(PyExc_TypeError, +                     "category must be a Warning subclass, not '%s'", +                     Py_TYPE(category)->tp_name);          return NULL;      } | 
