Fixes for RandomDataSeeder (#3073)

* Provide failsafe creation of invoice invitations

* URL Links for invitations

* open up route for invitations

* Set DB by Invite

* Set DB By invitation Key

* Tests for setting DB based on user email address

* Middleware for setting db by email address

* fixes for tets

* fixes for tests

* Tests for bulk actions

* Payments API

* Fixes for tests
This commit is contained in:
David Bomba 2019-11-16 14:12:29 +11:00 committed by GitHub
parent b3262b00b7
commit 81c481c071
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
27 changed files with 436 additions and 39 deletions

View File

@ -23,7 +23,7 @@ DB_USERNAME2=ninja
DB_PASSWORD2=ninja DB_PASSWORD2=ninja
DB_PORT2=3306 DB_PORT2=3306
DEMO_MODE=false
BROADCAST_DRIVER=log BROADCAST_DRIVER=log
LOG_CHANNEL=stack LOG_CHANNEL=stack

View File

@ -80,7 +80,7 @@ class Handler extends ExceptionHandler
} }
else if($exception instanceof FatalThrowableError) else if($exception instanceof FatalThrowableError)
{ {
return response()->json(['message'=>'Fatal error', 500]); return response()->json(['message'=>'Fatal error'], 500);
} }
else if($exception instanceof AuthorizationException) else if($exception instanceof AuthorizationException)
{ {

View File

@ -29,10 +29,12 @@ class InvitationController extends Controller
use MakesHash; use MakesHash;
use MakesDates; use MakesDates;
public function invoiceRouter(string $invitation_key) public function router(string $entity, string $invitation_key)
{ {
$key = $entity.'_id';
$entity_obj = ucfirst($entity).'Invitation';
$invitation = InvoiceInvitation::whereRaw("BINARY `invitation_key`= ?", [$invitation_key])->first(); $invitation = $entity_obj::whereRaw("BINARY `key`= ?", [$invitation_key])->first();
if($invitation){ if($invitation){
@ -41,7 +43,7 @@ class InvitationController extends Controller
$invitation->markViewed(); $invitation->markViewed();
return redirect()->route('client.invoice.show', ['invoice' => $this->encodePrimaryKey($invitation->invoice_id)]); return redirect()->route('client.'.$entity.'.show', [$entity => $this->encodePrimaryKey($invitation->{$key})]);
} }
else else
@ -49,9 +51,9 @@ class InvitationController extends Controller
} }
// public function invoiceRouterForIframe(string $client_hash, string $invitation_key) public function routerForIframe(string $entity, string $client_hash, string $invitation_key)
// { {
// } }
} }

View File

@ -6,6 +6,8 @@
* @OA\Property(property="id", type="string", example="WJxbojagwO", description="The company hash id"), * @OA\Property(property="id", type="string", example="WJxbojagwO", description="The company hash id"),
* @OA\Property(property="size_id", type="string", example="1", description="The company size ID"), * @OA\Property(property="size_id", type="string", example="1", description="The company size ID"),
* @OA\Property(property="industry_id", type="string", example="1", description="The company industry ID"), * @OA\Property(property="industry_id", type="string", example="1", description="The company industry ID"),
* @OA\Property(property="portal_mode", type="string", example="subdomain", description="Determines the client facing urls ie: subdomain,domain,iframe"),
* @OA\Property(property="portal_domain", type="string", example="https://subdomain.invoicing.co", description="The fully qualified domain for client facing URLS"),
* @OA\Property(property="enabled_tax_rates", type="integer", example="1", description="Number of taxes rates used per entity"), * @OA\Property(property="enabled_tax_rates", type="integer", example="1", description="Number of taxes rates used per entity"),
* @OA\Property(property="fill_products", type="boolean", example=true, description="Toggles filling a product description based on product key"), * @OA\Property(property="fill_products", type="boolean", example=true, description="Toggles filling a product description based on product key"),
* @OA\Property(property="convert_products", type="boolean", example=true, description="___________"), * @OA\Property(property="convert_products", type="boolean", example=true, description="___________"),

View File

