csomh / source-git / rpm

Forked from source-git/rpm 4 years ago
Clone
2ff057
#include "rpmsystem-py.h"
2ff057
2ff057
#include <rpm/rpmtypes.h>
2ff057
#include <rpm/rpmstring.h>
2ff057
#include <rpm/rpmlib.h>		/* rpmvercmp */
2ff057
2ff057
#include "header-py.h"
2ff057
#include "rpmds-py.h"
2ff057
#include "rpmstrpool-py.h"
2ff057
2ff057
struct rpmdsObject_s {
2ff057
    PyObject_HEAD
2ff057
    PyObject *md_dict;		/*!< to look like PyModuleObject */
2ff057
    int		active;
2ff057
    rpmds	ds;
2ff057
};
2ff057
2ff057
static PyObject *
2ff057
rpmds_Count(rpmdsObject * s)
2ff057
{
2ff057
    DEPRECATED_METHOD("use len(ds) instead");
2ff057
    return Py_BuildValue("i", PyMapping_Size((PyObject *)s));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_Ix(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("i", rpmdsIx(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_DNEVR(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("s", rpmdsDNEVR(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_N(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("s", rpmdsN(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_EVR(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("s", rpmdsEVR(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_Flags(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("i", rpmdsFlags(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_TagN(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("i", rpmdsTagN(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_Color(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("i", rpmdsColor(s->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_iternext(rpmdsObject * s)
2ff057
{
2ff057
    PyObject * result = NULL;
2ff057
2ff057
    /* Reset loop indices on 1st entry. */
2ff057
    if (!s->active) {
2ff057
	s->ds = rpmdsInit(s->ds);
2ff057
	s->active = 1;
2ff057
    }
2ff057
2ff057
    /* If more to do, return a (N, EVR, Flags) tuple. */
2ff057
    if (rpmdsNext(s->ds) >= 0) {
2ff057
	result = rpmds_Wrap(Py_TYPE(s), rpmdsCurrent(s->ds));
2ff057
    } else
2ff057
	s->active = 0;
2ff057
2ff057
    return result;
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_SetNoPromote(rpmdsObject * s, PyObject * args, PyObject * kwds)
2ff057
{
2ff057
    int nopromote;
2ff057
    char * kwlist[] = {"noPromote", NULL};
2ff057
2ff057
    if (!PyArg_ParseTupleAndKeywords(args, kwds, "i:SetNoPromote", kwlist,
2ff057
	    &nopromote))
2ff057
	return NULL;
2ff057
2ff057
    return Py_BuildValue("i", rpmdsSetNoPromote(s->ds, nopromote));
2ff057
}
2ff057
2ff057
/* XXX rpmdsFind uses bsearch on s->ds, so a sort is needed. */
2ff057
static PyObject *
2ff057
rpmds_Sort(rpmdsObject * s)
2ff057
{
2ff057
    /* XXX sort on (N,EVR,F) here. */
2ff057
    Py_RETURN_NONE;
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_Find(rpmdsObject * s, PyObject * arg)
2ff057
{
2ff057
    rpmdsObject * o;
2ff057
2ff057
    if (!PyArg_Parse(arg, "O!:Find", &rpmds_Type, &o))
2ff057
	return NULL;
2ff057
2ff057
    /* XXX make sure ods index is valid, real fix in lib/rpmds.c. */
2ff057
    if (rpmdsIx(o->ds) == -1)	rpmdsSetIx(o->ds, 0);
2ff057
2ff057
    return Py_BuildValue("i", rpmdsFind(s->ds, o->ds));
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_Merge(rpmdsObject * s, PyObject * arg)
2ff057
{
2ff057
    rpmdsObject * o;
2ff057
2ff057
    if (!PyArg_Parse(arg, "O!:Merge", &rpmds_Type, &o))
2ff057
	return NULL;
2ff057
2ff057
    return Py_BuildValue("i", rpmdsMerge(&s->ds, o->ds));
2ff057
}
2ff057
static PyObject *
2ff057
rpmds_Search(rpmdsObject * s, PyObject * arg)
2ff057
{
2ff057
    rpmdsObject * o;
2ff057
2ff057
    if (!PyArg_Parse(arg, "O!:Merge", &rpmds_Type, &o))
2ff057
        return NULL;
2ff057
2ff057
    return Py_BuildValue("i", rpmdsSearch(s->ds, o->ds));
2ff057
}
2ff057
2ff057
static PyObject *rpmds_Compare(rpmdsObject * s, PyObject * o)
2ff057
{
2ff057
    rpmdsObject * ods;
2ff057
2ff057
    if (!PyArg_Parse(o, "O!:Compare", &rpmds_Type, &ods))
2ff057
	return NULL;
2ff057
2ff057
    return PyBool_FromLong(rpmdsCompare(s->ds, ods->ds));
2ff057
}
2ff057
2ff057
static PyObject *rpmds_Instance(rpmdsObject * s)
2ff057
{
2ff057
    return Py_BuildValue("i", rpmdsInstance(s->ds));
2ff057
}
2ff057
2ff057
static PyObject * rpmds_Rpmlib(rpmdsObject * s, PyObject *args, PyObject *kwds)
2ff057
{
2ff057
    rpmstrPool pool = NULL;
2ff057
    rpmds ds = NULL;
2ff057
    char * kwlist[] = {"pool", NULL};
2ff057
2ff057
    if (!PyArg_ParseTupleAndKeywords(args, kwds, "|O&:rpmds_Rpmlib", kwlist, 
2ff057
		 &poolFromPyObject, &pool))
2ff057
	return NULL;
2ff057
2ff057
    /* XXX check return code, permit arg (NULL uses system default). */
2ff057
    rpmdsRpmlibPool(pool, &ds, NULL);
2ff057
2ff057
    return rpmds_Wrap(&rpmds_Type, ds);
2ff057
}
2ff057
2ff057
static struct PyMethodDef rpmds_methods[] = {
2ff057
 {"Count",	(PyCFunction)rpmds_Count,	METH_NOARGS,
2ff057
	"Deprecated, use len(ds) instead.\n" },
2ff057
 {"Ix",		(PyCFunction)rpmds_Ix,		METH_NOARGS,
2ff057
	"ds.Ix -> Ix -- Return current element index.\n" },
2ff057
 {"DNEVR",	(PyCFunction)rpmds_DNEVR,	METH_NOARGS,
2ff057
	"ds.DNEVR -> DNEVR -- Return current DNEVR.\n" },
2ff057
 {"N",		(PyCFunction)rpmds_N,		METH_NOARGS,
2ff057
	"ds.N -> N -- Return current N.\n" },
2ff057
 {"EVR",	(PyCFunction)rpmds_EVR,		METH_NOARGS,
2ff057
	"ds.EVR -> EVR -- Return current EVR.\n" },
2ff057
 {"Flags",	(PyCFunction)rpmds_Flags,	METH_NOARGS,
2ff057
	"ds.Flags -> Flags -- Return current Flags.\n" },
2ff057
 {"TagN",	(PyCFunction)rpmds_TagN,	METH_NOARGS,
2ff057
  "ds.TagN -> TagN -- Return TagN (RPMTAG_*NAME)\n\n"
2ff057
  "the type of all dependencies in this set.\n" },
2ff057
 {"Color",	(PyCFunction)rpmds_Color,	METH_NOARGS,
2ff057
	"ds.Color -> Color -- Return current Color.\n" },
2ff057
 {"SetNoPromote",(PyCFunction)rpmds_SetNoPromote, METH_VARARGS|METH_KEYWORDS,
2ff057
  "ds.SetNoPromote(noPromote) -- Set noPromote for this instance.\n\n"
2ff057
  "If True non existing epochs are no longer equal to an epoch of 0."},
2ff057
 {"Sort",	(PyCFunction)rpmds_Sort,	METH_NOARGS,
2ff057
	NULL},
2ff057
 {"Find",	(PyCFunction)rpmds_Find,	METH_O,
2ff057
  "ds.find(other_ds) -- Return index of other_ds in ds"},
2ff057
 {"Merge",	(PyCFunction)rpmds_Merge,	METH_O,
2ff057
	NULL},
2ff057
 {"Search",     (PyCFunction)rpmds_Search,      METH_O,
2ff057
"ds.Search(element) -> matching ds index (-1 on failure)\n\
2ff057
Check that element dependency range overlaps some member of ds.\n\
2ff057
The current index in ds is positioned at overlapping member." },
2ff057
 {"Rpmlib",     (PyCFunction)rpmds_Rpmlib,      METH_VARARGS|METH_KEYWORDS|METH_STATIC,
2ff057
	"ds.Rpmlib -> nds -- Return internal rpmlib dependency set.\n"},
2ff057
 {"Compare",	(PyCFunction)rpmds_Compare,	METH_O,
2ff057
  "ds.compare(other) -- Compare current entries of self and other.\n\nReturns True if the entries match each other, False otherwise"},
2ff057
 {"Instance",	(PyCFunction)rpmds_Instance,	METH_NOARGS,
2ff057
  "ds.Instance() -- Return rpmdb key of corresponding package or 0."},
2ff057
 {NULL,		NULL}		/* sentinel */
2ff057
};
2ff057
2ff057
/* ---------- */
2ff057
2ff057
static void
2ff057
rpmds_dealloc(rpmdsObject * s)
2ff057
{
2ff057
    s->ds = rpmdsFree(s->ds);
2ff057
    Py_TYPE(s)->tp_free((PyObject *)s);
2ff057
}
2ff057
2ff057
static Py_ssize_t rpmds_length(rpmdsObject * s)
2ff057
{
2ff057
    return rpmdsCount(s->ds);
2ff057
}
2ff057
2ff057
static PyObject *
2ff057
rpmds_subscript(rpmdsObject * s, PyObject * key)
2ff057
{
2ff057
    int ix;
2ff057
2ff057
    if (!PyInt_Check(key)) {
2ff057
	PyErr_SetString(PyExc_TypeError, "integer expected");
2ff057
	return NULL;
2ff057
    }
2ff057
2ff057
    ix = (int) PyInt_AsLong(key);
2ff057
    rpmdsSetIx(s->ds, ix);
2ff057
    return Py_BuildValue("s", rpmdsDNEVR(s->ds));
2ff057
}
2ff057
2ff057
static PyMappingMethods rpmds_as_mapping = {
2ff057
        (lenfunc) rpmds_length,		/* mp_length */
2ff057
        (binaryfunc) rpmds_subscript,	/* mp_subscript */
2ff057
        (objobjargproc)0,		/* mp_ass_subscript */
2ff057
};
2ff057
2ff057
static int rpmds_init(rpmdsObject * s, PyObject *args, PyObject *kwds)
2ff057
{
2ff057
    s->active = 0;
2ff057
    return 0;
2ff057
}
2ff057
2ff057
static int depflags(PyObject *o, rpmsenseFlags *senseFlags)
2ff057
{
2ff057
    int ok = 0;
2ff057
    PyObject *str = NULL;
2ff057
    rpmsenseFlags flags = RPMSENSE_ANY;
2ff057
2ff057
    if (PyInt_Check(o)) {
2ff057
	ok = 1;
2ff057
	flags = PyInt_AsLong(o);
2ff057
    } else if (utf8FromPyObject(o, &str)) {
2ff057
	ok = 1;
2ff057
	for (const char *s = PyBytes_AsString(str); *s; s++) {
2ff057
	    switch (*s) {
2ff057
	    case '=':
2ff057
		flags |= RPMSENSE_EQUAL;
2ff057
		break;
2ff057
	    case '<':
2ff057
		flags |= RPMSENSE_LESS;
2ff057
		break;
2ff057
	    case '>':
2ff057
		flags |= RPMSENSE_GREATER;
2ff057
		break;
2ff057
	    default:
2ff057
		ok = 0;
2ff057
		break;
2ff057
	    }
2ff057
	}
2ff057
	Py_DECREF(str);
2ff057
    }
2ff057
2ff057
    if (flags == (RPMSENSE_EQUAL|RPMSENSE_LESS|RPMSENSE_GREATER))
2ff057
	ok = 0;
2ff057
2ff057
    if (ok)
2ff057
	*senseFlags = flags;
2ff057
2ff057
    return ok;
2ff057
}
2ff057
2ff057
static PyObject * rpmds_new(PyTypeObject * subtype, PyObject *args, PyObject *kwds)
2ff057
{
2ff057
    PyObject *obj;
2ff057
    rpmTagVal tagN = RPMTAG_REQUIRENAME;
2ff057
    rpmds ds = NULL;
2ff057
    Header h = NULL;
2ff057
    rpmstrPool pool = NULL;
2ff057
    char * kwlist[] = {"obj", "tag", "pool", NULL};
2ff057
2ff057
    if (!PyArg_ParseTupleAndKeywords(args, kwds, "OO&|O&:rpmds_new", kwlist, 
2ff057
	    	 &obj, tagNumFromPyObject, &tagN,
2ff057
		 &poolFromPyObject, &pool))
2ff057
	return NULL;
2ff057
2ff057
    if (PyTuple_Check(obj)) {
2ff057
	const char *name = NULL;
2ff057
	const char *evr = NULL;
2ff057
	rpmsenseFlags flags = RPMSENSE_ANY;
2ff057
	/* TODO: if flags are specified, evr should be required too */
2ff057
	if (PyArg_ParseTuple(obj, "s|O&s", &name, depflags, &flags, &evr)) {
2ff057
	    ds = rpmdsSinglePool(pool, tagN, name, evr, flags);
2ff057
	} else {
2ff057
	    PyErr_SetString(PyExc_ValueError, "invalid dependency tuple");
2ff057
	    return NULL;
2ff057
	}
2ff057
    } else if (hdrFromPyObject(obj, &h)) {
2ff057
	if (tagN == RPMTAG_NEVR) {
2ff057
	    ds = rpmdsThisPool(pool, h, RPMTAG_PROVIDENAME, RPMSENSE_EQUAL);
2ff057
	} else {
2ff057
	    ds = rpmdsNewPool(pool, h, tagN, 0);
2ff057
	}
2ff057
    } else {
2ff057
	PyErr_SetString(PyExc_TypeError, "header or tuple expected");
2ff057
	return NULL;
2ff057
    }
2ff057
    
2ff057
    return rpmds_Wrap(subtype, ds);
2ff057
}
2ff057
2ff057
static char rpmds_doc[] =
2ff057
    "rpm.ds (dependendcy set) gives a more convenient access to dependencies\n\n"
2ff057
    "It can hold multiple entries of Name Flags and EVR.\n"
2ff057
    "It typically represents all dependencies of one kind of a package\n"
2ff057
    "e.g. all Requires or all Conflicts.\n"
2ff057
    ;
2ff057
2ff057
PyTypeObject rpmds_Type = {
2ff057
	PyVarObject_HEAD_INIT(&PyType_Type, 0)
2ff057
	"rpm.ds",			/* tp_name */
2ff057
	sizeof(rpmdsObject),		/* tp_basicsize */
2ff057
	0,				/* tp_itemsize */
2ff057
	/* methods */
2ff057
	(destructor) rpmds_dealloc,	/* tp_dealloc */
2ff057
	0,				/* tp_print */
2ff057
	(getattrfunc)0,			/* tp_getattr */
2ff057
	(setattrfunc)0,			/* tp_setattr */
2ff057
	0,				/* tp_compare */
2ff057
	(reprfunc)0,			/* tp_repr */
2ff057
	0,				/* tp_as_number */
2ff057
	0,				/* tp_as_sequence */
2ff057
	&rpmds_as_mapping,		/* tp_as_mapping */
2ff057
	(hashfunc)0,			/* tp_hash */
2ff057
	(ternaryfunc)0,			/* tp_call */
2ff057
	(reprfunc)0,			/* tp_str */
2ff057
	PyObject_GenericGetAttr,	/* tp_getattro */
2ff057
	PyObject_GenericSetAttr,	/* tp_setattro */
2ff057
	0,				/* tp_as_buffer */
2ff057
	Py_TPFLAGS_DEFAULT | Py_TPFLAGS_BASETYPE, /* tp_flags */
2ff057
	rpmds_doc,			/* tp_doc */
2ff057
	0,				/* tp_traverse */
2ff057
	0,				/* tp_clear */
2ff057
	0,				/* tp_richcompare */
2ff057
	0,				/* tp_weaklistoffset */
2ff057
	PyObject_SelfIter,		/* tp_iter */
2ff057
	(iternextfunc) rpmds_iternext,	/* tp_iternext */
2ff057
	rpmds_methods,			/* tp_methods */
2ff057
	0,				/* tp_members */
2ff057
	0,				/* tp_getset */
2ff057
	0,				/* tp_base */
2ff057
	0,				/* tp_dict */
2ff057
	0,				/* tp_descr_get */
2ff057
	0,				/* tp_descr_set */
2ff057
	0,				/* tp_dictoffset */
2ff057
	(initproc) rpmds_init,		/* tp_init */
2ff057
	0,				/* tp_alloc */
2ff057
	(newfunc) rpmds_new,		/* tp_new */
2ff057
	0,				/* tp_free */
2ff057
	0,				/* tp_is_gc */
2ff057
};
2ff057
2ff057
/* ---------- */
2ff057
2ff057
rpmds dsFromDs(rpmdsObject * s)
2ff057
{
2ff057
    return s->ds;
2ff057
}
2ff057
2ff057
PyObject * rpmds_Wrap(PyTypeObject *subtype, rpmds ds)
2ff057
{
2ff057
    rpmdsObject * s = (rpmdsObject *)subtype->tp_alloc(subtype, 0);
2ff057
    if (s == NULL) return NULL;
2ff057
2ff057
    s->ds = ds;
2ff057
    s->active = 0;
2ff057
    return (PyObject *) s;
2ff057
}
2ff057