AliasController.php 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189
  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\AdditionalUsername;
  9. use App\Models\Domain;
  10. use Illuminate\Support\Str;
  11. use Ramsey\Uuid\Uuid;
  12. class AliasController extends Controller
  13. {
  14. public function index(IndexAliasRequest $request)
  15. {
  16. $aliases = user()->aliases()->with('recipients')->latest();
  17. // Keep /aliases?deleted=with for backwards compatibility
  18. if ($request->deleted === 'with' || $request->input('filter.deleted') === 'with') {
  19. $aliases->withTrashed();
  20. }
  21. if ($request->deleted === 'only' || $request->input('filter.deleted') === 'only') {
  22. $aliases->onlyTrashed();
  23. }
  24. if ($request->input('filter.search')) {
  25. $searchTerm = strtolower($request->input('filter.search'));
  26. $aliases = $aliases->get()->filter(function ($alias) use ($searchTerm) {
  27. return Str::contains(strtolower($alias->email), $searchTerm) || Str::contains(strtolower($alias->description), $searchTerm);
  28. })->values();
  29. }
  30. $aliases = $aliases->jsonPaginate($request->input('page.size') ?? 100);
  31. return AliasResource::collection($aliases);
  32. }
  33. public function show($id)
  34. {
  35. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  36. return new AliasResource($alias->load('recipients'));
  37. }
  38. public function store(StoreAliasRequest $request)
  39. {
  40. if (user()->hasExceededNewAliasLimit()) {
  41. return response('You have reached your hourly limit for creating new aliases', 429);
  42. }
  43. if (isset($request->validated()['local_part'])) {
  44. $localPart = $request->validated()['local_part'];
  45. // Local part has extension
  46. if (Str::contains($localPart, '+')) {
  47. $extension = Str::after($localPart, '+');
  48. $localPart = Str::before($localPart, '+');
  49. }
  50. $data = [
  51. 'email' => $localPart . '@' . $request->domain,
  52. 'local_part' => $localPart,
  53. 'extension' => $extension ?? null
  54. ];
  55. } else {
  56. if ($request->input('format', 'random_characters') === 'random_words') {
  57. $localPart = user()->generateRandomWordLocalPart();
  58. $data = [
  59. 'email' => $localPart . '@' . $request->domain,
  60. 'local_part' => $localPart,
  61. ];
  62. } elseif ($request->input('format', 'random_characters') === 'random_characters') {
  63. $localPart = user()->generateRandomCharacterLocalPart(8);
  64. $data = [
  65. 'email' => $localPart . '@' . $request->domain,
  66. 'local_part' => $localPart,
  67. ];
  68. } else {
  69. $uuid = Uuid::uuid4();
  70. $data = [
  71. 'id' => $uuid,
  72. 'email' => $uuid . '@' . $request->domain,
  73. 'local_part' => $uuid,
  74. ];
  75. }
  76. }
  77. // Check if domain is for additional username or custom domain
  78. $parentDomain = collect(config('anonaddy.all_domains'))
  79. ->filter(function ($name) use ($request) {
  80. return Str::endsWith($request->domain, $name);
  81. })
  82. ->first();
  83. $aliasable = null;
  84. // This is an AnonAddy domain.
  85. if ($parentDomain) {
  86. $subdomain = substr($request->domain, 0, strrpos($request->domain, '.'.$parentDomain));
  87. if ($additionalUsername = AdditionalUsername::where('username', $subdomain)->first()) {
  88. $aliasable = $additionalUsername;
  89. }
  90. } else {
  91. if ($customDomain = Domain::where('domain', $request->domain)->first()) {
  92. $aliasable = $customDomain;
  93. }
  94. }
  95. $data['aliasable_id'] = $aliasable->id ?? null;
  96. $data['aliasable_type'] = $aliasable ? 'App\\Models\\'.class_basename($aliasable) : null;
  97. $data['domain'] = $request->domain;
  98. $data['description'] = $request->description;
  99. $alias = user()->aliases()->create($data);
  100. if ($request->recipient_ids) {
  101. $alias->recipients()->sync($request->recipient_ids);
  102. }
  103. return new AliasResource($alias->refresh()->load('recipients'));
  104. }
  105. public function update(UpdateAliasRequest $request, $id)
  106. {
  107. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  108. $alias->update(['description' => $request->description]);
  109. return new AliasResource($alias->refresh()->load('recipients'));
  110. }
  111. public function restore($id)
  112. {
  113. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  114. $alias->restore();
  115. return new AliasResource($alias->refresh()->load('recipients'));
  116. }
  117. public function destroy($id)
  118. {
  119. $alias = user()->aliases()->findOrFail($id);
  120. $alias->recipients()->detach();
  121. $alias->delete();
  122. return response('', 204);
  123. }
  124. public function forget($id)
  125. {
  126. $alias = user()->aliases()->withTrashed()->findOrFail($id);
  127. $alias->recipients()->detach();
  128. if ($alias->hasSharedDomain()) {
  129. // Remove all data from the alias and change user_id
  130. $alias->update([
  131. 'user_id' => '00000000-0000-0000-0000-000000000000',
  132. 'extension' => null,
  133. 'description' => null,
  134. 'emails_forwarded' => 0,
  135. 'emails_blocked' => 0,
  136. 'emails_replied' => 0,
  137. 'emails_sent' => 0
  138. ]);
  139. // Soft delete to prevent from being regenerated
  140. $alias->delete();
  141. } else {
  142. $alias->forceDelete();
  143. }
  144. return response('', 204);
  145. }
  146. }