@ -603,11 +603,11 @@ class PaymentController extends BaseController
switch ($action) { switch ($action) {
case 'clone_to_invoice': case 'clone_to_invoice':
$payment = CloneInvoiceFactory::create($payment, auth()->user()->id); //$payment = CloneInvoiceFactory::create($payment, auth()->user()->id);
return $this->itemResponse($payment); //return $this->itemResponse($payment);
break; break;
case 'clone_to_quote': case 'clone_to_quote':
$quote = CloneInvoiceToQuoteFactory::create($payment, auth()->user()->id); //$quote = CloneInvoiceToQuoteFactory::create($payment, auth()->user()->id);
// todo build the quote transformer and return response here // todo build the quote transformer and return response here
break; break;
case 'history': case 'history':

View File

@ -106,6 +106,8 @@ class Kernel extends HttpKernel
'contact_token_auth' => \App\Http\Middleware\ContactTokenAuth::class, 'contact_token_auth' => \App\Http\Middleware\ContactTokenAuth::class,
'contact_db' => \App\Http\Middleware\ContactSetDb::class, 'contact_db' => \App\Http\Middleware\ContactSetDb::class,
'domain_db' => \App\Http\Middleware\SetDomainNameDb::class, 'domain_db' => \App\Http\Middleware\SetDomainNameDb::class,
'email_db' => \App\Http\Middleware\SetEmailDb::class,
'invite_db' => \App\Http\Middleware\SetInviteDb::class,
'password_protected' => \App\Http\Middleware\PasswordProtection::class, 'password_protected' => \App\Http\Middleware\PasswordProtection::class,
'signed' => \Illuminate\Routing\Middleware\ValidateSignature::class, 'signed' => \Illuminate\Routing\Middleware\ValidateSignature::class,
'portal_enabled' => \App\Http\Middleware\ClientPortalEnabled::class, 'portal_enabled' => \App\Http\Middleware\ClientPortalEnabled::class,

View File

@ -37,7 +37,7 @@ class ApiSecretCheck
return response() return response()
->json(json_encode($error, JSON_PRETTY_PRINT) ,403) ->json(json_encode($error, JSON_PRETTY_PRINT) ,403)
->header('X-App-Version', config('ninja.app_version')) ->header('X-App-Version', config('ninja.app_version'))
->header('X-API-VERSION', config('ninja.api_version')); ->header('X-Api-Version', config('ninja.api_version'));
} }

View File

@ -0,0 +1,58 @@
<?php
/**
* Invoice Ninja (https://invoiceninja.com)
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2019. Invoice Ninja LLC (https://invoiceninja.com)
*
* @license https://opensource.org/licenses/AAL
*/
namespace App\Http\Middleware;
use App\Libraries\MultiDB;
use App\Models\CompanyToken;
use Closure;
class SetEmailDb
{
/**
* Handle an incoming request.
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @return mixed
*/
public function handle($request, Closure $next)
{
$error = [
'message' => 'Email not set or not found',
'errors' => []
];
if( $request->input('email') && config('ninja.db.multi_db_enabled'))
{
if(! MultiDB::userFindAndSetDb($request->input('email')))
{
return response()->json($error, 403);
}
}
else {
return response()->json($error, 403);
}
return $next($request);
}
}

View File

@ -0,0 +1,49 @@
<?php
/**
* Invoice Ninja (https://invoiceninja.com)
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2019. Invoice Ninja LLC (https://invoiceninja.com)
*
* @license https://opensource.org/licenses/AAL
*/
namespace App\Http\Middleware;
use App\Libraries\MultiDB;
use Closure;
class SetInviteDb
{
/**
* Handle an incoming request.
*
* @param \Illuminate\Http\Request $request
* @param \Closure $next
* @return mixed
*/
public function handle($request, Closure $next)
{
$error = [
'message' => 'Invalid URL',
'errors' => []
];
/*
* Use the host name to set the active DB
**/
if( $request->getSchemeAndHttpHost() && config('ninja.db.multi_db_enabled') && ! MultiDB::findAndSetDbByInvitation($request->route('entity'),$request->route('invitation_key')))
{
if(request()->json)
return response()->json(json_encode($error, JSON_PRETTY_PRINT) ,403);
else
abort(404);
}
return $next($request);
}
}

View File

