2024-07-23 02:02:13 +02:00
|
|
|
<?php
|
|
|
|
|
|
|
|
namespace App\Http\Controllers;
|
|
|
|
|
2024-08-02 04:32:20 +02:00
|
|
|
use \Closure;
|
2024-07-23 02:02:13 +02:00
|
|
|
use App\Utils\Ninja;
|
2024-08-02 04:32:20 +02:00
|
|
|
use App\Models\Company;
|
2024-07-23 02:02:13 +02:00
|
|
|
use Illuminate\Support\Str;
|
|
|
|
use Illuminate\Http\Request;
|
|
|
|
use Illuminate\Http\Response;
|
|
|
|
use App\Utils\Traits\MakesHash;
|
|
|
|
use Illuminate\Support\Facades\Cache;
|
|
|
|
use App\Jobs\Import\QuickbooksIngest;
|
2024-08-02 04:32:20 +02:00
|
|
|
use Illuminate\Support\Facades\Validator;
|
2024-07-30 20:19:15 +02:00
|
|
|
use App\Services\Import\Quickbooks\Service as QuickbooksService;
|
2024-07-23 02:02:13 +02:00
|
|
|
|
|
|
|
class ImportQuickbooksController extends BaseController
|
|
|
|
{
|
2024-07-30 20:19:15 +02:00
|
|
|
protected QuickbooksService $service;
|
|
|
|
private $import_entities = [
|
|
|
|
'client' => 'Customer',
|
|
|
|
'invoice' => 'Invoice',
|
|
|
|
'product' => 'Item',
|
|
|
|
'payment' => 'Payment'
|
|
|
|
];
|
|
|
|
|
|
|
|
public function __construct(QuickbooksService $service) {
|
|
|
|
parent::__construct();
|
2024-08-02 04:32:20 +02:00
|
|
|
|
2024-07-30 20:19:15 +02:00
|
|
|
$this->service = $service;
|
2024-08-02 04:32:20 +02:00
|
|
|
$this->middleware(
|
|
|
|
function (Request $request, Closure $next) {
|
|
|
|
|
|
|
|
// Check for the required query parameters
|
|
|
|
if (!$request->has(['code', 'state', 'realmId'])) {
|
|
|
|
return abort(400,'Unauthorized');
|
|
|
|
}
|
|
|
|
|
|
|
|
$rules = [
|
|
|
|
'state' => [
|
|
|
|
'required',
|
|
|
|
'valid' => function ($attribute, $value, $fail) {
|
|
|
|
if (!Cache::has($value)) {
|
|
|
|
$fail('The state is invalid.');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
]
|
|
|
|
];
|
|
|
|
// Custom error messages
|
|
|
|
$messages = [
|
|
|
|
'state.required' => 'The state is required.',
|
|
|
|
'state.valid' => 'state token not valid'
|
|
|
|
];
|
|
|
|
// Perform the validation
|
|
|
|
$validator = Validator::make($request->all(), $rules, $messages);
|
|
|
|
if ($validator->fails()) {
|
|
|
|
// If validation fails, redirect back with errors and input
|
|
|
|
return redirect('/')
|
|
|
|
->withErrors($validator)
|
|
|
|
->withInput();
|
|
|
|
}
|
|
|
|
|
|
|
|
$token = Cache::pull($request->state);
|
|
|
|
$request->merge(['company' => Cache::get($token) ]);
|
|
|
|
|
|
|
|
return $next($request);
|
|
|
|
}
|
|
|
|
)->only('onAuthorized');
|
|
|
|
$this->middleware(
|
|
|
|
function ( Request $request, Closure $next) {
|
|
|
|
$rules = [
|
|
|
|
'token' => [
|
|
|
|
'required',
|
|
|
|
'valid' => function ($attribute, $value, $fail) {
|
|
|
|
if (!Cache::has($value) || (!Company::where('company_key', (Cache::get($value))['company_key'])->exists() )) {
|
|
|
|
$fail('The company is invalid.');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
]
|
|
|
|
];
|
|
|
|
// Custom error messages
|
|
|
|
$messages = [
|
|
|
|
'token.required' => 'The token is required.',
|
|
|
|
'token.valid' => 'Token note valid!'
|
|
|
|
];
|
|
|
|
// Perform the validation
|
|
|
|
$validator = Validator::make(['token' => $request->token ], $rules, $messages);
|
|
|
|
if ($validator->fails()) {
|
2024-08-08 18:53:45 +02:00
|
|
|
return redirect()
|
|
|
|
->back()
|
2024-08-02 04:32:20 +02:00
|
|
|
->withErrors($validator)
|
|
|
|
->withInput();
|
|
|
|
}
|
|
|
|
|
|
|
|
//If validation passes, proceed to the next middleware/controller
|
|
|
|
return $next($request);
|
|
|
|
}
|
|
|
|
)->only('authorizeQuickbooks');
|
|
|
|
}
|
|
|
|
|
2024-08-08 18:53:45 +02:00
|
|
|
public function onAuthorized(Request $request)
|
|
|
|
{
|
|
|
|
$realm = $request->query('realmId');
|
|
|
|
$company_key = $request->input('company.company_key');
|
|
|
|
$company_id = $request->input('company.id');
|
|
|
|
$tokens = ($auth_service = $this->service->getOAuth())->accessToken($request->query('code'), $realm);
|
|
|
|
$auth_service->saveTokens($company_key, ['realm' => $realm] + $tokens);
|
|
|
|
|
2024-08-02 04:32:20 +02:00
|
|
|
return response(200);
|
2024-08-08 18:53:45 +02:00
|
|
|
}
|
2024-07-30 20:19:15 +02:00
|
|
|
|
2024-07-23 02:02:13 +02:00
|
|
|
/**
|
|
|
|
* Determine if the user is authorized to make this request.
|
|
|
|
*
|
|
|
|
* @return bool
|
|
|
|
*/
|
2024-08-02 04:32:20 +02:00
|
|
|
public function authorizeQuickbooks(Request $request)
|
2024-07-23 02:02:13 +02:00
|
|
|
{
|
2024-08-02 04:32:20 +02:00
|
|
|
$token = $request->token;
|
|
|
|
$auth = $this->service->getOAuth();
|
|
|
|
$authorizationUrl = $auth->getAuthorizationUrl();
|
|
|
|
$state = $auth->getState();
|
|
|
|
|
2024-08-08 18:53:45 +02:00
|
|
|
Cache::put($state, $token, 190);
|
2024-07-23 02:02:13 +02:00
|
|
|
|
2024-08-02 04:32:20 +02:00
|
|
|
return redirect()->to($authorizationUrl);
|
2024-07-23 02:02:13 +02:00
|
|
|
}
|
|
|
|
|
2024-08-08 18:53:45 +02:00
|
|
|
public function preimport(string $type, string $hash)
|
2024-07-23 02:02:13 +02:00
|
|
|
{
|
|
|
|
// Check for authorization otherwise
|
|
|
|
// Create a reference
|
|
|
|
$data = [
|
2024-07-30 20:19:15 +02:00
|
|
|
'hash' => $hash,
|
2024-08-08 18:53:45 +02:00
|
|
|
'type' => $type
|
2024-07-23 02:02:13 +02:00
|
|
|
];
|
2024-07-30 20:19:15 +02:00
|
|
|
$this->getData($data);
|
2024-07-23 02:02:13 +02:00
|
|
|
}
|
|
|
|
|
2024-07-30 20:19:15 +02:00
|
|
|
protected function getData($data) {
|
2024-07-23 02:02:13 +02:00
|
|
|
|
2024-07-30 20:19:15 +02:00
|
|
|
$entity = $this->import_entities[$data['type']];
|
|
|
|
$cache_name = "{$data['hash']}-{$data['type']}";
|
|
|
|
// TODO: Get or put cache or DB?
|
2024-08-08 18:53:45 +02:00
|
|
|
if(! Cache::has($cache_name) )
|
2024-07-30 20:19:15 +02:00
|
|
|
{
|
2024-08-08 18:53:45 +02:00
|
|
|
$contents = call_user_func([$this->service, "fetch{$entity}s"]);
|
|
|
|
if($contents->isEmpty()) return;
|
|
|
|
|
2024-07-30 20:19:15 +02:00
|
|
|
Cache::put($cache_name, base64_encode( $contents->toJson()), 600);
|
|
|
|
}
|
2024-07-23 02:02:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @OA\Post(
|
|
|
|
* path="/api/v1/import_json",
|
|
|
|
* operationId="getImportJson",
|
|
|
|
* tags={"import"},
|
|
|
|
* summary="Import data from the system",
|
|
|
|
* description="Import data from the system",
|
|
|
|
* @OA\Parameter(ref="#/components/parameters/X-Requested-With"),
|
|
|
|
* @OA\Response(
|
|
|
|
* response=200,
|
|
|
|
* description="success",
|
|
|
|
* @OA\Header(header="X-MINIMUM-CLIENT-VERSION", ref="#/components/headers/X-MINIMUM-CLIENT-VERSION"),
|
|
|
|
* @OA\Header(header="X-RateLimit-Remaining", ref="#/components/headers/X-RateLimit-Remaining"),
|
|
|
|
* @OA\Header(header="X-RateLimit-Limit", ref="#/components/headers/X-RateLimit-Limit"),
|
|
|
|
* ),
|
|
|
|
* @OA\Response(
|
|
|
|
* response=422,
|
|
|
|
* description="Validation error",
|
|
|
|
* @OA\JsonContent(ref="#/components/schemas/ValidationError"),
|
|
|
|
* ),
|
|
|
|
* @OA\Response(
|
|
|
|
* response="default",
|
|
|
|
* description="Unexpected Error",
|
|
|
|
* @OA\JsonContent(ref="#/components/schemas/Error"),
|
|
|
|
* ),
|
|
|
|
* )
|
|
|
|
*/
|
|
|
|
public function import(Request $request)
|
|
|
|
{
|
2024-08-08 18:53:45 +02:00
|
|
|
$hash = Str::random(32);
|
|
|
|
foreach($request->input('import_types') as $type)
|
|
|
|
{
|
|
|
|
$this->preimport($type, $hash);
|
|
|
|
}
|
2024-07-23 02:02:13 +02:00
|
|
|
/** @var \App\Models\User $user */
|
2024-08-08 18:53:45 +02:00
|
|
|
$user = auth()->user() ?? Auth::loginUsingId(60);
|
|
|
|
$data = ['import_types' => $request->input('import_types') ] + compact('hash');
|
2024-07-23 02:02:13 +02:00
|
|
|
if (Ninja::isHosted()) {
|
2024-08-08 18:53:45 +02:00
|
|
|
QuickbooksIngest::dispatch( $data , $user->company() );
|
2024-07-23 02:02:13 +02:00
|
|
|
} else {
|
2024-08-08 18:53:45 +02:00
|
|
|
QuickbooksIngest::dispatch($data, $user->company() );
|
2024-07-23 02:02:13 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
return response()->json(['message' => 'Processing'], 200);
|
|
|
|
}
|
|
|
|
}
|