mirror of
				https://github.com/invoiceninja/invoiceninja.git
				synced 2025-11-03 20:07:31 -05:00 
			
		
		
		
	
		
			
				
	
	
		
			430 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			430 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php namespace App\Http\Controllers;
 | 
						|
 | 
						|
use Auth;
 | 
						|
use Illuminate\Support\Facades\Log;
 | 
						|
use Illuminate\Support\Facades\Request;
 | 
						|
use Utils;
 | 
						|
use Response;
 | 
						|
use Input;
 | 
						|
use Validator;
 | 
						|
use App\Models\Invoice;
 | 
						|
use App\Models\Client;
 | 
						|
use App\Models\Contact;
 | 
						|
use App\Models\Product;
 | 
						|
use App\Models\Invitation;
 | 
						|
use App\Ninja\Repositories\ClientRepository;
 | 
						|
use App\Ninja\Repositories\PaymentRepository;
 | 
						|
use App\Ninja\Repositories\InvoiceRepository;
 | 
						|
use App\Ninja\Mailers\ContactMailer as Mailer;
 | 
						|
use App\Http\Controllers\BaseAPIController;
 | 
						|
use App\Ninja\Transformers\InvoiceTransformer;
 | 
						|
use App\Http\Requests\CreateInvoiceRequest;
 | 
						|
use App\Http\Requests\UpdateInvoiceRequest;
 | 
						|
use App\Services\InvoiceService;
 | 
						|
 | 
						|