@ -13,9 +13,11 @@ namespace App\Http\Requests\Payment;
use App\Http\Requests\Request; use App\Http\Requests\Request;
use App\Models\Payment; use App\Models\Payment;
use App\Utils\Traits\MakesHash;
class StorePaymentRequest extends Request class StorePaymentRequest extends Request
{ {
use MakesHash;
/** /**
* Determine if the user is authorized to make this request. * Determine if the user is authorized to make this request.
* *
@ -29,6 +31,8 @@ class StorePaymentRequest extends Request
public function rules() public function rules()
{ {
$this->sanitize();
return [ return [
'documents' => 'mimes:png,ai,svg,jpeg,tiff,pdf,gif,psd,txt,doc,xls,ppt,xlsx,docx,pptx', 'documents' => 'mimes:png,ai,svg,jpeg,tiff,pdf,gif,psd,txt,doc,xls,ppt,xlsx,docx,pptx',
'client_id' => 'integer|nullable', 'client_id' => 'integer|nullable',
@ -41,7 +45,15 @@ class StorePaymentRequest extends Request
public function sanitize() public function sanitize()
{ {
//do post processing of Payment request here, ie. Payment_items
$input = $this->all();
if(isset($input['invoices']))
$input['invoices'] = $this->transformKeys(array_column($input['invoices']), 'id');
$this->replace($input);
return $this->all();
} }
public function messages() public function messages()

View File

@ -12,11 +12,16 @@
namespace App\Http\Requests\Quote; namespace App\Http\Requests\Quote;
use App\Http\Requests\Request; use App\Http\Requests\Request;
use App\Utils\Traits\CleanLineItems;
use App\Utils\Traits\MakesHash;
use Illuminate\Support\Facades\Log; use Illuminate\Support\Facades\Log;
use Illuminate\Validation\Rule; use Illuminate\Validation\Rule;
class UpdateQuoteRequest extends Request class UpdateQuoteRequest extends Request
{ {
use MakesHash;
use CleanLineItems;
/** /**
* Determine if the user is authorized to make this request. * Determine if the user is authorized to make this request.
* *
@ -33,10 +38,26 @@ class UpdateQuoteRequest extends Request
public function rules() public function rules()
{ {
$this->sanitize();
return [ return [
'documents' => 'mimes:png,ai,svg,jpeg,tiff,pdf,gif,psd,txt,doc,xls,ppt,xlsx,docx,pptx', 'documents' => 'mimes:png,ai,svg,jpeg,tiff,pdf,gif,psd,txt,doc,xls,ppt,xlsx,docx,pptx',
'client_id' => 'required|integer',
]; ];
} }
public function sanitize()
{
$input = $this->all();
// if(isset($input['client_id']))
// $input['client_id'] = $this->decodePrimaryKey($input['client_id']);
if(isset($input['line_items']))
$input['line_items'] = isset($input['line_items']) ? $this->cleanItems($input['line_items']) : [];
$this->replace($input);
return $this->all();
}
} }

View File

@ -0,0 +1,67 @@
<?php
/**
* Invoice Ninja (https://invoiceninja.com)
*
* @link https://github.com/invoiceninja/invoiceninja source repository
*
* @copyright Copyright (c) 2019. Invoice Ninja LLC (https://invoiceninja.com)
*
* @license https://opensource.org/licenses/AAL
*/
namespace App\Jobs\Invoice;
use App\Factory\InvoiceInvitationFactory;
use App\Models\Invoice;
use App\Models\InvoiceInvitation;
use Illuminate\Bus\Queueable;
use Illuminate\Contracts\Queue\ShouldQueue;
use Illuminate\Foundation\Bus\Dispatchable;
use Illuminate\Queue\InteractsWithQueue;
use Illuminate\Queue\SerializesModels;
use Symfony\Component\Debug\Exception\FatalThrowableError;
class CreateInvoiceInvitations implements ShouldQueue
{
use Dispatchable, InteractsWithQueue, Queueable, SerializesModels;
public $invoice;
/**
* Create a new job instance.
*
* @return void
*/
public function __construct(Invoice $invoice)
{
$this->invoice = $invoice;
}
public function handle()
{
$contacts = $this->invoice->client->contacts;
$contacts->each(function ($contact) {
$invitation = InvoiceInvitation::whereCompanyId($this->invoice->company_id)
->whereClientContactId($contact->id)
->whereInvoiceId($this->invoice->id)
->first();
if(!$invitation && $contact->send_invoice) {
$ii = InvoiceInvitationFactory::create($this->invoice->company_id, $this->invoice->user_id);
$ii->invoice_id = $this->invoice->id;
$ii->client_contact_id = $contact->id;
$ii->save();
}
else if($invitation && !$contact->send_invoice) {
$invitation->delete();
}
});
}
}

View File

@ -123,6 +123,21 @@ class MultiDB
} }
public static function userFindAndSetDb($email) : bool
{
//multi-db active
foreach (self::$dbs as $db)
{
if(User::on($db)->where(['email' => $email])->get()->count() >=1) // if user already exists, validation will fail
return true;
}
return false;
}
public static function findAndSetDb($token) :bool public static function findAndSetDb($token) :bool
{ {
@ -161,6 +176,21 @@ class MultiDB
} }
public static function findAndSetDbByInvitation($entity, $invitation_key)
{
$entity.'Invitation';
foreach (self::$dbs as $db)
{
if($invite = $entity::on($db)->whereKey($invitation_key)->first())
{
self::setDb($db);
return true;
}
}
return false;
}
/** /**
* @param $database * @param $database
*/ */

