signature: add bytes accessors for name/email
Now Signature.email returns unicode. The name and email are available as byte strings through the Signature._name and Signature._email accessors respectively. Signature._encoding returns the encoding used in the signature.
This commit is contained in:
2
TODO.txt
2
TODO.txt
@@ -1,6 +1,4 @@
|
|||||||
Signature
|
Signature
|
||||||
=========
|
=========
|
||||||
- Implement equality interface
|
- Implement equality interface
|
||||||
- Return unicode for the email
|
|
||||||
- Implement interface to access the name/email as bytes
|
|
||||||
- In Repository's create_commit/create_tag check signatures encoding is right
|
- In Repository's create_commit/create_tag check signatures encoding is right
|
||||||
|
56
pygit2.c
56
pygit2.c
@@ -44,7 +44,19 @@
|
|||||||
#define PyString_Size PyBytes_Size
|
#define PyString_Size PyBytes_Size
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* Utilities */
|
||||||
|
static PyObject *
|
||||||
|
to_str(const char *value, const char *encoding, const char *errors)
|
||||||
|
{
|
||||||
|
if (encoding == NULL)
|
||||||
|
encoding = "utf-8";
|
||||||
|
return PyUnicode_Decode(value, strlen(value), encoding, errors);
|
||||||
|
}
|
||||||
|
|
||||||
|
#define to_bytes(value) PyString_FromString(value)
|
||||||
|
|
||||||
|
|
||||||
|
/* Python objects */
|
||||||
typedef struct {
|
typedef struct {
|
||||||
PyObject_HEAD
|
PyObject_HEAD
|
||||||
git_repository *repo;
|
git_repository *repo;
|
||||||
@@ -358,15 +370,11 @@ py_str_to_c_str(PyObject *value, const char *encoding)
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
#define c_str_to_py_str(c_str) \
|
|
||||||
PyUnicode_DecodeUTF8(c_str, strlen(c_str), "strict")
|
|
||||||
|
|
||||||
#define py_path_to_c_str(py_path) \
|
#define py_path_to_c_str(py_path) \
|
||||||
py_str_to_c_str(py_path, Py_FileSystemDefaultEncoding)
|
py_str_to_c_str(py_path, Py_FileSystemDefaultEncoding)
|
||||||
|
|
||||||
#define c_str_to_py_path(c_str) \
|
#define c_str_to_py_path(c_str) \
|
||||||
PyUnicode_Decode(c_str, strlen(c_str), \
|
to_str(c_str, Py_FileSystemDefaultEncoding, "strict")
|
||||||
Py_FileSystemDefaultEncoding, "strict")
|
|
||||||
|
|
||||||
|
|
||||||
static int
|
static int
|
||||||
@@ -1136,9 +1144,7 @@ Commit_get_message(Commit *commit)
|
|||||||
|
|
||||||
message = git_commit_message(commit->commit);
|
message = git_commit_message(commit->commit);
|
||||||
encoding = git_commit_message_encoding(commit->commit);
|
encoding = git_commit_message_encoding(commit->commit);
|
||||||
if (encoding == NULL)
|
return to_str(message, encoding, "strict");
|
||||||
encoding = "utf-8";
|
|
||||||
return PyUnicode_Decode(message, strlen(message), encoding, "strict");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
@@ -1676,7 +1682,7 @@ Tag_get_name(Tag *self)
|
|||||||
name = git_tag_name(self->tag);
|
name = git_tag_name(self->tag);
|
||||||
if (!name)
|
if (!name)
|
||||||
Py_RETURN_NONE;
|
Py_RETURN_NONE;
|
||||||
return c_str_to_py_str(name);
|
return to_str(name, "utf-8", "strict");
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
@@ -1696,7 +1702,7 @@ Tag_get_message(Tag *self)
|
|||||||
message = git_tag_message(self->tag);
|
message = git_tag_message(self->tag);
|
||||||
if (!message)
|
if (!message)
|
||||||
Py_RETURN_NONE;
|
Py_RETURN_NONE;
|
||||||
return c_str_to_py_str(message);
|
return to_str(message, "utf-8", "strict");
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyGetSetDef Tag_getseters[] = {
|
static PyGetSetDef Tag_getseters[] = {
|
||||||
@@ -2645,24 +2651,39 @@ Signature_dealloc(Signature *self)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
Signature_get_name(Signature *self)
|
Signature_get_encoding(Signature *self)
|
||||||
{
|
{
|
||||||
const char *encoding;
|
const char *encoding;
|
||||||
git_object *object;
|
|
||||||
|
|
||||||
encoding = self->encoding;
|
encoding = self->encoding;
|
||||||
if (encoding == NULL)
|
if (encoding == NULL)
|
||||||
encoding = "utf-8";
|
encoding = "utf-8";
|
||||||
|
|
||||||
return PyUnicode_Decode(self->signature->name,
|
return PyUnicode_DecodeASCII(encoding, strlen(encoding), "strict");
|
||||||
strlen(self->signature->name),
|
}
|
||||||
encoding, "strict");
|
|
||||||
|
static PyObject *
|
||||||
|
Signature_get_raw_name(Signature *self)
|
||||||
|
{
|
||||||
|
return to_bytes(self->signature->name);
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *
|
||||||
|
Signature_get_raw_email(Signature *self)
|
||||||
|
{
|
||||||
|
return to_bytes(self->signature->email);
|
||||||
|
}
|
||||||
|
|
||||||
|
static PyObject *
|
||||||
|
Signature_get_name(Signature *self)
|
||||||
|
{
|
||||||
|
return to_str(self->signature->name, self->encoding, "strict");
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
Signature_get_email(Signature *self)
|
Signature_get_email(Signature *self)
|
||||||
{
|
{
|
||||||
return PyString_FromString(self->signature->email);
|
return to_str(self->signature->email, self->encoding, "strict");
|
||||||
}
|
}
|
||||||
|
|
||||||
static PyObject *
|
static PyObject *
|
||||||
@@ -2678,6 +2699,9 @@ Signature_get_offset(Signature *self)
|
|||||||
}
|
}
|
||||||
|
|
||||||
static PyGetSetDef Signature_getseters[] = {
|
static PyGetSetDef Signature_getseters[] = {
|
||||||
|
{"_encoding", (getter)Signature_get_encoding, NULL, "encoding", NULL},
|
||||||
|
{"_name", (getter)Signature_get_raw_name, NULL, "Name (bytes)", NULL},
|
||||||
|
{"_email", (getter)Signature_get_raw_email, NULL, "Email (bytes)", NULL},
|
||||||
{"name", (getter)Signature_get_name, NULL, "Name", NULL},
|
{"name", (getter)Signature_get_name, NULL, "Name", NULL},
|
||||||
{"email", (getter)Signature_get_email, NULL, "Email", NULL},
|
{"email", (getter)Signature_get_email, NULL, "Email", NULL},
|
||||||
{"time", (getter)Signature_get_time, NULL, "Time", NULL},
|
{"time", (getter)Signature_get_time, NULL, "Time", NULL},
|
||||||
|
@@ -36,7 +36,7 @@ import unittest
|
|||||||
|
|
||||||
|
|
||||||
names = ['blob', 'commit', 'index', 'refs', 'repository', 'revwalk', 'tag',
|
names = ['blob', 'commit', 'index', 'refs', 'repository', 'revwalk', 'tag',
|
||||||
'tree', 'status']
|
'tree', 'signature', 'status']
|
||||||
def test_suite():
|
def test_suite():
|
||||||
modules = ['test.test_%s' % n for n in names]
|
modules = ['test.test_%s' % n for n in names]
|
||||||
return unittest.defaultTestLoader.loadTestsFromNames(modules)
|
return unittest.defaultTestLoader.loadTestsFromNames(modules)
|
||||||
|
57
test/test_signature.py
Normal file
57
test/test_signature.py
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
# -*- coding: UTF-8 -*-
|
||||||
|
#
|
||||||
|
# Copyright 2011 J. David Ibáñez
|
||||||
|
#
|
||||||
|
# This file is free software; you can redistribute it and/or modify
|
||||||
|
# it under the terms of the GNU General Public License, version 2,
|
||||||
|
# as published by the Free Software Foundation.
|
||||||
|
#
|
||||||
|
# In addition to the permissions in the GNU General Public License,
|
||||||
|
# the authors give you unlimited permission to link the compiled
|
||||||
|
# version of this file into combinations with other programs,
|
||||||
|
# and to distribute those combinations without any restriction
|
||||||
|
# coming from the use of this file. (The General Public License
|
||||||
|
# restrictions do apply in other respects; for example, they cover
|
||||||
|
# modification of the file, and distribution when not linked into
|
||||||
|
# a combined executable.)
|
||||||
|
#
|
||||||
|
# This file is distributed in the hope that it will be useful, but
|
||||||
|
# WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
||||||
|
# General Public License for more details.
|
||||||
|
#
|
||||||
|
# You should have received a copy of the GNU General Public License
|
||||||
|
# along with this program; see the file COPYING. If not, write to
|
||||||
|
# the Free Software Foundation, 51 Franklin Street, Fifth Floor,
|
||||||
|
# Boston, MA 02110-1301, USA.
|
||||||
|
|
||||||
|
from __future__ import absolute_import
|
||||||
|
from __future__ import unicode_literals
|
||||||
|
import unittest
|
||||||
|
|
||||||
|
from pygit2 import Signature
|
||||||
|
from .utils import NoRepoTestCase
|
||||||
|
|
||||||
|
|
||||||
|
__author__ = 'jdavid.ibp@gmail.com (J. David Ibáñez)'
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
class SignatureTest(NoRepoTestCase):
|
||||||
|
|
||||||
|
def test_default(self):
|
||||||
|
signature = Signature('Foo Ibáñez', 'foo@example.com', 1322174594, 60)
|
||||||
|
encoding = signature._encoding
|
||||||
|
self.assertEqual(signature.name, signature._name.decode(encoding))
|
||||||
|
self.assertEqual(signature.name.encode(encoding), signature._name)
|
||||||
|
|
||||||
|
def test_latin1(self):
|
||||||
|
encoding = 'iso-8859-1'
|
||||||
|
signature = Signature('Foo Ibáñez', 'foo@example.com', 1322174594, 60,
|
||||||
|
encoding)
|
||||||
|
self.assertEqual(signature.name, signature._name.decode(encoding))
|
||||||
|
self.assertEqual(signature.name.encode(encoding), signature._name)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == '__main__':
|
||||||
|
unittest.main()
|
Reference in New Issue
Block a user