class InvoiceApiController extends BaseAPIController
 | 
						|
{
 | 
						|
    protected $invoiceRepo;
 | 
						|
 | 
						|
    public function __construct(InvoiceService $invoiceService, InvoiceRepository $invoiceRepo, ClientRepository $clientRepo, PaymentRepository $paymentRepo, Mailer $mailer)
 | 
						|
    {
 | 
						|
        parent::__construct();
 | 
						|
 | 
						|
        $this->invoiceRepo = $invoiceRepo;
 | 
						|
        $this->clientRepo = $clientRepo;
 | 
						|
        $this->paymentRepo = $paymentRepo;
 | 
						|
        $this->invoiceService = $invoiceService;
 | 
						|
        $this->mailer = $mailer;
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @SWG\Get(
 | 
						|
     *   path="/invoices",
 | 
						|
     *   summary="List of invoices",
 | 
						|
     *   tags={"invoice"},
 | 
						|
     *   @SWG\Response(
 | 
						|
     *     response=200,
 | 
						|
     *     description="A list with invoices",
 | 
						|
     *      @SWG\Schema(type="array", @SWG\Items(ref="#/definitions/Invoice"))
 | 
						|
     *   ),
 | 
						|
     *   @SWG\Response(
 | 
						|
     *     response="default",
 | 
						|
     *     description="an ""unexpected"" error"
 | 
						|
     *   )
 | 
						|
     * )
 | 
						|
     */
 | 
						|
    public function index()
 | 
						|
    {
 | 
						|
        $paginator = Invoice::scope()->withTrashed();
 | 
						|
        $invoices = Invoice::scope()->withTrashed()
 | 
						|
                        ->with(array_merge(['invoice_items'], $this->getIncluded()));
 | 
						|
 | 
						|
        if ($clientPublicId = Input::get('client_id')) {
 | 
						|
            $filter = function($query) use ($clientPublicId) {
 | 
						|
                $query->where('public_id', '=', $clientPublicId);
 | 
						|
            };
 | 
						|
            $invoices->whereHas('client', $filter);
 | 
						|
            $paginator->whereHas('client', $filter);
 | 
						|
        }
 | 
						|
 | 
						|
        $invoices = $invoices->orderBy('created_at', 'desc')->paginate();
 | 
						|
 | 
						|
        /*
 | 
						|
        // Add the first invitation link to the data
 | 
						|
        foreach ($invoices as $key => $invoice) {
 | 
						|
            foreach ($invoice->invitations as $subKey => $invitation) {
 | 
						|
                $invoices[$key]['link'] = $invitation->getLink();
 | 
						|
            }
 | 
						|
            unset($invoice['invitations']);
 | 
						|
        }
 | 
						|
        */
 | 
						|
 | 
						|
        $transformer = new InvoiceTransformer(Auth::user()->account, Input::get('serializer'));
 | 
						|
        $paginator = $paginator->paginate();
 | 
						|
 | 
						|
        $data = $this->createCollection($invoices, $transformer, 'invoices', $paginator);
 | 
						|
 | 
						|
        return $this->response($data);
 | 
						|
    }
 | 
						|
 | 
						|
        /**
 | 
						|
         * @SWG\Get(
 | 
						|
         *   path="/invoices/{invoice_id}",
 | 
						|
         *   summary="Individual Invoice",
 | 
						|
         *   tags={"invoice"},
 | 
						|
         *   @SWG\Response(
 | 
						|
         *     response=200,
 | 
						|
         *     description="A single invoice",
 | 
						|
         *      @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice"))
 | 
						|
         *   ),
 | 
						|
         *   @SWG\Response(
 | 
						|
         *     response="default",
 | 
						|
         *     description="an ""unexpected"" error"
 | 
						|
         *   )
 | 
						|
         * )
 | 
						|
         */
 | 
						|
 | 
						|
    public function show($publicId)
 | 
						|
    {
 | 
						|
 | 
						|
        $invoice = Invoice::scope($publicId)->withTrashed()->first();
 | 
						|
 | 
						|
        if(!$invoice)
 | 
						|
            return $this->errorResponse(['message'=>'Invoice does not exist!'], 404);
 | 
						|
 | 
						|
        $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
        $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
        return $this->response($data);
 | 
						|
    }
 | 
						|
 | 
						|
    /**
 | 
						|
     * @SWG\Post(
 | 
						|
     *   path="/invoices",
 | 
						|
     *   tags={"invoice"},
 | 
						|
     *   summary="Create an invoice",
 | 
						|
     *   @SWG\Parameter(
 | 
						|
     *     in="body",
 | 
						|
     *     name="body",
 | 
						|
     *     @SWG\Schema(ref="#/definitions/Invoice")
 | 
						|
     *   ),
 | 
						|
     *   @SWG\Response(
 | 
						|
     *     response=200,
 | 
						|
     *     description="New invoice",
 | 
						|
     *      @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice"))
 | 
						|
     *   ),
 | 
						|
     *   @SWG\Response(
 | 
						|
     *     response="default",
 | 
						|
     *     description="an ""unexpected"" error"
 | 
						|
     *   )
 | 
						|
     * )
 | 
						|
     */
 | 
						|
    public function store(CreateInvoiceRequest $request)
 | 
						|
    {
 | 
						|
        $data = Input::all();
 | 
						|
        $error = null;
 | 
						|
 | 
						|
        if (isset($data['email'])) {
 | 
						|
            $email = $data['email'];
 | 
						|
            $client = Client::scope()->whereHas('contacts', function($query) use ($email) {
 | 
						|
                $query->where('email', '=', $email);
 | 
						|
            })->first();
 | 
						|
 | 
						|
            if (!$client) {
 | 
						|
                $validator = Validator::make(['email'=>$email], ['email' => 'email']);
 | 
						|
                if ($validator->fails()) {
 | 
						|
                    $messages = $validator->messages();
 | 
						|
                    return $messages->first();
 | 
						|
                }
 | 
						|
 | 
						|
                $clientData = ['contact' => ['email' => $email]];
 | 
						|
                foreach ([
 | 
						|
                    'name',
 | 
						|
                    'address1',
 | 
						|
                    'address2',
 | 
						|
                    'city',
 | 
						|
                    'state',
 | 
						|
                    'postal_code',
 | 
						|
                    'private_notes',
 | 
						|
                ] as $field) {
 | 
						|
                    if (isset($data[$field])) {
 | 
						|
                        $clientData[$field] = $data[$field];
 | 
						|
                    }
 | 
						|
                }
 | 
						|
                foreach ([
 | 
						|
                    'first_name',
 | 
						|
                    'last_name',
 | 
						|
                    'phone',
 | 
						|
                ] as $field) {
 | 
						|
                    if (isset($data[$field])) {
 | 
						|
                        $clientData['contact'][$field] = $data[$field];
 | 
						|
                    }
 | 
						|
                }
 | 
						|
 | 
						|
                $client = $this->clientRepo->save($clientData);
 | 
						|
            }
 | 
						|
        } else if (isset($data['client_id'])) {
 | 
						|
            $client = Client::scope($data['client_id'])->firstOrFail();
 | 
						|
        }
 | 
						|
 | 
						|
        $data = self::prepareData($data, $client);
 | 
						|
        $data['client_id'] = $client->id;
 | 
						|
        $invoice = $this->invoiceService->save($data);
 | 
						|
        $payment = false;
 | 
						|
 | 
						|
        // Optionally create payment with invoice
 | 
						|
        if (isset($data['paid']) && $data['paid']) {
 | 
						|
            $payment = $this->paymentRepo->save([
 | 
						|
                'invoice_id' => $invoice->id,
 | 
						|
                'client_id' => $client->id,
 | 
						|
                'amount' => $data['paid']
 | 
						|
            ]);
 | 
						|
        }
 | 
						|
 | 
						|
        if (isset($data['email_invoice']) && $data['email_invoice']) {
 | 
						|
            if ($payment) {
 | 
						|
                $this->mailer->sendPaymentConfirmation($payment);
 | 
						|
            } else {
 | 
						|
                $this->mailer->sendInvoice($invoice);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        $invoice = Invoice::scope($invoice->public_id)->with('client', 'invoice_items', 'invitations')->first();
 | 
						|
        $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
        $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
        return $this->response($data);
 | 
						|
    }
 | 
						|
 | 
						|
    private function prepareData($data, $client)
 | 
						|
    {
 | 
						|
        $account = Auth::user()->account;
 | 
						|
        $account->loadLocalizationSettings($client);
 | 
						|
 | 
						|
        // set defaults for optional fields
 | 
						|
        $fields = [
 | 
						|
            'discount' => 0,
 | 
						|
            'is_amount_discount' => false,
 | 
						|
            'terms' => '',
 | 
						|
            'invoice_footer' => '',
 | 
						|
            'public_notes' => '',
 | 
						|
            'po_number' => '',
 | 
						|
            'invoice_design_id' => $account->invoice_design_id,
 | 
						|
            'invoice_items' => [],
 | 
						|
            'custom_value1' => 0,
 | 
						|
            'custom_value2' => 0,
 | 
						|
            'custom_taxes1' => false,
 | 
						|
            'custom_taxes2' => false,
 | 
						|
            'partial' => 0
 | 
						|
        ];
 | 
						|
 | 
						|
        if (!isset($data['invoice_status_id']) || $data['invoice_status_id'] == 0) {
 | 
						|
            $data['invoice_status_id'] = INVOICE_STATUS_DRAFT;
 | 
						|
        }
 | 
						|
 | 
						|
        if (!isset($data['invoice_date'])) {
 | 
						|
            $fields['invoice_date_sql'] = date_create()->format('Y-m-d');
 | 
						|
        }
 | 
						|
        if (!isset($data['due_date'])) {
 | 
						|
            $fields['due_date_sql'] = false;
 | 
						|
        }
 | 
						|
 | 
						|
        foreach ($fields as $key => $val) {
 | 
						|
            if (!isset($data[$key])) {
 | 
						|
                $data[$key] = $val;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        // initialize the line items
 | 
						|
        if (isset($data['product_key']) || isset($data['cost']) || isset($data['notes']) || isset($data['qty'])) {
 | 
						|
            $data['invoice_items'] = [self::prepareItem($data)];
 | 
						|
 | 
						|
            // make sure the tax isn't applied twice (for the invoice and the line item)
 | 
						|
            unset($data['invoice_items'][0]['tax_name']);
 | 
						|
            unset($data['invoice_items'][0]['tax_rate']);
 | 
						|
        } else {
 | 
						|
            foreach ($data['invoice_items'] as $index => $item) {
 | 
						|
                $data['invoice_items'][$index] = self::prepareItem($item);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return $data;
 | 
						|
    }
 | 
						|
 | 
						|
    private function prepareItem($item)
 | 
						|
    {
 | 
						|
        // if only the product key is set we'll load the cost and notes
 | 
						|
        if (!empty($item['product_key']) && empty($item['cost']) && empty($item['notes'])) {
 | 
						|
            $product = Product::findProductByKey($item['product_key']);
 | 
						|
            if ($product) {
 | 
						|
                if (empty($item['cost'])) {
 | 
						|
                    $item['cost'] = $product->cost;
 | 
						|
                }
 | 
						|
                if (empty($item['notes'])) {
 | 
						|
                    $item['notes'] = $product->notes;
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        $fields = [
 | 
						|
            'cost' => 0,
 | 
						|
            'product_key' => '',
 | 
						|
            'notes' => '',
 | 
						|
            'qty' => 1
 | 
						|
        ];
 | 
						|
 | 
						|
        foreach ($fields as $key => $val) {
 | 
						|
            if (!isset($item[$key])) {
 | 
						|
                $item[$key] = $val;
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return $item;
 | 
						|
    }
 | 
						|
 | 
						|
    public function emailInvoice()
 | 
						|
    {
 | 
						|
        $data = Input::all();
 | 
						|
        $error = null;
 | 
						|
 | 
						|
        $invoice = Invoice::scope($data['id'])->withTrashed()->first();
 | 
						|
 | 
						|
        if(!$invoice)
 | 
						|
            return $this->errorResponse(['message'=>'Invoice does not exist.'], 400);
 | 
						|
 | 
						|
 | 
						|
        $this->mailer->sendInvoice($invoice, false, false);
 | 
						|
 | 
						|
 | 
						|
        if($error) {
 | 
						|
            return $this->errorResponse(['message'=>'There was an error sending the invoice'], 400);
 | 
						|
        }
 | 
						|
        else {
 | 
						|
            $response = json_encode(RESULT_SUCCESS, JSON_PRETTY_PRINT);
 | 
						|
        }
 | 
						|
 | 
						|
        $headers = Utils::getApiHeaders();
 | 
						|
        return Response::make($response, $error ? 400 : 200, $headers);
 | 
						|
    }
 | 
						|
 | 
						|
 | 
						|
        /**
 | 
						|
         * @SWG\Put(
 | 
						|
         *   path="/invoices",
 | 
						|
         *   tags={"invoice"},
 | 
						|
         *   summary="Update an invoice",
 | 
						|
         *   @SWG\Parameter(
 | 
						|
         *     in="body",
 | 
						|
         *     name="body",
 | 
						|
         *     @SWG\Schema(ref="#/definitions/Invoice")
 | 
						|
         *   ),
 | 
						|
         *   @SWG\Response(
 | 
						|
         *     response=200,
 | 
						|
         *     description="Update invoice",
 | 
						|
         *      @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice"))
 | 
						|
         *   ),
 | 
						|
         *   @SWG\Response(
 | 
						|
         *     response="default",
 | 
						|
         *     description="an ""unexpected"" error"
 | 
						|
         *   )
 | 
						|
         * )
 | 
						|
         */
 | 
						|
    public function update(UpdateInvoiceRequest $request, $publicId)
 | 
						|
    {
 | 
						|
        if ($request->action == ACTION_ARCHIVE) {
 | 
						|
            $invoice = Invoice::scope($publicId)->firstOrFail();
 | 
						|
            $this->invoiceRepo->archive($invoice);
 | 
						|
 | 
						|
            $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
            $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
            return $this->response($data);
 | 
						|
        }
 | 
						|
        else if ($request->action == ACTION_CONVERT) {
 | 
						|
            $quote = Invoice::scope($publicId)->firstOrFail();
 | 
						|
            $invoice = $this->invoiceRepo->cloneInvoice($quote, $quote->id);
 | 
						|
 | 
						|
            $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
            $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
            return $this->response($data);
 | 
						|
        }
 | 
						|
        else if ($request->action == ACTION_RESTORE) {
 | 
						|
            $invoice = Invoice::scope($publicId)->withTrashed()->firstOrFail();
 | 
						|
            $this->invoiceRepo->restore($invoice);
 | 
						|
 | 
						|
            $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
            $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
            return $this->response($data);
 | 
						|
        }
 | 
						|
 | 
						|
        $data = $request->input();
 | 
						|
        $data['public_id'] = $publicId;
 | 
						|
        $this->invoiceService->save($data);
 | 
						|
 | 
						|
        $invoice = Invoice::scope($publicId)->with('client', 'invoice_items', 'invitations')->firstOrFail();
 | 
						|
        $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
        $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
        return $this->response($data);
 | 
						|
    }
 | 
						|
 | 
						|
        /**
 | 
						|
         * @SWG\Delete(
 | 
						|
         *   path="/invoices",
 | 
						|
         *   tags={"invoice"},
 | 
						|
         *   summary="Delete an invoice",
 | 
						|
         *   @SWG\Parameter(
 | 
						|
         *     in="body",
 | 
						|
         *     name="body",
 | 
						|
         *     @SWG\Schema(ref="#/definitions/Invoice")
 | 
						|
         *   ),
 | 
						|
         *   @SWG\Response(
 | 
						|
         *     response=200,
 | 
						|
         *     description="Delete invoice",
 | 
						|
         *      @SWG\Schema(type="object", @SWG\Items(ref="#/definitions/Invoice"))
 | 
						|
         *   ),
 | 
						|
         *   @SWG\Response(
 | 
						|
         *     response="default",
 | 
						|
         *     description="an ""unexpected"" error"
 | 
						|
         *   )
 | 
						|
         * )
 | 
						|
         */
 | 
						|
 | 
						|
    public function destroy($publicId)
 | 
						|
    {
 | 
						|
        $data['public_id'] = $publicId;
 | 
						|
        $invoice = Invoice::scope($publicId)->firstOrFail();
 | 
						|
 | 
						|
        $this->invoiceRepo->delete($invoice);
 | 
						|
 | 
						|
        $transformer = new InvoiceTransformer(\Auth::user()->account, Input::get('serializer'));
 | 
						|
        $data = $this->createItem($invoice, $transformer, 'invoice');
 | 
						|
 | 
						|
        return $this->response($data);
 | 
						|
 | 
						|
    }
 | 
						|
 | 
						|
}
 |