View File

@ -12,6 +12,7 @@
namespace App\Models; namespace App\Models;
use App\Models\Invoice; use App\Models\Invoice;
use App\Utils\Traits\Inviteable;
use App\Utils\Traits\MakesDates; use App\Utils\Traits\MakesDates;
use Illuminate\Database\Eloquent\Model; use Illuminate\Database\Eloquent\Model;
use Illuminate\Database\Eloquent\SoftDeletes; use Illuminate\Database\Eloquent\SoftDeletes;
@ -22,7 +23,8 @@ class InvoiceInvitation extends BaseModel
use MakesDates; use MakesDates;
use SoftDeletes; use SoftDeletes;
use Inviteable;
protected $fillable = [ protected $fillable = [
'id', 'id',
'client_contact_id', 'client_contact_id',
@ -79,11 +81,6 @@ class InvoiceInvitation extends BaseModel
return $this->invitation_key; return $this->invitation_key;
} }
public function getLink()
{
}
public function markViewed() public function markViewed()
{ {
$this->viewed_date = Carbon::now(); $this->viewed_date = Carbon::now();

View File

@ -73,6 +73,12 @@ class Quote extends BaseModel
return $this->belongsTo(User::class); return $this->belongsTo(User::class);
} }
public function client()
{
return $this->belongsTo(Client::class);
}
public function assigned_user() public function assigned_user()
{ {
return $this->belongsTo(User::class ,'assigned_user_id', 'id'); return $this->belongsTo(User::class ,'assigned_user_id', 'id');

View File

@ -17,6 +17,7 @@ use App\Factory\InvoiceInvitationFactory;
use App\Helpers\Invoice\InvoiceSum; use App\Helpers\Invoice\InvoiceSum;
use App\Jobs\Company\UpdateCompanyLedgerWithInvoice; use App\Jobs\Company\UpdateCompanyLedgerWithInvoice;
use App\Jobs\Invoice\ApplyInvoiceNumber; use App\Jobs\Invoice\ApplyInvoiceNumber;
use App\Jobs\Invoice\CreateInvoiceInvitations;
use App\Listeners\Invoice\CreateInvoiceInvitation; use App\Listeners\Invoice\CreateInvoiceInvitation;
use App\Models\ClientContact; use App\Models\ClientContact;
use App\Models\Invoice; use App\Models\Invoice;
@ -58,6 +59,7 @@ class InvoiceRepository extends BaseRepository
$starting_amount = $invoice->amount; $starting_amount = $invoice->amount;
$invoice->fill($data); $invoice->fill($data);
$invoice->save(); $invoice->save();
if(isset($data['client_contacts'])) if(isset($data['client_contacts']))
@ -105,7 +107,9 @@ class InvoiceRepository extends BaseRepository
} }
//event(new CreateInvoiceInvitation($invoice)); /* If no invitations have been created, this is our fail safe to maintain state*/
if($invoice->invitations->count() == 0)
CreateInvoiceInvitations::dispatchNow($invoice);
$invoice = $invoice->calc()->getInvoice(); $invoice = $invoice->calc()->getInvoice();

View File

