AliasController.php 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220
  1. <?php
  2. namespace App\Http\Controllers\Api;
  3. use App\Http\Controllers\Controller;
  4. use App\Http\Requests\IndexAliasRequest;
  5. use App\Http\Requests\StoreAliasRequest;
  6. use App\Http\Requests\UpdateAliasRequest;
  7. use App\Http\Resources\AliasResource;
  8. use App\Models\Domain;
  9. use App\Models\Username;
  10. use Illuminate\Support\Str;
  11. use Ramsey\Uuid\Uuid;
  12. use Pdp\Rules;
  13. use Pdp\Domain\fromIDNA2008;
  14. class AliasController extends Controller
  15. {
  16. public function index(IndexAliasRequest $request)
  17. {
  18. $aliases = user()->aliases()->with('recipients')
  19. ->when($request->input('sort'), function ($query, $sort) {
  20. $direction = strpos($sort, '-') === 0 ? 'desc' : 'asc';
  21. return $query->orderBy(ltrim($sort, '-'), $direction);
  22. }, function ($query) {
  23. return $query->latest();
  24. })
  25. ->when($request->input('filter.active'), function ($query, $value) {
  26. $active = $value === 'true' ? true : false;
  27. return $query->where('active', $active);
  28. });
  29. // Keep /aliases?deleted=with for backwards compatibility
  30. if ($request->deleted === 'with' || $request->input('filter.deleted') === 'with') {
  31. $aliases->withTrashed();
  32. }
  33. if ($request->deleted === 'only' || $request->input('filter.deleted') === 'only') {
  34. $aliases->onlyTrashed();
  35. }
  36. if ($request->input('filter.search')) {
  37. $searchTerm = strtolower($request->input('filter.search'));
  38. $aliases = $aliases->get()->filter(function ($alias) use ($searchTerm) {
  39. return Str::contains(strtolower($alias->email), $searchTerm) || Str::contains(strtolower($alias->description), $searchTerm);
  40. })->values();
  41. }
  42. $aliases = $aliases->jsonPaginate($request->input('page.size') ?? 100);
  43. return AliasResource::collection($aliases);
  44. }
  45. public function show($id)
  46. {
  47. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  48. return new AliasResource($alias->load('recipients'));
  49. }
  50. public function store(StoreAliasRequest $request)
  51. {
  52. if (user()->hasExceededNewAliasLimit()) {
  53. return response('You have reached your hourly limit for creating new aliases', 429);
  54. }
  55. $formatEmail = function(string $localPart, string $domain, string $prefix) {
  56. if (isset($prefix) {
  57. $localPart = $prefix . '.' . $localPart;
  58. }
  59. return $localPart . '@' . $domain;
  60. };
  61. if (isset($request->validated()['local_part'])) {
  62. $localPart = $request->validated()['local_part'];
  63. // Local part has extension
  64. if (Str::contains($localPart, '+')) {
  65. $extension = Str::after($localPart, '+');
  66. $localPart = Str::before($localPart, '+');
  67. }
  68. $data = [
  69. 'email' => $localPart . '@' . $request->domain,
  70. 'local_part' => $localPart,
  71. 'extension' => $extension ?? null
  72. ];
  73. } else {
  74. $prefix = null;
  75. if (isset($request->validated()['hostname'])) {
  76. // TODO this should be cached, perhaps on boot, from https://publicsuffix.org/
  77. // see https://github.com/jeremykendall/php-domain-parser
  78. $publicSuffixList = Rules::fromPath(sys_get_temp_dir() . '/public_suffix_list.dat');
  79. $hostname = $request->validated()['hostname'];
  80. $domain = Domain::fromIDNA2008($hostname);
  81. $prefix = $publicSuffixList->resolve($domain)->secondLevelDomain()->toString();
  82. }
  83. if ($request->input('format', 'random_characters') === 'random_words') {
  84. $localPart = user()->generateRandomWordLocalPart();
  85. $data = [
  86. 'email' => formatEmail($localPart, $request->domain, $prefix),
  87. 'local_part' => $localPart,
  88. ];
  89. } elseif ($request->input('format', 'random_characters') === 'random_characters') {
  90. $localPart = user()->generateRandomCharacterLocalPart(8);
  91. $data = [
  92. 'email' => formatEmail($localPart, $request->domain, $prefix),
  93. 'local_part' => $localPart,
  94. ];
  95. } else {
  96. $uuid = Uuid::uuid4();
  97. $data = [
  98. 'id' => $uuid,
  99. 'emai' => formatEmail($uuid, $request->domain, $prefix),
  100. 'local_part' => $uuid,
  101. ];
  102. }
  103. }
  104. // Check if domain is for username or custom domain
  105. $parentDomain = collect(config('anonaddy.all_domains'))
  106. ->filter(function ($name) use ($request) {
  107. return Str::endsWith($request->domain, $name);
  108. })
  109. ->first();
  110. $aliasable = null;
  111. // This is an AnonAddy domain.
  112. if ($parentDomain) {
  113. $subdomain = substr($request->domain, 0, strrpos($request->domain, '.'.$parentDomain));
  114. if ($username = Username::where('username', $subdomain)->first()) {
  115. $aliasable = $username;
  116. }
  117. } else {
  118. if ($customDomain = Domain::where('domain', $request->domain)->first()) {
  119. $aliasable = $customDomain;
  120. }
  121. }
  122. $data['aliasable_id'] = $aliasable->id ?? null;
  123. $data['aliasable_type'] = $aliasable ? 'App\\Models\\'.class_basename($aliasable) : null;
  124. $data['domain'] = $request->domain;
  125. $data['description'] = $request->description;
  126. $alias = user()->aliases()->create($data);
  127. if ($request->recipient_ids) {
  128. $alias->recipients()->sync($request->recipient_ids);
  129. }
  130. return new AliasResource($alias->refresh()->load('recipients'));
  131. }
  132. public function update(UpdateAliasRequest $request, $id)
  133. {
  134. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  135. $alias->update(['description' => $request->description]);
  136. return new AliasResource($alias->refresh()->load('recipients'));
  137. }
  138. public function restore($id)
  139. {
  140. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  141. $alias->restore();
  142. return new AliasResource($alias->refresh()->load('recipients'));
  143. }
  144. public function destroy($id)
  145. {
  146. $alias = user()->aliases()->findOrFail($id);
  147. $alias->recipients()->detach();
  148. $alias->delete();
  149. return response('', 204);
  150. }
  151. public function forget($id)
  152. {
  153. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  154. $alias->recipients()->detach();
  155. if ($alias->hasSharedDomain()) {
  156. // Remove all data from the alias and change user_id
  157. $alias->update([
  158. 'user_id' => '00000000-0000-0000-0000-000000000000',
  159. 'extension' => null,
  160. 'description' => null,
  161. 'emails_forwarded' => 0,
  162. 'emails_blocked' => 0,
  163. 'emails_replied' => 0,
  164. 'emails_sent' => 0
  165. ]);
  166. // Soft delete to prevent from being regenerated
  167. $alias->delete();
  168. } else {
  169. $alias->forceDelete();
  170. }
  171. return response('', 204);
  172. }
  173. }