user(); $request = request(); if (!$user) { redirect(page_link_to('')); } $action = 'list'; if ($request->has('action')) { $action = $request->input('action'); } switch ($action) { case 'view': return user_controller(); case 'delete': return user_delete_controller(); case 'edit_vouchers': return user_edit_vouchers_controller(); case 'list': default: return users_list_controller(); } } /** * Delete a user, requires to enter own password for reasons. * * @return array */ function user_delete_controller() { $user = auth()->user(); $auth = auth(); $request = request(); if ($request->has('user_id')) { $user_source = User::find($request->query->get('user_id')); } else { $user_source = $user; } if (!auth()->can('admin_user')) { redirect(page_link_to('')); } // You cannot delete yourself if ($user->id == $user_source->id) { error(__('You cannot delete yourself.')); redirect(user_link($user->id)); } if ($request->hasPostData('submit')) { $valid = true; if (!( $request->has('password') && $auth->verifyPassword($user, $request->postData('password')) )) { $valid = false; error(__('Your password is incorrect. Please try it again.')); } if ($valid) { // Load data before user deletion to prevent errors when displaying $user_source->load(['contact', 'personalData', 'settings', 'state']); $user_source->delete(); mail_user_delete($user_source); success(__('User deleted.')); engelsystem_log(sprintf('Deleted %s', User_Nick_render($user_source, true))); redirect(users_link()); } } return [ sprintf(__('Delete %s'), $user_source->name), User_delete_view($user_source) ]; } /** * @return string */ function users_link() { return page_link_to('users'); } /** * @param int $userId * @return string */ function user_edit_link($userId) { return page_link_to('admin_user', ['user_id' => $userId]); } /** * @param int $userId * @return string */ function user_delete_link($userId) { return page_link_to('users', ['action' => 'delete', 'user_id' => $userId]); } /** * @param int $userId * @return string */ function user_link($userId) { return page_link_to('users', ['action' => 'view', 'user_id' => $userId]); } /** * @return array */ function user_edit_vouchers_controller() { $user = auth()->user(); $request = request(); if ($request->has('user_id')) { $user_source = User::find($request->input('user_id')); } else { $user_source = $user; } if (!auth()->can('admin_user')) { redirect(page_link_to('')); } if ($request->hasPostData('submit')) { $valid = true; $vouchers = ''; if ( $request->has('vouchers') && test_request_int('vouchers') && trim($request->input('vouchers')) >= 0 ) { $vouchers = trim($request->input('vouchers')); } else { $valid = false; error(__('Please enter a valid number of vouchers.')); } if ($valid) { $user_source->state->got_voucher = $vouchers; $user_source->state->save(); success(__('Saved the number of vouchers.')); engelsystem_log(User_Nick_render($user_source, true) . ': ' . sprintf('Got %s vouchers', $user_source->state->got_voucher)); redirect(user_link($user_source->id)); } } return [ sprintf(__('%s\'s vouchers'), $user_source->name), User_edit_vouchers_view($user_source) ]; } /** * @return array */ function user_controller() { $user = auth()->user(); $request = request(); $user_source = $user; if ($request->has('user_id')) { $user_source = User::find($request->input('user_id')); if (!$user_source) { error(__('User not found.')); redirect(page_link_to('/')); } } $shifts = Shifts_by_user($user_source->id, auth()->can('user_shifts_admin')); foreach ($shifts as &$shift) { // TODO: Move queries to model $shift['needed_angeltypes'] = DB::select(' SELECT DISTINCT `AngelTypes`.* FROM `ShiftEntry` JOIN `AngelTypes` ON `ShiftEntry`.`TID`=`AngelTypes`.`id` WHERE `ShiftEntry`.`SID` = ? ORDER BY `AngelTypes`.`name` ', [$shift['SID']] ); foreach ($shift['needed_angeltypes'] as &$needed_angeltype) { $needed_angeltype['users'] = DB::select(' SELECT `ShiftEntry`.`freeloaded`, `users`.* FROM `ShiftEntry` JOIN `users` ON `ShiftEntry`.`UID`=`users`.`id` WHERE `ShiftEntry`.`SID` = ? AND `ShiftEntry`.`TID` = ? ', [$shift['SID'], $needed_angeltype['id']] ); } } if (empty($user_source->api_key)) { User_reset_api_key($user_source, false); } if ($user_source->state->force_active) { $tshirt_score = __('Enough'); } else { $tshirt_score = sprintf('%.2f', User_tshirt_score($user_source->id)) . ' h'; } return [ $user_source->name, User_view( $user_source, auth()->can('admin_user'), User_is_freeloader($user_source), User_angeltypes($user_source->id), User_groups($user_source->id), $shifts, $user->id == $user_source->id, $tshirt_score, auth()->can('admin_active'), auth()->can('admin_user_worklog'), UserWorkLogsForUser($user_source->id) ) ]; } /** * List all users. * * @return array */ function users_list_controller() { $request = request(); if (!auth()->can('admin_user')) { redirect(page_link_to('')); } $order_by = 'name'; if ($request->has('OrderBy') && in_array($request->input('OrderBy'), [ 'name', 'last_name', 'first_name', 'dect', 'email', 'shirt_size', 'arrived', 'active', 'force_active', 'got_shirt', 'last_login_at', ])) { $order_by = $request->input('OrderBy'); } /** @var User[] $users */ $users = User::query() ->leftJoin('users_contact', 'users.id', '=', 'users_contact.user_id') ->leftJoin('users_personal_data', 'users.id', '=', 'users_personal_data.user_id') ->leftJoin('users_state', 'users.id', '=', 'users_state.user_id') ->orderBy($order_by) ->orderBy('name') ->get(); foreach ($users as $user) { $user->setAttribute('freeloads', count(ShiftEntries_freeloaded_by_user($user->id))); } return [ __('All users'), Users_view( $users, $order_by, State::whereArrived(true)->count(), State::whereActive(true)->count(), State::whereForceActive(true)->count(), ShiftEntries_freeloaded_count(), State::whereGotShirt(true)->count(), State::query()->sum('got_voucher') ) ]; } /** * Second step of password recovery: set a new password using the token link from email * * @return string */ function user_password_recovery_set_new_controller() { $request = request(); $passwordReset = PasswordReset::whereToken($request->input('token'))->first(); if (!$passwordReset) { error(__('Token is not correct.')); redirect(page_link_to('login')); } if ($request->hasPostData('submit')) { $valid = true; if ( $request->has('password') && strlen($request->postData('password')) >= config('min_password_length') ) { if ($request->postData('password') != $request->postData('password2')) { $valid = false; error(__('Your passwords don\'t match.')); } } else { $valid = false; error(__('Your password is to short (please use at least 6 characters).')); } if ($valid) { auth()->setPassword($passwordReset->user, $request->postData('password')); success(__('Password saved.')); $passwordReset->delete(); redirect(page_link_to('login')); } } return User_password_set_view(); } /** * First step of password recovery: display a form that asks for your email and send email with recovery link * * @return string */ function user_password_recovery_start_controller() { $request = request(); if ($request->hasPostData('submit')) { $valid = true; $user_source = null; if ($request->has('email') && strlen(strip_request_item('email')) > 0) { $email = strip_request_item('email'); if (check_email($email)) { /** @var User $user_source */ $user_source = User::whereEmail($email)->first(); if (!$user_source) { $valid = false; error(__('E-mail address is not correct.')); } } else { $valid = false; error(__('E-mail address is not correct.')); } } else { $valid = false; error(__('Please enter your e-mail.')); } if ($valid) { $token = User_generate_password_recovery_token($user_source); engelsystem_email_to_user( $user_source, __('Password recovery'), sprintf( __('Please visit %s to recover your password.'), page_link_to('user_password_recovery', ['token' => $token]) ) ); success(__('We sent an email containing your password recovery link.')); redirect(page_link_to('login')); } } return User_password_recovery_view(); } /** * User password recovery in 2 steps. * (By email) * * @return string */ function user_password_recovery_controller() { if (request()->has('token')) { return user_password_recovery_set_new_controller(); } return user_password_recovery_start_controller(); } /** * Menu title for password recovery. * * @return string */ function user_password_recovery_title() { return __('Password recovery'); } /** * Loads a user from param user_id. * * @return User */ function load_user() { $request = request(); if (!$request->has('user_id')) { redirect(page_link_to()); } $user = User::find($request->input('user_id')); if (!$user) { error(__('User doesn\'t exist.')); redirect(page_link_to()); } return $user; } /** * @param ShiftsFilter $shiftsFilter * @return ShiftCalendarRenderer */ function shiftCalendarRendererByShiftFilter(ShiftsFilter $shiftsFilter) { $shifts = Shifts_by_ShiftsFilter($shiftsFilter); $needed_angeltypes_source = NeededAngeltypes_by_ShiftsFilter($shiftsFilter); $shift_entries_source = ShiftEntries_by_ShiftsFilter($shiftsFilter); $needed_angeltypes = []; $shift_entries = []; foreach ($shifts as $shift) { $needed_angeltypes[$shift['SID']] = []; $shift_entries[$shift['SID']] = []; } foreach ($shift_entries_source as $shift_entry) { if (isset($shift_entries[$shift_entry['SID']])) { $shift_entries[$shift_entry['SID']][] = $shift_entry; } } foreach ($needed_angeltypes_source as $needed_angeltype) { if (isset($needed_angeltypes[$needed_angeltype['SID']])) { $needed_angeltypes[$needed_angeltype['SID']][] = $needed_angeltype; } } unset($needed_angeltypes_source); unset($shift_entries_source); if ( in_array(ShiftsFilter::FILLED_FREE, $shiftsFilter->getFilled()) && in_array(ShiftsFilter::FILLED_FILLED, $shiftsFilter->getFilled()) ) { return new ShiftCalendarRenderer($shifts, $needed_angeltypes, $shift_entries, $shiftsFilter); } $filtered_shifts = []; foreach ($shifts as $shift) { $needed_angels_count = 0; $taken = 0; foreach ($needed_angeltypes[$shift['SID']] as $needed_angeltype) { $taken = 0; if ( !in_array(ShiftsFilter::FILLED_FILLED, $shiftsFilter->getFilled()) && !in_array($needed_angeltype['angel_type_id'], $shiftsFilter->getTypes()) ) { continue; } foreach ($shift_entries[$shift['SID']] as $shift_entry) { if ( $needed_angeltype['angel_type_id'] == $shift_entry['TID'] && $shift_entry['freeloaded'] == 0 ) { $taken++; } } $needed_angels_count += max(0, $needed_angeltype['count'] - $taken); } if ( in_array(ShiftsFilter::FILLED_FREE, $shiftsFilter->getFilled()) && $taken < $needed_angels_count ) { $filtered_shifts[] = $shift; } if ( in_array(ShiftsFilter::FILLED_FILLED, $shiftsFilter->getFilled()) && $taken >= $needed_angels_count ) { $filtered_shifts[] = $shift; } } return new ShiftCalendarRenderer($filtered_shifts, $needed_angeltypes, $shift_entries, $shiftsFilter); }