@ -28,8 +28,19 @@ class PaymentRepository extends BaseRepository
public function save(Request $request, Payment $payment) : ?Payment public function save(Request $request, Payment $payment) : ?Payment
{ {
$payment->fill($request->input()); $payment->fill($request->input());
if($request->input('invoices'))
{
$invoices = Invoice::whereIn('id', $request->input('invoices'))->get();
}
//parse invoices[] and attach to paymentables
//parse invoices[] and apply payments and subfunctions
$payment->save(); $payment->save();
return $payment; return $payment;

View File

@ -12,6 +12,7 @@
namespace App\Repositories; namespace App\Repositories;
use App\Helpers\Invoice\InvoiceSum; use App\Helpers\Invoice\InvoiceSum;
use App\Models\Client;
use App\Models\Quote; use App\Models\Quote;
use Illuminate\Http\Request; use Illuminate\Http\Request;
@ -29,12 +30,12 @@ class QuoteRepository extends BaseRepository
public function save(Request $request, Quote $quote) : ?Quote public function save(Request $request, Quote $quote) : ?Quote
{ {
$quote->fill($request->input()); $quote->fill($request->input());
$quote->save(); $quote->save();
$invoice_calc = new InvoiceSum($quote);
$invoice_calc = new InvoiceSum($quote, $quote->settings);
$quote = $invoice_calc->build()->getInvoice(); $quote = $invoice_calc->build()->getInvoice();

View File

@ -28,7 +28,7 @@ class InvoiceInvitationTransformer extends EntityTransformer
'key' => $invitation->key, 'key' => $invitation->key,
'link' => $invitation->getLink() ?: '', 'link' => $invitation->getLink() ?: '',
'sent_date' => $invitation->sent_date ?: '', 'sent_date' => $invitation->sent_date ?: '',
'viewed_date' => $invitation->sent_date ?: '', 'viewed_date' => $invitation->viewed_date ?: '',
'opened_date' => $invitation->opened_date ?: '', 'opened_date' => $invitation->opened_date ?: '',
]; ];
} }

View File

@ -34,7 +34,6 @@ trait Inviteable
if(isset($this->opened_date)) if(isset($this->opened_date))
$status = ctrans('texts.invitation_status_opened'); $status = ctrans('texts.invitation_status_opened');
if(isset($this->viewed_date)) if(isset($this->viewed_date))
$status = ctrans('texts.invitation_status_viewed'); $status = ctrans('texts.invitation_status_viewed');
@ -42,4 +41,30 @@ trait Inviteable
return $status; return $status;
} }
public function getLink() : string
{
$entity_type = strtolower(class_basename($this->entityType()));
//$this->with('company','contact',$this->entity_type);
$this->with('company');
$domain = isset($this->company->portal_domain) ?: $this->company->domain;
switch ($this->company->portal_mode) {
case 'subdomain':
return $domain . $entity_type .'/'. $this->key;
break;
case 'iframe':
return $domain . $entity_type .'/'. $this->key;
//return $domain . $entity_type .'/'. $this->contact->client->client_hash .'/'. $this->key;
break;
case 'domain':
return $domain . $entity_type .'/'. $this->key;
break;
}
}
} }

View File

@ -161,6 +161,9 @@ class CreateUsersTable extends Migration
$table->unsignedInteger('size_id')->nullable(); $table->unsignedInteger('size_id')->nullable();
$table->string('first_day_of_week')->nullable(); $table->string('first_day_of_week')->nullable();
$table->string('first_month_of_year')->nullable(); $table->string('first_month_of_year')->nullable();
$table->string('portal_mode')->default('subdomain');
$table->string('portal_domain')->nullable();
$table->smallInteger('enable_modules')->default(0); $table->smallInteger('enable_modules')->default(0);
$table->mediumText('custom_fields'); $table->mediumText('custom_fields');
$table->mediumText('settings'); $table->mediumText('settings');

View File

