1f678e45dSDimitry Andric //===--------------------- TildeExpressionResolver.cpp ----------*- C++ -*-===//
2f678e45dSDimitry Andric //
3f678e45dSDimitry Andric //                     The LLVM Compiler Infrastructure
4f678e45dSDimitry Andric //
5f678e45dSDimitry Andric // This file is distributed under the University of Illinois Open Source
6f678e45dSDimitry Andric // License. See LICENSE.TXT for details.
7f678e45dSDimitry Andric //
8f678e45dSDimitry Andric //===----------------------------------------------------------------------===//
9f678e45dSDimitry Andric 
10f678e45dSDimitry Andric #include "lldb/Utility/TildeExpressionResolver.h"
11f678e45dSDimitry Andric 
12*b5893f02SDimitry Andric #include <assert.h>
13*b5893f02SDimitry Andric #include <system_error>
14f678e45dSDimitry Andric 
15*b5893f02SDimitry Andric #include "llvm/ADT/STLExtras.h"
16*b5893f02SDimitry Andric #include "llvm/ADT/SmallVector.h"
17f678e45dSDimitry Andric #include "llvm/Support/FileSystem.h"
18f678e45dSDimitry Andric #include "llvm/Support/Path.h"
19*b5893f02SDimitry Andric #include "llvm/Support/raw_ostream.h"
20f678e45dSDimitry Andric 
214ba319b5SDimitry Andric #if !defined(_WIN32)
22f678e45dSDimitry Andric #include <pwd.h>
23f678e45dSDimitry Andric #endif
24f678e45dSDimitry Andric 
25f678e45dSDimitry Andric using namespace lldb_private;
26f678e45dSDimitry Andric using namespace llvm;
27f678e45dSDimitry Andric 
28f678e45dSDimitry Andric namespace fs = llvm::sys::fs;
29f678e45dSDimitry Andric namespace path = llvm::sys::path;
30f678e45dSDimitry Andric 
~TildeExpressionResolver()31f678e45dSDimitry Andric TildeExpressionResolver::~TildeExpressionResolver() {}
32f678e45dSDimitry Andric 
ResolveExact(StringRef Expr,SmallVectorImpl<char> & Output)33f678e45dSDimitry Andric bool StandardTildeExpressionResolver::ResolveExact(
34f678e45dSDimitry Andric     StringRef Expr, SmallVectorImpl<char> &Output) {
35f678e45dSDimitry Andric   // We expect the tilde expression to be ONLY the expression itself, and
36f678e45dSDimitry Andric   // contain no separators.
37f678e45dSDimitry Andric   assert(!llvm::any_of(Expr, [](char c) { return path::is_separator(c); }));
38f678e45dSDimitry Andric   assert(Expr.empty() || Expr[0] == '~');
39f678e45dSDimitry Andric 
40f678e45dSDimitry Andric   return !fs::real_path(Expr, Output, true);
41f678e45dSDimitry Andric }
42f678e45dSDimitry Andric 
ResolvePartial(StringRef Expr,StringSet<> & Output)43f678e45dSDimitry Andric bool StandardTildeExpressionResolver::ResolvePartial(StringRef Expr,
44f678e45dSDimitry Andric                                                      StringSet<> &Output) {
45f678e45dSDimitry Andric   // We expect the tilde expression to be ONLY the expression itself, and
46f678e45dSDimitry Andric   // contain no separators.
47f678e45dSDimitry Andric   assert(!llvm::any_of(Expr, [](char c) { return path::is_separator(c); }));
48f678e45dSDimitry Andric   assert(Expr.empty() || Expr[0] == '~');
49f678e45dSDimitry Andric 
50f678e45dSDimitry Andric   Output.clear();
514ba319b5SDimitry Andric #if defined(_WIN32) || defined(__ANDROID__)
52f678e45dSDimitry Andric   return false;
53f678e45dSDimitry Andric #else
54f678e45dSDimitry Andric   if (Expr.empty())
55f678e45dSDimitry Andric     return false;
56f678e45dSDimitry Andric 
57f678e45dSDimitry Andric   SmallString<32> Buffer("~");
58f678e45dSDimitry Andric   setpwent();
59f678e45dSDimitry Andric   struct passwd *user_entry;
60f678e45dSDimitry Andric   Expr = Expr.drop_front();
61f678e45dSDimitry Andric 
62*b5893f02SDimitry Andric   while ((user_entry = getpwent()) != nullptr) {
63f678e45dSDimitry Andric     StringRef ThisName(user_entry->pw_name);
64f678e45dSDimitry Andric     if (!ThisName.startswith(Expr))
65f678e45dSDimitry Andric       continue;
66f678e45dSDimitry Andric 
67f678e45dSDimitry Andric     Buffer.resize(1);
68f678e45dSDimitry Andric     Buffer.append(ThisName);
69f678e45dSDimitry Andric     Buffer.append(path::get_separator());
70f678e45dSDimitry Andric     Output.insert(Buffer);
71f678e45dSDimitry Andric   }
72f678e45dSDimitry Andric 
73f678e45dSDimitry Andric   return true;
74f678e45dSDimitry Andric #endif
75f678e45dSDimitry Andric }
76f678e45dSDimitry Andric 
ResolveFullPath(StringRef Expr,llvm::SmallVectorImpl<char> & Output)77f678e45dSDimitry Andric bool TildeExpressionResolver::ResolveFullPath(
78f678e45dSDimitry Andric     StringRef Expr, llvm::SmallVectorImpl<char> &Output) {
79f678e45dSDimitry Andric   Output.clear();
80f678e45dSDimitry Andric   if (!Expr.startswith("~")) {
81f678e45dSDimitry Andric     Output.append(Expr.begin(), Expr.end());
82f678e45dSDimitry Andric     return false;
83f678e45dSDimitry Andric   }
84f678e45dSDimitry Andric 
85f678e45dSDimitry Andric   namespace path = llvm::sys::path;
86f678e45dSDimitry Andric   StringRef Left =
87f678e45dSDimitry Andric       Expr.take_until([](char c) { return path::is_separator(c); });
88f678e45dSDimitry Andric 
89f678e45dSDimitry Andric   if (!ResolveExact(Left, Output))
90f678e45dSDimitry Andric     return false;
91f678e45dSDimitry Andric 
92f678e45dSDimitry Andric   Output.append(Expr.begin() + Left.size(), Expr.end());
93f678e45dSDimitry Andric   return true;
94f678e45dSDimitry Andric }
95