[+] AuProcesses::RevealInDirectory(...)
[*] Fix dshit shell portal process spawning
This commit is contained in:
parent
5a0bd702c5
commit
9a172d9fca
@ -18,8 +18,15 @@ namespace Aurora::Processes
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* @brief Opens a file or directory
|
* @brief Opens a file or directory
|
||||||
* @param file unexpanded relative or absolute filepath
|
* @param file: unexpanded relative or absolute filepath
|
||||||
* @return
|
* @return
|
||||||
*/
|
*/
|
||||||
AUKN_SYM void OpenFile(const AuROString &file);
|
AUKN_SYM void OpenFile(const AuROString &file);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @brief Reveals a file or directory in the platform shell
|
||||||
|
* @param file: unexpanded relative or absolute filepath
|
||||||
|
* @return
|
||||||
|
*/
|
||||||
|
AUKN_SYM void RevealInDirectory(const AuROString &file);
|
||||||
}
|
}
|
@ -428,6 +428,9 @@ namespace Aurora
|
|||||||
ADD_GET_PROC(Shell, SHGetFolderPathA)
|
ADD_GET_PROC(Shell, SHGetFolderPathA)
|
||||||
ADD_GET_PROC(Shell, CommandLineToArgvW)
|
ADD_GET_PROC(Shell, CommandLineToArgvW)
|
||||||
ADD_GET_PROC(Shell, ShellExecuteW)
|
ADD_GET_PROC(Shell, ShellExecuteW)
|
||||||
|
ADD_GET_PROC(Shell, ILCreateFromPathW)
|
||||||
|
ADD_GET_PROC(Shell, ILFree)
|
||||||
|
ADD_GET_PROC(Shell, SHOpenFolderAndSelectItems)
|
||||||
|
|
||||||
ADD_GET_PROC(User32, GetClipboardData);
|
ADD_GET_PROC(User32, GetClipboardData);
|
||||||
ADD_GET_PROC(User32, MapVirtualKeyA);
|
ADD_GET_PROC(User32, MapVirtualKeyA);
|
||||||
|
@ -768,6 +768,21 @@ namespace Aurora
|
|||||||
LPSTR pszPath
|
LPSTR pszPath
|
||||||
);
|
);
|
||||||
|
|
||||||
|
inline void *(__stdcall *pILCreateFromPathW)(
|
||||||
|
LPCWSTR lpPath
|
||||||
|
);
|
||||||
|
|
||||||
|
inline HRESULT (__stdcall *pSHOpenFolderAndSelectItems)(
|
||||||
|
void * pidlFolder,
|
||||||
|
UINT cidl,
|
||||||
|
void * apidl,
|
||||||
|
DWORD dwFlags
|
||||||
|
);
|
||||||
|
|
||||||
|
inline void (__stdcall *pILFree)(
|
||||||
|
void * pidl
|
||||||
|
);
|
||||||
|
|
||||||
inline LPWSTR *(__stdcall *pCommandLineToArgvW)(
|
inline LPWSTR *(__stdcall *pCommandLineToArgvW)(
|
||||||
LPCWSTR lpCmdLine,
|
LPCWSTR lpCmdLine,
|
||||||
int * pNumArgs
|
int * pNumArgs
|
||||||
|
@ -14,14 +14,14 @@
|
|||||||
|
|
||||||
namespace Aurora::Processes
|
namespace Aurora::Processes
|
||||||
{
|
{
|
||||||
static void UnixOpenAsyncThread(AuString uri, bool bType)
|
static void UnixOpenAsyncThread(AuString uri, int iType)
|
||||||
{
|
{
|
||||||
bool bDirExists {};
|
bool bDirExists {};
|
||||||
bool bFileExists {};
|
bool bFileExists {};
|
||||||
|
|
||||||
bFileExists = AuIOFS::FileExists(uri);
|
bFileExists = AuIOFS::FileExists(uri);
|
||||||
|
|
||||||
if (!bType)
|
if (iType)
|
||||||
{
|
{
|
||||||
bDirExists = AuIOFS::DirExists(uri);
|
bDirExists = AuIOFS::DirExists(uri);
|
||||||
|
|
||||||
@ -56,70 +56,170 @@ namespace Aurora::Processes
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (fork() == 0)
|
static AuInitOnce gInitOnce;
|
||||||
|
static bool gUseDShid {};
|
||||||
|
|
||||||
|
gInitOnce.Call([]()
|
||||||
|
{
|
||||||
|
auto optStringA = AuProcess::EnvironmentGetOne("container");
|
||||||
|
auto optStringB = AuProcess::EnvironmentGetOne("AURORA_RUNTIME_USE_GDBUS_BIN_TO_PORTAL");
|
||||||
|
auto optStringC = AuProcess::EnvironmentGetOne("AURORA_RUNTIME_NO_MICROSOFT_REDHAT_SABOTAGE");
|
||||||
|
|
||||||
|
bool bIsFireJail = optStringA && optStringA.Value() == "firejail";
|
||||||
|
bool bIsForcedDBUS = optStringB && optStringB.Value() == "YES";
|
||||||
|
bool bAllowDbus = !optStringC && (AuFS::FileExists("/usr/bin/gdbus") || AuFS::FileExists("/bin/gdbus"));
|
||||||
|
|
||||||
|
if (bIsFireJail || bIsForcedDBUS || bAllowDbus)
|
||||||
|
{
|
||||||
|
gUseDShid = true;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
gUseDShid = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (!gUseDShid)
|
||||||
|
{
|
||||||
|
if (iType == 2)
|
||||||
|
{
|
||||||
|
AuROString out;
|
||||||
|
if (AuEndsWith(uri, '/'))
|
||||||
|
{
|
||||||
|
AuFS::GoUpToSeparator(out, uri);
|
||||||
|
AuFS::GoUpToSeparator(out, out);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
AuFS::GoUpToSeparator(out, uri);
|
||||||
|
}
|
||||||
|
uri = AuString(out);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
AuSemaphore semaphore;
|
||||||
|
volatile int type2 = iType;
|
||||||
|
|
||||||
|
auto iFork = fork();
|
||||||
|
if (iFork == 0)
|
||||||
{
|
{
|
||||||
setsid();
|
setsid();
|
||||||
|
|
||||||
|
// isn't posix fun?
|
||||||
PosixDoForkHooks();
|
PosixDoForkHooks();
|
||||||
PosixShutup();
|
PosixShutup();
|
||||||
PosixFDYeetus();
|
PosixFDYeetus();
|
||||||
|
|
||||||
auto optStringA = AuProcess::EnvironmentGetOne("container");
|
auto pBaseURI = (char *)SysAllocateLarge(uri.size() + 1);
|
||||||
auto optStringB = AuProcess::EnvironmentGetOne("AURORA_RUNTIME_USE_GDBUS_BIN_TO_PORTAL");
|
if (pBaseURI)
|
||||||
bool bIsFireJail = optStringA && optStringA.Value() == "firejail";
|
|
||||||
bool bIsForcedDBUS = optStringB && optStringB.Value() == "YES";
|
|
||||||
if (bIsFireJail || bIsForcedDBUS)
|
|
||||||
{
|
{
|
||||||
|
AuMemcpy(pBaseURI, uri.c_str(), uri.size() + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
int iType = type2;
|
||||||
|
|
||||||
|
// ...super fun
|
||||||
|
semaphore->Unlock(1);
|
||||||
|
// ...the original iType and uri buffer will be trashed from this point onwards
|
||||||
|
|
||||||
|
// and as if dealing with posix isn't bad enough...
|
||||||
|
// here's some more redhat/lennart poettering/dbus/xdg bullshid
|
||||||
|
if (gUseDShid)
|
||||||
|
{
|
||||||
|
const char *pExecString {};
|
||||||
|
bool bFuckRedHat { true };
|
||||||
|
char incompetentRedCunts[32];
|
||||||
|
|
||||||
|
if (iType == 2)
|
||||||
|
{
|
||||||
|
pExecString = "org.freedesktop.portal.OpenURI.OpenDirectory";
|
||||||
|
}
|
||||||
|
else if (bFileExists || bDirExists)
|
||||||
|
{
|
||||||
|
pExecString = "org.freedesktop.portal.OpenURI.OpenFile";
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
pExecString = "org.freedesktop.portal.OpenURI.OpenURI";
|
||||||
|
bFuckRedHat = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// it only gets worse, doesn't it?
|
||||||
|
if (bFuckRedHat)
|
||||||
|
{
|
||||||
|
int fu = PosixOpen(pBaseURI,
|
||||||
|
O_RDONLY,
|
||||||
|
0664); // dont care. use a real os that isnt controlled by redhat
|
||||||
|
snprintf(incompetentRedCunts, 32, "%i", fu);
|
||||||
|
pBaseURI = incompetentRedCunts;
|
||||||
|
}
|
||||||
|
|
||||||
|
// and now we pray to the demons of dshid
|
||||||
execlp("gdbus",
|
execlp("gdbus",
|
||||||
"gdbus",
|
"gdbus",
|
||||||
"call",
|
"call",
|
||||||
"--session",
|
"--session",
|
||||||
"--dest", "org.freedesktop.portal.Desktop",
|
"--dest", "org.freedesktop.portal.Desktop",
|
||||||
"--object-path", "/org/freedesktop/portal/desktop",
|
"--object-path", "/org/freedesktop/portal/desktop",
|
||||||
"--method", bFileExists ? "org.freedesktop.portal.OpenURI.OpenFile" : "org.freedesktop.portal.OpenURI.OpenURI",
|
"--method", pExecString,
|
||||||
"",
|
"",
|
||||||
uri.c_str(),
|
pBaseURI,
|
||||||
"{}",
|
"{}",
|
||||||
(char *)nullptr);
|
(char *)nullptr);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
if (AuFS::FileExists("/usr/bin/xdg-open"))
|
// how did we go from this to the above redhat bs?
|
||||||
|
// freetards have no standards whatsoever - in both a literal and figurative sense
|
||||||
|
if (AuFS::FileExists("/bin/xdg-open"))
|
||||||
{
|
{
|
||||||
execl("/usr/bin/xdg-open", "xdg-open", uri.c_str(), (char *)nullptr);
|
execl("/bin/xdg-open", "xdg-open", pBaseURI, (char *)nullptr);
|
||||||
|
}
|
||||||
|
else if (AuFS::FileExists("/usr/bin/xdg-open"))
|
||||||
|
{
|
||||||
|
execl("/usr/bin/xdg-open", "xdg-open", pBaseURI, (char *)nullptr);
|
||||||
}
|
}
|
||||||
else if (AuFS::FileExists("/usr/local/bin/xdg-open"))
|
else if (AuFS::FileExists("/usr/local/bin/xdg-open"))
|
||||||
{
|
{
|
||||||
execl("/usr/local/bin/xdg-open", "xdg-open", uri.c_str(), (char *)nullptr);
|
execl("/usr/local/bin/xdg-open", "xdg-open", pBaseURI, (char *)nullptr);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
execlp("xdg-open", "xdg-open", uri.c_str(), (char *)nullptr);
|
execlp("xdg-open", "xdg-open", pBaseURI, (char *)nullptr);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
exit(0);
|
exit(0);
|
||||||
}
|
}
|
||||||
|
else if (iFork > 0)
|
||||||
|
{
|
||||||
|
semaphore->Lock();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
static void UnixOpenAsync(const AuROString &uri, bool bType)
|
static void UnixOpenAsync(const AuROString &uri, int iType)
|
||||||
{
|
{
|
||||||
if (uri.empty())
|
if (uri.empty())
|
||||||
{
|
{
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
AuThreads::Spawn(std::bind(&UnixOpenAsyncThread, AuString(uri), bType), true);
|
AuThreads::Spawn(std::bind(&UnixOpenAsyncThread, AuString(uri), iType), true);
|
||||||
}
|
}
|
||||||
|
|
||||||
AUKN_SYM void OpenUri(const AuROString &uri)
|
AUKN_SYM void OpenUri(const AuROString &uri)
|
||||||
{
|
{
|
||||||
UnixOpenAsync(uri, true);
|
UnixOpenAsync(uri, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
AUKN_SYM void OpenFile(const AuROString &file)
|
AUKN_SYM void OpenFile(const AuROString &file)
|
||||||
{
|
{
|
||||||
UnixOpenAsync(file, false);
|
UnixOpenAsync(file, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
AUKN_SYM void RevealInDirectory(const AuROString &file)
|
||||||
|
{
|
||||||
|
UnixOpenAsync(file, 2);
|
||||||
}
|
}
|
||||||
}
|
}
|
@ -15,7 +15,7 @@
|
|||||||
|
|
||||||
namespace Aurora::Processes
|
namespace Aurora::Processes
|
||||||
{
|
{
|
||||||
static AuList<AuPair<AuString, bool>> gOpenItems;
|
static AuList<AuPair<AuString, int>> gOpenItems;
|
||||||
static AuConditionMutex gCondMutex;
|
static AuConditionMutex gCondMutex;
|
||||||
static AuConditionVariable gCondVariable(AuUnsafeRaiiToShared(gCondMutex.AsPointer()));
|
static AuConditionVariable gCondVariable(AuUnsafeRaiiToShared(gCondMutex.AsPointer()));
|
||||||
static AuThreads::ThreadUnique_t gOpenerThread;
|
static AuThreads::ThreadUnique_t gOpenerThread;
|
||||||
@ -24,7 +24,7 @@ namespace Aurora::Processes
|
|||||||
{
|
{
|
||||||
AU_LOCK_GUARD(gCondMutex);
|
AU_LOCK_GUARD(gCondMutex);
|
||||||
|
|
||||||
while (AuIsThreadRunning())
|
while (AuIsThreadRunning() || gOpenItems.size())
|
||||||
{
|
{
|
||||||
try
|
try
|
||||||
{
|
{
|
||||||
@ -48,7 +48,7 @@ namespace Aurora::Processes
|
|||||||
|
|
||||||
bFileExists = AuIOFS::FileExists(uri);
|
bFileExists = AuIOFS::FileExists(uri);
|
||||||
|
|
||||||
if (!type)
|
if (type)
|
||||||
{
|
{
|
||||||
bDirExists = AuIOFS::DirExists(uri);
|
bDirExists = AuIOFS::DirExists(uri);
|
||||||
|
|
||||||
@ -79,18 +79,34 @@ namespace Aurora::Processes
|
|||||||
|
|
||||||
auto u16OpenURI = Locale::ConvertFromUTF8(uri);
|
auto u16OpenURI = Locale::ConvertFromUTF8(uri);
|
||||||
auto pOpenType = bDirExists ? L"explore" : L"open";
|
auto pOpenType = bDirExists ? L"explore" : L"open";
|
||||||
|
bool bReveal = type == 2 &&
|
||||||
|
pILCreateFromPathW &&
|
||||||
|
pSHOpenFolderAndSelectItems &&
|
||||||
|
pILFree;
|
||||||
|
|
||||||
gCondMutex->Unlock();
|
gCondMutex->Unlock();
|
||||||
|
if (bReveal)
|
||||||
|
{
|
||||||
|
if (auto pIL = pILCreateFromPathW(u16OpenURI.c_str()))
|
||||||
|
{
|
||||||
|
pSHOpenFolderAndSelectItems(pIL, 0, nullptr, 0);
|
||||||
|
pILFree(pIL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
pShellExecuteW(nullptr,
|
pShellExecuteW(nullptr,
|
||||||
pOpenType,
|
pOpenType,
|
||||||
u16OpenURI.c_str(),
|
u16OpenURI.c_str(),
|
||||||
nullptr,
|
nullptr,
|
||||||
nullptr,
|
nullptr,
|
||||||
SW_SHOWNORMAL);
|
SW_SHOWNORMAL);
|
||||||
|
}
|
||||||
gCondMutex->Lock();
|
gCondMutex->Lock();
|
||||||
|
|
||||||
// Move all N-1 elements for each N, vs clearing later (requires lock?), vs making a copy of the work (requires another alloc?).
|
// Work mostly in-place and move all N-1 elements for each N, vs do all in place and clear all later (requires uninterrupted mutex lock?), vs making a copy of the work queue (requires another alloc unless AuExchanged in-place, + needs another nested indentation for a for itr loop).
|
||||||
// We can just do ShellExecuteW outside of the lock, and access a locally owned U16-translated string container, without bothering the write side mutex or mutable work container.
|
// The former is pretty much the same as the latter with an AuExchange, except we use .erase(itr) moves to single step through the queue instead of batching & erasing all at once.
|
||||||
|
// We can just do ShellExecuteW outside of the lock and access a locally owned U16-translated string container, without bothering the write side mutex or mutable work container.
|
||||||
// A work queue buffer could solve this, but eh.
|
// A work queue buffer could solve this, but eh.
|
||||||
gOpenItems.erase(gOpenItems.begin());
|
gOpenItems.erase(gOpenItems.begin());
|
||||||
}
|
}
|
||||||
@ -145,7 +161,7 @@ namespace Aurora::Processes
|
|||||||
AUKN_SYM void OpenUri(const AuROString &uri)
|
AUKN_SYM void OpenUri(const AuROString &uri)
|
||||||
{
|
{
|
||||||
AU_LOCK_GLOBAL_GUARD(gCondMutex);
|
AU_LOCK_GLOBAL_GUARD(gCondMutex);
|
||||||
AuTryInsert(gOpenItems, AuMakePair(AuString(uri), true));
|
AuTryInsert(gOpenItems, AuMakePair(AuString(uri), 0));
|
||||||
gCondVariable->Signal();
|
gCondVariable->Signal();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -161,7 +177,24 @@ namespace Aurora::Processes
|
|||||||
|
|
||||||
{
|
{
|
||||||
AU_LOCK_GLOBAL_GUARD(gCondMutex);
|
AU_LOCK_GLOBAL_GUARD(gCondMutex);
|
||||||
AuTryInsert(gOpenItems, AuMove(AuMakePair(AuMove(path), false)));
|
AuTryInsert(gOpenItems, AuMove(AuMakePair(AuMove(path), 1)));
|
||||||
|
gCondVariable->Signal();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
AUKN_SYM void RevealInDirectory(const AuROString &file)
|
||||||
|
{
|
||||||
|
auto path = AuIOFS::NormalizePathRet(file);
|
||||||
|
|
||||||
|
if (path.empty())
|
||||||
|
{
|
||||||
|
SysPushErrorMemory();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
AU_LOCK_GLOBAL_GUARD(gCondMutex);
|
||||||
|
AuTryInsert(gOpenItems, AuMove(AuMakePair(AuMove(path), 2)));
|
||||||
gCondVariable->Signal();
|
gCondVariable->Signal();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user