@ -25,7 +25,7 @@ Route::group(['middleware' => ['api_secret_check']], function () {
}); });
Route::group(['api_secret_check','domain_db'], function () { Route::group(['api_secret_check','email_db'], function () {
Route::post('api/v1/login', 'Auth\LoginController@apiLogin')->name('login.submit'); Route::post('api/v1/login', 'Auth\LoginController@apiLogin')->name('login.submit');
Route::post('api/v1/reset_password', 'Auth\ForgotPasswordController@sendResetLinkEmail')->name('password.reset'); Route::post('api/v1/reset_password', 'Auth\ForgotPasswordController@sendResetLinkEmail')->name('password.reset');

View File

@ -11,7 +11,6 @@ Route::get('client/password/reset/{token}', 'Auth\ContactResetPasswordController
Route::post('client/password/reset', 'Auth\ContactResetPasswordController@reset')->name('client.password.update'); Route::post('client/password/reset', 'Auth\ContactResetPasswordController@reset')->name('client.password.update');
//todo implement domain DB //todo implement domain DB
//Route::group(['middleware' => ['auth:contact', 'domain_db'], 'prefix' => 'client', 'as' => 'client.'], function () {
Route::group(['middleware' => ['auth:contact'], 'prefix' => 'client', 'as' => 'client.'], function () { Route::group(['middleware' => ['auth:contact'], 'prefix' => 'client', 'as' => 'client.'], function () {
Route::get('dashboard', 'ClientPortal\DashboardController@index')->name('dashboard'); // name = (dashboard. index / create / show / update / destroy / edit Route::get('dashboard', 'ClientPortal\DashboardController@index')->name('dashboard'); // name = (dashboard. index / create / show / update / destroy / edit
@ -45,11 +44,11 @@ Route::group(['middleware' => ['auth:contact'], 'prefix' => 'client', 'as' => 'c
}); });
Route::group(['middleware' => ['domain_db'], 'prefix' => 'client', 'as' => 'client.'], function () { Route::group(['middleware' => ['invite_db'], 'prefix' => 'client', 'as' => 'client.'], function () {
/*Invitation catches*/ /*Invitation catches*/
Route::get('invoice/{invitation_id}','ClientPortal\InvitationController@invoiceRouter'); Route::get('{entity}/{invitation_key}','ClientPortal\InvitationController@router');
//Route::get('invoice/{client_hash}/{invitation_id}','ClientPortal\InvitationController@invoiceRouterForIframe'); we shouldn't need this if we force subdomain for the clients Route::get('{entity}/{client_hash}/{invitation_key}','ClientPortal\InvitationController@routerForIframe'); //should never need this
Route::get('payment_hook/{company_gateway_id}/{gateway_type_id}','ClientPortal\PaymentHookController@process'); Route::get('payment_hook/{company_gateway_id}/{gateway_type_id}','ClientPortal\PaymentHookController@process');
}); });

View File

@ -22,6 +22,7 @@ use Tests\TestCase;
/** /**
* @test * @test
* @covers App\Http\Controllers\ClientController
*/ */
class ClientApiTest extends TestCase class ClientApiTest extends TestCase
{ {
@ -89,5 +90,68 @@ class ClientApiTest extends TestCase
$response->assertStatus(200); $response->assertStatus(200);
} }
public function testClientNotArchived()
{
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token
])->get('/api/v1/clients/'.$this->encodePrimaryKey($this->client->id));
$arr = $response->json();
$this->assertNull($arr['data']['deleted_at']);
}
public function testClientArchived()
{
$data = [
'ids' => [$this->encodePrimaryKey($this->client->id)],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token
])->post('/api/v1/clients/bulk?action=archive', $data);
$arr = $response->json();
$this->assertNotNull($arr['data'][0]['deleted_at']);
}
public function testClientRestored()
{
$data = [
'ids' => [$this->encodePrimaryKey($this->client->id)],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token
])->post('/api/v1/clients/bulk?action=restore', $data);
$arr = $response->json();
$this->assertNull($arr['data'][0]['deleted_at']);
}
public function testClientDeleted()
{
$data = [
'ids' => [$this->encodePrimaryKey($this->client->id)],
];
$response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $this->token
])->post('/api/v1/clients/bulk?action=delete', $data);
$arr = $response->json();
$this->assertTrue($arr['data'][0]['is_deleted']);
}
} }

View File

