Cache user and group name as these can be expensive to get on some systems.

git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@49043 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
Michael Wetherell 2007-10-04 15:59:09 +00:00
parent c005fb2809
commit f2e582dda8
3 changed files with 102 additions and 16 deletions

View File

@ -4637,6 +4637,14 @@ fi
WX_CHECK_FUNCS(fdopen) WX_CHECK_FUNCS(fdopen)
if test "$wxUSE_TARSTREAM" = "yes"; then
WX_CHECK_FUNCS(sysconf)
WX_CHECK_FUNCS(getpwuid_r, [], [], [#define _REENTRANT
#include <pwd.h>])
WX_CHECK_FUNCS(getgrgid_r, [], [], [#define _REENTRANT
#include <grp.h>])
fi
fi fi

View File

@ -1104,6 +1104,15 @@
/* Define if fdopen is available. */ /* Define if fdopen is available. */
#undef HAVE_FDOPEN #undef HAVE_FDOPEN
/* Define if sysconf is available. */
#undef HAVE_SYSCONF
/* Define if getpwuid_r is available. */
#undef HAVE_GETPWUID_R
/* Define if getgrgid_r is available. */
#undef HAVE_GETGRGID_R
/* ------------------------------------------------------------------------- /* -------------------------------------------------------------------------
Win32 adjustments section Win32 adjustments section
------------------------------------------------------------------------- */ ------------------------------------------------------------------------- */

View File

@ -28,10 +28,12 @@
#include "wx/datetime.h" #include "wx/datetime.h"
#include "wx/ptr_scpd.h" #include "wx/ptr_scpd.h"
#include "wx/filename.h" #include "wx/filename.h"
#include "wx/thread.h"
#include <ctype.h> #include <ctype.h>
#ifdef __UNIX__ #ifdef __UNIX__
#include <pwd.h>
#include <grp.h> #include <grp.h>
#endif #endif
@ -344,32 +346,99 @@ static wxFileOffset RoundUpSize(wxFileOffset size, int factor = 1)
return ((size + chunk - 1) / chunk) * chunk; return ((size + chunk - 1) / chunk) * chunk;
} }
static wxString GroupName()
{
#ifdef __UNIX__ #ifdef __UNIX__
group *gr;
if ((gr = getgrgid(getgid())) != NULL) static wxString wxTarUserName(int uid)
return wxString(gr->gr_name, wxConvLibc); {
struct passwd *ppw;
#ifdef HAVE_GETPWUID_R
#if defined HAVE_SYSCONF && defined _SC_GETPW_R_SIZE_MAX
long pwsize = sysconf(_SC_GETPW_R_SIZE_MAX);
size_t bufsize(wxMin(wxMax(1024l, pwsize), 32768l));
#else
size_t bufsize = 1024;
#endif
wxCharBuffer buf(bufsize);
struct passwd pw;
if (getpwuid_r(uid, &pw, buf.data(), bufsize, &ppw) == 0)
return wxString(pw.pw_name, wxConvLibc);
#else
if ((ppw = getpwuid(uid)) != NULL)
return wxString(ppw->pw_name, wxConvLibc);
#endif #endif
return _("unknown"); return _("unknown");
} }
static inline int UserId() static wxString wxTarGroupName(int gid)
{ {
#ifdef __UNIX__ struct group *pgr;
return getuid(); #ifdef HAVE_GETGRGID_R
#if defined HAVE_SYSCONF && defined _SC_GETGR_R_SIZE_MAX
long grsize = sysconf(_SC_GETGR_R_SIZE_MAX);
size_t bufsize(wxMin(wxMax(1024l, grsize), 32768l));
#else #else
return 0; size_t bufsize = 1024;
#endif #endif
wxCharBuffer buf(bufsize);
struct group gr;
if (getgrgid_r(gid, &gr, buf.data(), bufsize, &pgr) == 0)
return wxString(gr.gr_name, wxConvLibc);
#else
if ((pgr = getgrgid(gid)) != NULL)
return wxString(pgr->gr_name, wxConvLibc);
#endif
return _("unknown");
} }
static inline int GroupId() #endif // __UNIX__
// Cache the user and group names since getting them can be expensive,
// get both names and ids at the same time.
//
struct wxTarUser
{
wxTarUser();
~wxTarUser() { delete [] uname; delete [] gname; }
int uid;
int gid;
wxChar *uname;
wxChar *gname;
};
wxTarUser::wxTarUser()
{ {
#ifdef __UNIX__ #ifdef __UNIX__
return getgid(); uid = getuid();
gid = getgid();
wxString usr = wxTarUserName(uid);
wxString grp = wxTarGroupName(gid);
#else #else
return 0; uid = 0;
gid = 0;
wxString usr = wxGetUserId();
wxString grp = _("unknown");
#endif #endif
uname = new wxChar[usr.length() + 1];
wxStrcpy(uname, usr.c_str());
gname = new wxChar[grp.length() + 1];
wxStrcpy(gname, grp.c_str());
}
static const wxTarUser& wxGetTarUser()
{
#if wxUSE_THREADS
static wxCriticalSection cs;
wxCriticalSectionLocker lock(cs);
#endif
static wxTarUser tu;
return tu;
} }
// ignore the size field for entry types 3, 4, 5 and 6 // ignore the size field for entry types 3, 4, 5 and 6
@ -397,14 +466,14 @@ wxTarEntry::wxTarEntry(const wxString& name /*=wxEmptyString*/,
wxFileOffset size /*=0*/) wxFileOffset size /*=0*/)
: m_Mode(0644), : m_Mode(0644),
m_IsModeSet(false), m_IsModeSet(false),
m_UserId(UserId()), m_UserId(wxGetTarUser().uid),
m_GroupId(GroupId()), m_GroupId(wxGetTarUser().gid),
m_Size(size), m_Size(size),
m_Offset(wxInvalidOffset), m_Offset(wxInvalidOffset),
m_ModifyTime(dt), m_ModifyTime(dt),
m_TypeFlag(wxTAR_REGTYPE), m_TypeFlag(wxTAR_REGTYPE),
m_UserName(wxGetUserId()), m_UserName(wxGetTarUser().uname),
m_GroupName(GroupName()), m_GroupName(wxGetTarUser().gname),
m_DevMajor(~0), m_DevMajor(~0),
m_DevMinor(~0) m_DevMinor(~0)
{ {