15ffd83dbSDimitry Andric //===-- TildeExpressionResolver.cpp ---------------------------------------===//
20b57cec5SDimitry Andric //
30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information.
50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
60b57cec5SDimitry Andric //
70b57cec5SDimitry Andric //===----------------------------------------------------------------------===//
80b57cec5SDimitry Andric 
90b57cec5SDimitry Andric #include "lldb/Utility/TildeExpressionResolver.h"
100b57cec5SDimitry Andric 
11*5f7ddb14SDimitry Andric #include <cassert>
120b57cec5SDimitry Andric #include <system_error>
130b57cec5SDimitry Andric 
140b57cec5SDimitry Andric #include "llvm/ADT/STLExtras.h"
150b57cec5SDimitry Andric #include "llvm/ADT/SmallVector.h"
160b57cec5SDimitry Andric #include "llvm/Support/FileSystem.h"
170b57cec5SDimitry Andric #include "llvm/Support/Path.h"
180b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h"
190b57cec5SDimitry Andric 
200b57cec5SDimitry Andric #if !defined(_WIN32)
210b57cec5SDimitry Andric #include <pwd.h>
220b57cec5SDimitry Andric #endif
230b57cec5SDimitry Andric 
240b57cec5SDimitry Andric using namespace lldb_private;
250b57cec5SDimitry Andric using namespace llvm;
260b57cec5SDimitry Andric 
270b57cec5SDimitry Andric namespace fs = llvm::sys::fs;
280b57cec5SDimitry Andric namespace path = llvm::sys::path;
290b57cec5SDimitry Andric 
30*5f7ddb14SDimitry Andric TildeExpressionResolver::~TildeExpressionResolver() = default;
310b57cec5SDimitry Andric 
ResolveExact(StringRef Expr,SmallVectorImpl<char> & Output)320b57cec5SDimitry Andric bool StandardTildeExpressionResolver::ResolveExact(
330b57cec5SDimitry Andric     StringRef Expr, SmallVectorImpl<char> &Output) {
340b57cec5SDimitry Andric   // We expect the tilde expression to be ONLY the expression itself, and
350b57cec5SDimitry Andric   // contain no separators.
360b57cec5SDimitry Andric   assert(!llvm::any_of(Expr, [](char c) { return path::is_separator(c); }));
370b57cec5SDimitry Andric   assert(Expr.empty() || Expr[0] == '~');
380b57cec5SDimitry Andric 
390b57cec5SDimitry Andric   return !fs::real_path(Expr, Output, true);
400b57cec5SDimitry Andric }
410b57cec5SDimitry Andric 
ResolvePartial(StringRef Expr,StringSet<> & Output)420b57cec5SDimitry Andric bool StandardTildeExpressionResolver::ResolvePartial(StringRef Expr,
430b57cec5SDimitry Andric                                                      StringSet<> &Output) {
440b57cec5SDimitry Andric   // We expect the tilde expression to be ONLY the expression itself, and
450b57cec5SDimitry Andric   // contain no separators.
460b57cec5SDimitry Andric   assert(!llvm::any_of(Expr, [](char c) { return path::is_separator(c); }));
470b57cec5SDimitry Andric   assert(Expr.empty() || Expr[0] == '~');
480b57cec5SDimitry Andric 
490b57cec5SDimitry Andric   Output.clear();
500b57cec5SDimitry Andric #if defined(_WIN32) || defined(__ANDROID__)
510b57cec5SDimitry Andric   return false;
520b57cec5SDimitry Andric #else
530b57cec5SDimitry Andric   if (Expr.empty())
540b57cec5SDimitry Andric     return false;
550b57cec5SDimitry Andric 
560b57cec5SDimitry Andric   SmallString<32> Buffer("~");
570b57cec5SDimitry Andric   setpwent();
580b57cec5SDimitry Andric   struct passwd *user_entry;
590b57cec5SDimitry Andric   Expr = Expr.drop_front();
600b57cec5SDimitry Andric 
610b57cec5SDimitry Andric   while ((user_entry = getpwent()) != nullptr) {
620b57cec5SDimitry Andric     StringRef ThisName(user_entry->pw_name);
630b57cec5SDimitry Andric     if (!ThisName.startswith(Expr))
640b57cec5SDimitry Andric       continue;
650b57cec5SDimitry Andric 
660b57cec5SDimitry Andric     Buffer.resize(1);
670b57cec5SDimitry Andric     Buffer.append(ThisName);
680b57cec5SDimitry Andric     Buffer.append(path::get_separator());
690b57cec5SDimitry Andric     Output.insert(Buffer);
700b57cec5SDimitry Andric   }
710b57cec5SDimitry Andric 
720b57cec5SDimitry Andric   return true;
730b57cec5SDimitry Andric #endif
740b57cec5SDimitry Andric }
750b57cec5SDimitry Andric 
ResolveFullPath(StringRef Expr,llvm::SmallVectorImpl<char> & Output)760b57cec5SDimitry Andric bool TildeExpressionResolver::ResolveFullPath(
770b57cec5SDimitry Andric     StringRef Expr, llvm::SmallVectorImpl<char> &Output) {
780b57cec5SDimitry Andric   if (!Expr.startswith("~")) {
79af732203SDimitry Andric     Output.assign(Expr.begin(), Expr.end());
800b57cec5SDimitry Andric     return false;
810b57cec5SDimitry Andric   }
820b57cec5SDimitry Andric 
830b57cec5SDimitry Andric   namespace path = llvm::sys::path;
840b57cec5SDimitry Andric   StringRef Left =
850b57cec5SDimitry Andric       Expr.take_until([](char c) { return path::is_separator(c); });
860b57cec5SDimitry Andric 
87af732203SDimitry Andric   if (!ResolveExact(Left, Output)) {
88af732203SDimitry Andric     Output.assign(Expr.begin(), Expr.end());
890b57cec5SDimitry Andric     return false;
90af732203SDimitry Andric   }
910b57cec5SDimitry Andric 
920b57cec5SDimitry Andric   Output.append(Expr.begin() + Left.size(), Expr.end());
930b57cec5SDimitry Andric   return true;
940b57cec5SDimitry Andric }
95