@ -178,17 +178,17 @@ class QuoteTest extends TestCase
$quote_update = [ $quote_update = [
'status_id' => Quote::STATUS_APPROVED, 'status_id' => Quote::STATUS_APPROVED,
'client_id' => $quote->client_id, // 'client_id' => $this->encodePrimaryKey($quote->client_id),
]; ];
$this->assertNotNull($quote); $this->assertNotNull($quote);
$response = $this->withHeaders([ $response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'), 'X-API-SECRET' => config('ninja.api_secret'),
'X-API-TOKEN' => $token, 'X-API-TOKEN' => $token,
])->put('/api/v1/quotes/'.$this->encodePrimaryKey($quote->id), $quote_update) ])->put('/api/v1/quotes/'.$this->encodePrimaryKey($quote->id), $quote_update);
->assertStatus(200);
$response->assertStatus(200);
$response = $this->withHeaders([ $response = $this->withHeaders([
'X-API-SECRET' => config('ninja.api_secret'), 'X-API-SECRET' => config('ninja.api_secret'),

View File

@ -112,10 +112,24 @@ class MultiDBUserTest extends TestCase
$this->assertFalse(MultiDB::checkUserEmailExists('bademail@example.com')); $this->assertFalse(MultiDB::checkUserEmailExists('bademail@example.com'));
} }
public function test_check_that_set_db_by_email_works()
{
$this->assertTrue(MultiDB::userFindAndSetDb('db1@example.com'));
}
public function test_check_that_set_db_by_email_works_db_2()
{
$this->assertTrue(MultiDB::userFindAndSetDb('db2@example.com'));
}
public function test_check_that_set_db_by_email_works_db_3()
{
$this->assertFalse(MultiDB::userFindAndSetDb('bademail@example.com'));
}
/* /*
* This is what you do when you demand 100% code coverage :/ * This is what you do when you demand 100% code coverage :/
*/ */
public function test_set_db_invokes() public function test_set_db_invokes()
{ {
$this->expectNotToPerformAssertions(MultiDB::setDB('db-ninja-01')); $this->expectNotToPerformAssertions(MultiDB::setDB('db-ninja-01'));

View File

@ -18,6 +18,7 @@ use App\Factory\ClientFactory;
use App\Factory\InvoiceFactory; use App\Factory\InvoiceFactory;
use App\Factory\InvoiceItemFactory; use App\Factory\InvoiceItemFactory;
use App\Factory\InvoiceToRecurringInvoiceFactory; use App\Factory\InvoiceToRecurringInvoiceFactory;
use App\Helpers\Invoice\InvoiceSum;
use App\Jobs\Company\UpdateCompanyLedgerWithInvoice; use App\Jobs\Company\UpdateCompanyLedgerWithInvoice;
use App\Models\Client; use App\Models\Client;
use App\Models\CompanyGateway; use App\Models\CompanyGateway;
@ -29,8 +30,9 @@ use App\Models\Quote;
use App\Models\RecurringInvoice; use App\Models\RecurringInvoice;
use App\Utils\Traits\GeneratesCounter; use App\Utils\Traits\GeneratesCounter;
use App\Utils\Traits\MakesHash; use App\Utils\Traits\MakesHash;
use App\Helpers\Invoice\InvoiceSum;
use Illuminate\Support\Carbon; use Illuminate\Support\Carbon;
use Illuminate\Support\Facades\Cache;
use Illuminate\Support\Facades\Schema;
/** /**
* Class MockAccountData * Class MockAccountData
@ -54,6 +56,34 @@ trait MockAccountData
public function makeTestData() public function makeTestData()
{ {
/* Warm up the cache !*/
$cached_tables = config('ninja.cached_tables');
foreach ($cached_tables as $name => $class) {
if (! Cache::has($name)) {
// check that the table exists in case the migration is pending
if (! Schema::hasTable((new $class())->getTable())) {
continue;
}
if ($name == 'payment_terms') {
$orderBy = 'num_days';
} elseif ($name == 'fonts') {
$orderBy = 'sort_order';
} elseif (in_array($name, ['currencies', 'industries', 'languages', 'countries', 'banks'])) {
$orderBy = 'name';
} else {
$orderBy = 'id';
}
$tableData = $class::orderBy($orderBy)->get();
if ($tableData->count()) {
Cache::forever($name, $tableData);
}
}
}
$this->account = factory(\App\Models\Account::class)->create(); $this->account = factory(\App\Models\Account::class)->create();
$this->company = factory(\App\Models\Company::class)->create([ $this->company = factory(\App\Models\Company::class)->create([
'account_id' => $this->account->id, 'account_id' => $this->account->id,