mirror of
https://github.com/snipe/snipe-it.git
synced 2025-01-11 13:57:41 -08:00
8c1cd87831
Signed-off-by: snipe <snipe@snipe.net>
325 lines
13 KiB
PHP
325 lines
13 KiB
PHP
<?php
|
|
|
|
namespace App\Http\Controllers\Api;
|
|
|
|
use App\Http\Controllers\Controller;
|
|
use App\Http\Transformers\LoginAttemptsTransformer;
|
|
use App\Models\Setting;
|
|
use App\Notifications\MailTest;
|
|
use App\Services\LdapAd;
|
|
use Illuminate\Http\JsonResponse;
|
|
use Illuminate\Http\Request;
|
|
use Illuminate\Http\Response;
|
|
use Illuminate\Support\Facades\DB;
|
|
use Illuminate\Support\Facades\Log;
|
|
use Illuminate\Support\Facades\Notification;
|
|
use GuzzleHttp\Client;
|
|
use Illuminate\Support\Facades\Storage;
|
|
use Illuminate\Support\Facades\Validator;
|
|
use App\Models\Ldap; // forward-port of v4 LDAP model for Sync
|
|
use App\Http\Requests\SlackSettingsRequest;
|
|
|
|
|
|
class SettingsController extends Controller
|
|
{
|
|
|
|
/**
|
|
* Test the ldap settings
|
|
*
|
|
* @author Wes Hulette <jwhulette@gmail.com>
|
|
*
|
|
* @since 5.0.0
|
|
*
|
|
* @param App\Models\LdapAd $ldap
|
|
*
|
|
* @return \Illuminate\Http\JsonResponse
|
|
*/
|
|
public function ldapAdSettingsTest(LdapAd $ldap): JsonResponse
|
|
{
|
|
if(!$ldap->init()) {
|
|
Log::info('LDAP is not enabled so we cannot test.');
|
|
return response()->json(['message' => 'LDAP is not enabled, cannot test.'], 400);
|
|
}
|
|
|
|
// The connect, bind and resulting users message
|
|
$message = [];
|
|
|
|
|
|
// This is all kinda fucked right now. The connection test doesn't actually do what you think,
|
|
// // and the way we parse the errors
|
|
// on the JS side is horrible.
|
|
Log::info('Preparing to test LDAP user login');
|
|
// Test user can connect to the LDAP server
|
|
try {
|
|
$ldap->testLdapAdUserConnection();
|
|
$message['login'] = [
|
|
'message' => 'Successfully connected to LDAP server.'
|
|
];
|
|
} catch (\Exception $ex) {
|
|
\Log::debug('Connection to LDAP server '.Setting::getSettings()->ldap_server.' failed. Please check your LDAP settings and try again. Server Responded with error: ' . $ex->getMessage());
|
|
return response()->json(
|
|
['message' => 'Connection to LDAP server '.Setting::getSettings()->ldap_server." failed. Verify that the LDAP hostname is entered correctly and that it can be reached from this web server. \n\nServer Responded with error: " . $ex->getMessage()
|
|
|
|
], 400);
|
|
}
|
|
|
|
Log::info('Preparing to test LDAP bind connection');
|
|
// Test user can bind to the LDAP server
|
|
try {
|
|
Log::info('Testing Bind');
|
|
$ldap->testLdapAdBindConnection();
|
|
$message['bind'] = [
|
|
'message' => 'Successfully bound to LDAP server.'
|
|
];
|
|
} catch (\Exception $ex) {
|
|
Log::info('LDAP Bind failed');
|
|
return response()->json(['message' => 'Connection to LDAP successful, but we were unable to Bind the LDAP user '.Setting::getSettings()->ldap_uname.". Verify your that your LDAP Bind username and password are correct. \n\nServer Responded with error: " . $ex->getMessage()
|
|
], 400);
|
|
}
|
|
|
|
|
|
Log::info('Preparing to get sample user set from LDAP directory');
|
|
// Get a sample of 10 users so user can verify the data is correct
|
|
$settings = Setting::getSettings();
|
|
try {
|
|
Log::info('Testing LDAP sync');
|
|
error_reporting(E_ALL & ~E_DEPRECATED); // workaround for php7.4, which deprecates ldap_control_paged_result
|
|
// $users = $ldap->testUserImportSync(); // from AdLdap2 from v5, disabling and falling back to v4's sync code
|
|
$users = collect(Ldap::findLdapUsers())->slice(0, 11)->filter(function ($value, $key) { //choosing ELEVEN because one is going to be the count, which we're about to filter out in the next line
|
|
return is_int($key);
|
|
})->map(function ($item) use ($settings) {
|
|
return (object) [
|
|
'username' => $item[$settings['ldap_username_field']][0] ?? null,
|
|
'employee_number' => $item[$settings['ldap_emp_num']][0] ?? null,
|
|
'lastname' => $item[$settings['ldap_lname_field']][0] ?? null,
|
|
'firstname' => $item[$settings['ldap_fname_field']][0] ?? null,
|
|
'email' => $item[$settings['ldap_email']][0] ?? null,
|
|
];
|
|
});
|
|
if ($users->count() > 0) {
|
|
$message['user_sync'] = [
|
|
'users' => $users
|
|
];
|
|
} else {
|
|
$message['user_sync'] = [
|
|
'message' => 'Connection to LDAP was successful, however there were no users returned from your query. You should confirm the Base Bind DN above.'
|
|
];
|
|
return response()->json($message, 400);
|
|
}
|
|
|
|
} catch (\Exception $ex) {
|
|
Log::info('LDAP sync failed');
|
|
$message['user_sync'] = [
|
|
'message' => 'Error getting users from LDAP directory, error: ' . $ex->getMessage()
|
|
];
|
|
return response()->json($message, 400);
|
|
}
|
|
|
|
return response()->json($message, 200);
|
|
}
|
|
|
|
public function ldaptestlogin(Request $request, LdapAd $ldap)
|
|
{
|
|
|
|
if (Setting::getSettings()->ldap_enabled!='1') {
|
|
\Log::debug('LDAP is not enabled. Cannot test.');
|
|
return response()->json(['message' => 'LDAP is not enabled, cannot test.'], 400);
|
|
}
|
|
|
|
|
|
$rules = array(
|
|
'ldaptest_user' => 'required',
|
|
'ldaptest_password' => 'required'
|
|
);
|
|
|
|
$validator = Validator::make($request->all(), $rules);
|
|
if ($validator->fails()) {
|
|
\Log::debug('LDAP Validation test failed.');
|
|
$validation_errors = implode(' ',$validator->errors()->all());
|
|
return response()->json(['message' => $validator->errors()->all()], 400);
|
|
}
|
|
|
|
|
|
\Log::debug('Preparing to test LDAP login');
|
|
try {
|
|
DB::beginTransaction(); //this was the easiest way to invoke a full test of an LDAP login without adding new users to the DB (which may not be desired)
|
|
|
|
// $results = $ldap->ldap->auth()->attempt($request->input('ldaptest_username'), $request->input('ldaptest_password'), true);
|
|
// can't do this because that's a protected property.
|
|
|
|
$results = $ldap->ldapLogin($request->input('ldaptest_user'), $request->input('ldaptest_password')); // this would normally create a user on success (if they didn't already exist), but for the transaction
|
|
if($results) {
|
|
return response()->json(['message' => 'It worked! '. $request->input('ldaptest_user').' successfully binded to LDAP.'], 200);
|
|
} else {
|
|
return response()->json(['message' => 'Login Failed. '. $request->input('ldaptest_user').' did not successfully bind to LDAP.'], 400);
|
|
}
|
|
} catch (\Exception $e) {
|
|
\Log::debug('Connection failed');
|
|
return response()->json(['message' => $e->getMessage()], 400);
|
|
} finally {
|
|
DB::rollBack(); // ALWAYS rollback, whether success or failure
|
|
}
|
|
|
|
|
|
}
|
|
|
|
public function slacktest(SlackSettingsRequest $request)
|
|
{
|
|
|
|
\Log::error(strncmp($request->input('slack_endpoint'), 'https://hooks.slack.com/', 24));
|
|
\Log::error('Endpoint: '.$request->input('slack_endpoint'));
|
|
\Log::error('Request: '.print_r($request->all(), true));
|
|
\Log::error('Rules: '.print_r(Setting::rules(), true));
|
|
|
|
|
|
|
|
// if (($request->filled('slack_endpoint')) && ((strncmp($request->input('slack_endpoint'), "https://hooks.slack.com/", 24) !== 0))) {
|
|
// return response()->json(['message' => 'Oops! Please check the endpoint URL. Slack endpoints should start with https://hooks.slack.com. Your current endpoint is '.$request->input('slack_endpoint'), ['errors' => 'Invalid slack endpoint.']], 422);
|
|
|
|
// }
|
|
|
|
// if (($request->filled('slack_channel')) && (strpos($request->input('slack_channel'), '#') !== 0)) {
|
|
// return response()->json(['message' => 'Oops! Please check the channel name. Slack channels should begin with #.', ['errors' => 'Invalid slack endpoint.']], 422);
|
|
// }
|
|
|
|
|
|
// if (!$validator = Validator::make($request->all(), [
|
|
// 'slack_endpoint' => 'url"',
|
|
// 'slack_channel' => 'required_with:slack_endpoint|starts_with:#|nullable',
|
|
// ])) {
|
|
|
|
\Log::error('Fails validation with '.$request->input('slack_endpoint'));
|
|
|
|
|
|
// Only attempt the slack request if the validation passes
|
|
// if (!$request->validate([
|
|
// 'slack_endpoint' => 'url|required_with:slack_channel|starts_with:https://hooks.slack.com|nullable',
|
|
// 'slack_channel' => 'required_with:slack_endpoint|starts_with:#|nullable',
|
|
// ])) {
|
|
// return response()->json(['message' => $validator->errors()], 422);
|
|
//} else {
|
|
|
|
\Log::error('Passes validation with '.$request->input('slack_endpoint'));
|
|
|
|
$slack = new Client([
|
|
'base_url' => e($request->input('slack_endpoint')),
|
|
'defaults' => [
|
|
'exceptions' => false,
|
|
],
|
|
]);
|
|
|
|
$payload = json_encode(
|
|
[
|
|
'channel' => e($request->input('slack_channel')),
|
|
'text' => trans('general.slack_test_msg'),
|
|
'username' => e($request->input('slack_botname')),
|
|
'icon_emoji' => ':heart:',
|
|
]);
|
|
|
|
try {
|
|
$slack->post($request->input('slack_endpoint'), ['body' => $payload]);
|
|
return response()->json(['message' => 'Success'], 200);
|
|
|
|
} catch (\Exception $e) {
|
|
return response()->json(['message' => 'Oops! Please check the channel name and webhook endpoint URL. Slack responded with: '.$e->getMessage()], 400);
|
|
}
|
|
|
|
//}
|
|
return response()->json(['message' => 'Something went wrong :( '], 400);
|
|
}
|
|
|
|
|
|
/**
|
|
* Test the email configuration
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v3.0]
|
|
* @return Redirect
|
|
*/
|
|
public function ajaxTestEmail()
|
|
{
|
|
if (!config('app.lock_passwords')) {
|
|
try {
|
|
Notification::send(Setting::first(), new MailTest());
|
|
return response()->json(['message' => 'Mail sent to '.config('mail.reply_to.address')], 200);
|
|
} catch (\Exception $e) {
|
|
return response()->json(['message' => $e->getMessage()], 500);
|
|
}
|
|
}
|
|
return response()->json(['message' => 'Mail would have been sent, but this application is in demo mode! '], 200);
|
|
|
|
}
|
|
|
|
|
|
/**
|
|
* Delete server-cached barcodes
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v5.0.0]
|
|
* @return Response
|
|
*/
|
|
public function purgeBarcodes()
|
|
{
|
|
|
|
$file_count = 0;
|
|
$files = Storage::disk('public')->files('barcodes');
|
|
|
|
foreach ($files as $file) { // iterate files
|
|
|
|
$file_parts = explode(".", $file);
|
|
$extension = end($file_parts);
|
|
\Log::debug($extension);
|
|
|
|
// Only generated barcodes would have a .png file extension
|
|
if ($extension =='png') {
|
|
|
|
\Log::debug('Deleting: '.$file);
|
|
|
|
|
|
try {
|
|
Storage::disk('public')->delete($file);
|
|
\Log::debug('Deleting: '.$file);
|
|
$file_count++;
|
|
} catch (\Exception $e) {
|
|
\Log::debug($e);
|
|
}
|
|
}
|
|
|
|
}
|
|
|
|
return response()->json(['message' => 'Deleted '.$file_count.' barcodes'], 200);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
* Get a list of login attempts
|
|
*
|
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
|
* @since [v5.0.0]
|
|
* @param \Illuminate\Http\Request $request
|
|
* @return array
|
|
*/
|
|
public function showLoginAttempts(Request $request)
|
|
{
|
|
$allowed_columns = ['id', 'username', 'remote_ip', 'user_agent','successful','created_at'];
|
|
|
|
$login_attempts = DB::table('login_attempts');
|
|
$order = $request->input('order') === 'asc' ? 'asc' : 'desc';
|
|
$sort = in_array($request->get('sort'), $allowed_columns) ? $request->get('sort') : 'created_at';
|
|
|
|
$total = $login_attempts->count();
|
|
$login_attempts->orderBy($sort, $order);
|
|
$login_attempt_results = $login_attempts->skip(request('offset', 0))->take(request('limit', 20))->get();
|
|
|
|
return (new LoginAttemptsTransformer)->transformLoginAttempts($login_attempt_results, $total);
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|