PortfolioController.php 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568
  1. <?php
  2. namespace App\Http\Controllers;
  3. use Illuminate\Http\Request;
  4. use App\Models\Portfolio;
  5. use Cache;
  6. use DB;
  7. use App\Status;
  8. use App\User;
  9. use App\Services\AccountService;
  10. use App\Services\StatusService;
  11. class PortfolioController extends Controller
  12. {
  13. const RSS_FEED_KEY = 'pf:portfolio:rss-feed:';
  14. const CACHED_FEED_KEY = 'pf:portfolio:cached-feed:';
  15. const RECENT_FEED_KEY = 'pf:portfolio:recent-feed:';
  16. public function index(Request $request)
  17. {
  18. return view('portfolio.index');
  19. }
  20. public function show(Request $request, $username)
  21. {
  22. $user = User::whereUsername($username)->first();
  23. if(!$user) {
  24. return view('portfolio.404');
  25. }
  26. $portfolio = Portfolio::whereUserId($user->id)->firstOrFail();
  27. $user = AccountService::get($user->profile_id);
  28. if($user['locked']) {
  29. return view('portfolio.404');
  30. }
  31. if($portfolio->active != true) {
  32. if(!$request->user()) {
  33. return view('portfolio.404');
  34. }
  35. if($request->user()->profile_id == $user['id']) {
  36. return redirect(config('portfolio.path') . '/settings');
  37. }
  38. return view('portfolio.404');
  39. }
  40. return view('portfolio.show', compact('user', 'portfolio'));
  41. }
  42. public function showPost(Request $request, $username, $id)
  43. {
  44. $authed = $request->user();
  45. $post = StatusService::get($id);
  46. if(!$post) {
  47. return view('portfolio.404');
  48. }
  49. $user = AccountService::get($post['account']['id']);
  50. $portfolio = Portfolio::whereProfileId($user['id'])->first();
  51. if(!$portfolio || $user['locked'] || $portfolio->active != true) {
  52. return view('portfolio.404');
  53. }
  54. if(!$post || $post['visibility'] != 'public' || !in_array($post['pf_type'], ['photo', 'photo:album']) || $user['id'] != $post['account']['id']) {
  55. return view('portfolio.404');
  56. }
  57. return view('portfolio.show_post', compact('user', 'post', 'authed'));
  58. }
  59. public function myRedirect(Request $request)
  60. {
  61. abort_if(!$request->user(), 404);
  62. $user = $request->user();
  63. if(Portfolio::whereProfileId($user->profile_id)->exists() === false) {
  64. $portfolio = new Portfolio;
  65. $portfolio->profile_id = $user->profile_id;
  66. $portfolio->user_id = $user->id;
  67. $portfolio->active = false;
  68. $portfolio->save();
  69. }
  70. $domain = config('portfolio.domain');
  71. $path = config('portfolio.path');
  72. $url = 'https://' . $domain . $path;
  73. return redirect($url);
  74. }
  75. public function settings(Request $request)
  76. {
  77. if(!$request->user()) {
  78. return redirect(route('home'));
  79. }
  80. $portfolio = Portfolio::whereUserId($request->user()->id)->first();
  81. if(!$portfolio) {
  82. $portfolio = new Portfolio;
  83. $portfolio->user_id = $request->user()->id;
  84. $portfolio->profile_id = $request->user()->profile_id;
  85. $portfolio->save();
  86. }
  87. return view('portfolio.settings', compact('portfolio'));
  88. }
  89. public function store(Request $request)
  90. {
  91. abort_unless($request->user(), 404);
  92. $this->validate($request, [
  93. 'profile_source' => 'required|in:recent,custom',
  94. 'layout' => 'required|in:grid,masonry',
  95. 'layout_container' => 'required|in:fixed,fluid',
  96. ]);
  97. $portfolio = Portfolio::whereUserId($request->user()->id)->first();
  98. if(!$portfolio) {
  99. $portfolio = new Portfolio;
  100. $portfolio->user_id = $request->user()->id;
  101. $portfolio->profile_id = $request->user()->profile_id;
  102. $portfolio->save();
  103. }
  104. $portfolio->active = $request->input('enabled') === 'on';
  105. $portfolio->show_captions = $request->input('show_captions') === 'on';
  106. $portfolio->show_license = $request->input('show_license') === 'on';
  107. $portfolio->show_location = $request->input('show_location') === 'on';
  108. $portfolio->show_timestamp = $request->input('show_timestamp') === 'on';
  109. $portfolio->show_link = $request->input('show_link') === 'on';
  110. $portfolio->profile_source = $request->input('profile_source');
  111. $portfolio->show_avatar = $request->input('show_avatar') === 'on';
  112. $portfolio->show_bio = $request->input('show_bio') === 'on';
  113. $portfolio->profile_layout = $request->input('layout');
  114. $portfolio->profile_container = $request->input('layout_container');
  115. $portfolio->metadata = $metadata;
  116. $portfolio->save();
  117. return redirect('/' . $request->user()->username);
  118. }
  119. public function getFeed(Request $request, $id)
  120. {
  121. $user = AccountService::get($id, true);
  122. if(!$user || !isset($user['id'])) {
  123. return response()->json([], 404);
  124. }
  125. $portfolio = Portfolio::whereProfileId($user['id'])->first();
  126. if(!$portfolio || !$portfolio->active) {
  127. return response()->json([], 404);
  128. }
  129. if($portfolio->profile_source === 'custom' && $portfolio->metadata) {
  130. return $this->getCustomFeed($portfolio);
  131. }
  132. return $this->getRecentFeed($user['id']);
  133. }
  134. protected function getCustomFeed($portfolio) {
  135. if(!isset($portfolio->metadata['posts']) || !$portfolio->metadata['posts']) {
  136. return response()->json([], 400);
  137. }
  138. $feed = Cache::remember(self::CACHED_FEED_KEY . $portfolio->profile_id, 86400, function() use($portfolio) {
  139. return collect($portfolio->metadata['posts'])->map(function($p) {
  140. return StatusService::get($p);
  141. })
  142. ->filter(function($p) {
  143. return $p && isset($p['account']);
  144. });
  145. });
  146. if($portfolio->metadata && isset($portfolio->metadata['feed_order']) && $portfolio->metadata['feed_order'] === 'recent') {
  147. return $feed->reverse()->values();
  148. } else {
  149. return $feed->values();
  150. }
  151. }
  152. protected function getRecentFeed($id) {
  153. $media = Cache::remember(self::RECENT_FEED_KEY . $id, 3600, function() use($id) {
  154. return DB::table('media')
  155. ->whereProfileId($id)
  156. ->whereNotNull('status_id')
  157. ->groupBy('status_id')
  158. ->orderByDesc('id')
  159. ->take(50)
  160. ->pluck('status_id');
  161. });
  162. return $media->map(function($sid) use($id) {
  163. return StatusService::get($sid);
  164. })
  165. ->filter(function($post) {
  166. return $post &&
  167. isset($post['media_attachments']) &&
  168. !empty($post['media_attachments']) &&
  169. $post['pf_type'] === 'photo' &&
  170. $post['visibility'] === 'public';
  171. })
  172. ->take(24)
  173. ->values();
  174. }
  175. public function getSettings(Request $request)
  176. {
  177. abort_if(!$request->user(), 403);
  178. $res = Portfolio::whereUserId($request->user()->id)->get();
  179. if(!$res) {
  180. return [];
  181. }
  182. return $res->map(function($p) {
  183. $metadata = $p->metadata;
  184. $bgColor = $metadata && isset($metadata['background_color']) ? $metadata['background_color'] : '#000000';
  185. $textColor = $metadata && isset($metadata['text_color']) ? $metadata['text_color'] : '#d4d4d8';
  186. $rssEnabled = $metadata && isset($metadata['rss_enabled']) ? $metadata['rss_enabled'] : false;
  187. $rssButton = $metadata && isset($metadata['show_rss_button']) ? $metadata['show_rss_button'] : false;
  188. $colorScheme = $metadata && isset($metadata['color_scheme']) ? $metadata['color_scheme'] : 'dark';
  189. $feedOrder = $metadata && isset($metadata['feed_order']) ? $metadata['feed_order'] : 'oldest';
  190. return [
  191. 'url' => $p->url(),
  192. 'pid' => (string) $p->profile_id,
  193. 'active' => (bool) $p->active,
  194. 'show_captions' => (bool) $p->show_captions,
  195. 'show_license' => (bool) $p->show_license,
  196. 'show_location' => (bool) $p->show_location,
  197. 'show_timestamp' => (bool) $p->show_timestamp,
  198. 'show_link' => (bool) $p->show_link,
  199. 'show_avatar' => (bool) $p->show_avatar,
  200. 'show_bio' => (bool) $p->show_bio,
  201. 'profile_layout' => $p->profile_layout,
  202. 'profile_source' => $p->profile_source,
  203. 'color_scheme' => $colorScheme,
  204. 'background_color' => $bgColor,
  205. 'text_color' => $textColor,
  206. 'show_profile_button' => true,
  207. 'rss_enabled' => $rssEnabled,
  208. 'show_rss_button' => $rssButton,
  209. 'feed_order' => $feedOrder,
  210. 'metadata' => $p->metadata
  211. ];
  212. })->first();
  213. }
  214. public function getAccountSettings(Request $request)
  215. {
  216. $this->validate($request, [
  217. 'id' => 'required|integer'
  218. ]);
  219. $account = AccountService::get($request->input('id'));
  220. abort_if(!$account, 404);
  221. $p = Portfolio::whereProfileId($request->input('id'))->whereActive(1)->firstOrFail();
  222. if(!$p) {
  223. return [];
  224. }
  225. $metadata = $p->metadata;
  226. $rssEnabled = $metadata && isset($metadata['rss_enabled']) ? $metadata['rss_enabled'] : false;
  227. $rssButton = $metadata && isset($metadata['show_rss_button']) ? $metadata['show_rss_button'] : false;
  228. $profileButton = $metadata && isset($metadata['show_profile_button']) ? $metadata['show_profile_button'] : false;
  229. $res = [
  230. 'url' => $p->url(),
  231. 'show_captions' => (bool) $p->show_captions,
  232. 'show_license' => (bool) $p->show_license,
  233. 'show_location' => (bool) $p->show_location,
  234. 'show_timestamp' => (bool) $p->show_timestamp,
  235. 'show_link' => (bool) $p->show_link,
  236. 'show_avatar' => (bool) $p->show_avatar,
  237. 'show_bio' => (bool) $p->show_bio,
  238. 'profile_layout' => $p->profile_layout,
  239. 'profile_source' => $p->profile_source,
  240. 'show_profile_button' => $profileButton,
  241. 'rss_enabled' => $rssEnabled,
  242. 'show_rss_button' => $rssButton,
  243. ];
  244. if($rssEnabled) {
  245. $res['rss_feed_url'] = $p->permalink('.rss');
  246. }
  247. if($p->metadata) {
  248. if(isset($p->metadata['background_color'])) {
  249. $res['background_color'] = $p->metadata['background_color'];
  250. }
  251. if(isset($p->metadata['text_color'])) {
  252. $res['text_color'] = $p->metadata['text_color'];
  253. }
  254. }
  255. return $res;
  256. }
  257. public function storeSettings(Request $request)
  258. {
  259. abort_if(!$request->user(), 403);
  260. $this->validate($request, [
  261. 'active' => 'sometimes|boolean',
  262. 'show_captions' => 'sometimes|boolean',
  263. 'show_license' => 'sometimes|boolean',
  264. 'show_location' => 'sometimes|boolean',
  265. 'show_timestamp' => 'sometimes|boolean',
  266. 'show_link' => 'sometimes|boolean',
  267. 'show_avatar' => 'sometimes|boolean',
  268. 'show_bio' => 'sometimes|boolean',
  269. 'profile_layout' => 'sometimes|in:grid,masonry,album',
  270. 'profile_source' => 'sometimes|in:recent,custom',
  271. 'color_scheme' => 'sometimes|in:light,dark,custom',
  272. 'show_profile_button' => 'sometimes|boolean',
  273. 'rss_enabled' => 'sometimes|boolean',
  274. 'show_rss_button' => 'sometimes|boolean',
  275. 'feed_order' => 'sometimes|in:oldest,recent',
  276. 'background_color' => [
  277. 'sometimes',
  278. 'nullable',
  279. 'regex:/^#([a-f0-9]{6}|[a-f0-9]{3})$/i'
  280. ],
  281. 'text_color' => [
  282. 'sometimes',
  283. 'nullable',
  284. 'regex:/^#([a-f0-9]{6}|[a-f0-9]{3})$/i'
  285. ],
  286. ]);
  287. $res = Portfolio::whereUserId($request->user()->id)->firstOrFail();
  288. $pid = $request->user()->profile_id;
  289. $metadata = $res->metadata;
  290. $clearFeedCache = false;
  291. if($request->has('color_scheme')) {
  292. $metadata['color_scheme'] = $request->input('color_scheme');
  293. }
  294. if($request->has('background_color')) {
  295. $metadata['background_color'] = $request->input('background_color');
  296. $bgc = $request->background_color;
  297. if($bgc && $bgc !== '#000000') {
  298. $metadata['color_scheme'] = 'custom';
  299. }
  300. }
  301. if($request->has('text_color')) {
  302. $metadata['text_color'] = $request->input('text_color');
  303. $txc = $request->text_color;
  304. if($txc && $txc !== '#d4d4d8') {
  305. $metadata['color_scheme'] = 'custom';
  306. }
  307. }
  308. if($request->has('show_profile_button')) {
  309. $metadata['show_profile_button'] = $request->input('show_profile_button');
  310. }
  311. if($request->has('rss_enabled')) {
  312. $metadata['rss_enabled'] = $request->input('rss_enabled');
  313. }
  314. if($request->has('show_rss_button')) {
  315. $metadata['show_rss_button'] = $metadata['rss_enabled'] ? $request->input('show_rss_button') : false;
  316. }
  317. if($request->has('feed_order')) {
  318. $metadata['feed_order'] = $request->input('feed_order');
  319. }
  320. if(isset($metadata['background_color']) || isset($metadata['text_color'])) {
  321. $bgc = isset($metadata['background_color']) ? $metadata['background_color'] : null;
  322. $txc = isset($metadata['text_color']) ? $metadata['text_color'] : null;
  323. if((!$bgc || $bgc == '#000000') && (!$txc || $txc === '#d4d4d8') && $request->color_scheme != 'light') {
  324. $metadata['color_scheme'] = 'dark';
  325. }
  326. }
  327. if($request->has('color_scheme') && $request->color_scheme === 'light') {
  328. $metadata['background_color'] = '#ffffff';
  329. $metadata['text_color'] = '#000000';
  330. $metadata['color_scheme'] = 'light';
  331. }
  332. if($request->metadata !== $metadata) {
  333. $res->metadata = $metadata;
  334. $res->save();
  335. }
  336. if($request->profile_layout != $res->profile_layout) {
  337. $clearFeedCache = true;
  338. }
  339. $res->update($request->only([
  340. 'active',
  341. 'show_captions',
  342. 'show_license',
  343. 'show_location',
  344. 'show_timestamp',
  345. 'show_link',
  346. 'show_avatar',
  347. 'show_bio',
  348. 'profile_layout',
  349. 'profile_source'
  350. ]));
  351. Cache::forget(self::RECENT_FEED_KEY . $pid);
  352. if($clearFeedCache) {
  353. Cache::forget(self::RSS_FEED_KEY . $pid);
  354. }
  355. return 200;
  356. }
  357. public function storeCurated(Request $request)
  358. {
  359. abort_if(!$request->user(), 403);
  360. $this->validate($request, [
  361. 'ids' => 'required|array|max:100'
  362. ]);
  363. $pid = $request->user()->profile_id;
  364. $ids = $request->input('ids');
  365. Status::whereProfileId($pid)
  366. ->whereScope('public')
  367. ->whereIn('type', ['photo', 'photo:album'])
  368. ->findOrFail($ids);
  369. $p = Portfolio::whereProfileId($pid)->firstOrFail();
  370. $metadata = $p->metadata;
  371. $metadata['posts'] = $ids;
  372. $p->metadata = $metadata;
  373. $p->save();
  374. Cache::forget(self::RECENT_FEED_KEY . $pid);
  375. Cache::forget(self::RSS_FEED_KEY . $pid);
  376. Cache::forget(self::CACHED_FEED_KEY . $pid);
  377. return $request->ids;
  378. }
  379. public function getRssFeed(Request $request, $username)
  380. {
  381. $user = User::whereUsername($username)->first();
  382. if(!$user) {
  383. return view('portfolio.404');
  384. }
  385. $portfolio = Portfolio::whereUserId($user->id)->where('active', 1)->firstOrFail();
  386. $metadata = $portfolio->metadata;
  387. abort_if(!$metadata || !isset($metadata['rss_enabled']), 404);
  388. abort_unless($metadata['rss_enabled'], 404);
  389. $account = AccountService::get($user->profile_id);
  390. $portfolioUrl = $portfolio->url();
  391. $portfolioLayout = $portfolio->profile_layout;
  392. if(!isset($metadata['posts']) || !count($metadata['posts'])) {
  393. $feed = [];
  394. } else {
  395. $feed = Cache::remember(
  396. self::RSS_FEED_KEY . $user->profile_id,
  397. 43200,
  398. function() use($portfolio, $portfolioUrl, $portfolioLayout) {
  399. return collect($portfolio->metadata['posts'])->map(function($post) {
  400. return StatusService::get($post);
  401. })
  402. ->filter()
  403. ->values()
  404. ->map(function($post, $idx) use($portfolioLayout, $portfolioUrl) {
  405. $ts = now()->parse($post['created_at']);
  406. $url = $portfolioLayout == 'album' ? $portfolioUrl . '?slide=' . ($idx + 1) : $portfolioUrl . '/' . $post['id'];
  407. return [
  408. 'title' => 'Post by ' . $post['account']['username'] . ' on ' . $ts->format('D, d M Y'),
  409. 'description' => $post['content_text'],
  410. 'pubDate' => date('D, d M Y H:i:s ', strtotime($post['created_at'])) . 'GMT',
  411. 'url' => $url
  412. ];
  413. })
  414. ->reverse()
  415. ->take(10)
  416. ->toArray();
  417. }
  418. );
  419. }
  420. $now = date('D, d M Y H:i:s ') . 'GMT';
  421. return response()
  422. ->view('portfolio.rss_feed', compact('account', 'now', 'feed', 'portfolioUrl'), 200)
  423. ->header('Content-Type', 'text/xml');
  424. return response($feed)->withHeaders(['Content-Type' => 'text/xml']);
  425. }
  426. public function getApFeed(Request $request, $username)
  427. {
  428. $user = User::whereUsername($username)->first();
  429. if(!$user) {
  430. return view('portfolio.404');
  431. }
  432. $portfolio = Portfolio::whereUserId($user->id)->where('active', 1)->firstOrFail();
  433. $metadata = $portfolio->metadata;
  434. $baseUrl = config('app.url');
  435. $page = $request->input('page');
  436. $res = [
  437. '@context' => 'https://www.w3.org/ns/activitystreams',
  438. 'id' => $portfolio->permalink('.json'),
  439. 'type' => 'OrderedCollection',
  440. 'totalItems' => isset($metadata['posts']) ? count($metadata['posts']) : 0,
  441. ];
  442. if($request->has('page')) {
  443. $start = $page == 1 ? 0 : ($page * 10 - 10);
  444. $res['id'] = $portfolio->permalink('.json?page=' . $page);
  445. $res['type'] = 'OrderedCollectionPage';
  446. $res['next'] = $portfolio->permalink('.json?page=' . $page + 1);
  447. $res['partOf'] = $portfolio->permalink('.json');
  448. $res['orderedItems'] = collect($metadata['posts'])->slice($start)->take(10)->map(function($p) {
  449. return StatusService::get($p);
  450. })
  451. ->filter()
  452. ->map(function($p) {
  453. return $p['url'];
  454. })
  455. ->values();
  456. if(!$res['orderedItems'] || $res['orderedItems']->count() != 10) {
  457. unset($res['next']);
  458. }
  459. } else {
  460. $res['first'] = $portfolio->permalink('.json?page=1');
  461. }
  462. return response()->json($res, 200, [], JSON_UNESCAPED_SLASHES|JSON_PRETTY_PRINT)
  463. ->header('Content-Type', 'application/activity+json');
  464. }
  465. }