=head1 NAME

WebService::HMRC::VAT - Interact with the UK HMRC VAT API

=head1 VERSION

Version 0.02


=cut

=head1 SYNOPSIS

    use WebService::HMRC::VAT;
    my $vat = WebService::HMRC::VAT->new({
        vrn => '123456789'
    });
    $vat->auth->access_token->('MY-ACCESS-TOKEN');

    # Display outstanding VAT returns
    my $result = $vat->obligations({
        from  => '2018-01-01',
        to    => '2018-12-31',
        state => 'O',
    });
    if($result->is_success) {
        foreach my $o (@{$result->data->{obligations}) {
            print "VAT return " . $o->{periodKey};
            print " due " . $o->{due} . "\n";
        }
    }

    # Submit a VAT return
    $result = $vat->submit_return({
        periodKey => "#001",                 # from ->obligations()
        vatDueSales => 100.00,               # Box 1 on paper form
        vatDueAcquisitions => 100.00,        # Box 2 on paper form
        totalVatDue => 200.00,               # Box 3 on paper form
        vatReclaimedCurrPeriod => 100.00,    # Box 4 on paper form
        netVatDue => 100.00,                 # Box 5 on paper form
        totalValueSalesExVAT => 500,         # Box 6 on paper form
        totalValuePurchasesExVAT => 500,     # Box 7 on paper form
        totalValueGoodsSuppliedExVAT => 500, # Box 8 on paper form
        totalAcquisitionsExVAT => 500,       # Box 9 on paper form
        finalised => 1,                      # Boolean user declaration
    });
    if($result->is_success) {
        print "VAT Return submitted:\n"
        printf "Receipt ID: %s\n", $result->header('Receipt-ID');
        printf "Form Bundle: %s\n", $result->data->{formBundleNumber};
    }

=head1 DESCRIPTION

Perl module to interact with the UK's HMRC Making Tax Digital
`VAT` API. This allows VAT returns to be submitted or viewed,
and obligations, payments and liabilities to be viewed.

For more information, see:
L<https://developer.service.hmrc.gov.uk/api-documentation/docs/api/service/vat-api/1.0>

=head1 REQUIRES

=over

=item * L<JSON::MaybeXS>

=item * L<Moose>

=item * L<namespace::autoclean>

=item * L<URI::Escape>

=back

=head1 EXPORTS

Nothing

=head1 PROPERTIES

Inherits from L<WebService::HMRC::Request>.

=head2 vrn

VAT registration number for which records are being queries or submitted.
Required parameter for initialisation.

The VAT registration number must be purely numeric, without any spaces or
GB prefix.


=cut

=head1 METHODS

Inherits from L<WebService::HMRC::Request>.

=head2 obligations({ from => 'YYYY-MM-DD', to => 'YYYY-MM-DD', [status => $status, ] [test_mode => $test_mode] })

Retrieve a set of VAT filing obligations for the specified date range. Returns
a WebService::HMRC::Response object reference. Requires permission for the
C<read:vat> service scope.

=head3 Parameters

=over

=item from

Return obligations from this date, specified as YYYY-MM-DD.
Required parameter.

=item to

Return obligations up to this date, specified as YYYY-MM-DD.
Required parameter.

=item status

Optional parameter to filter the obligations returned. May be set to 'O' to
return only 'open' obligations, or 'F' to return only 'fulfilled' obligations.

Default is to return all obligations, both fulfilled and open.

=item test_mode

Optional parameter used only for testing against the HMRC sandbox api.

This parameter should not be used in production systems - it causes dummy
test data to be returned.

By default, when testing against the sandbox with no C<test_mode> specified,
the test api simulates the scenario where the client has quarterly
obligations and one is fulfilled

Other test scenarios are available by setting the C<test_mode> parameter
as detailed below:

C<QUARTERLY_NONE_MET> simulates the scenario where the client has quarterly
obligations and none are fulfilled.

C<QUARTERLY_ONE_MET> simulates the scenario where the client has quarterly
obligations and one is fulfilled.

C<QUARTERLY_TWO_MET> simulates the scenario where the client has quarterly
obligations and two are fulfilled.

C<QUARTERLY_THREE_MET> simulates the scenario where the client has quarterly
obligations and three are fulfilled.

C<QUARTERLY_FOUR_MET> simulates the scenario where the client has quarterly
obligations and four are fulfilled.

C<MONTHLY_NONE_MET> simulates the scenario where the client has monthly
obligations and none are fulfilled.

C<MONTHLY_ONE_MET> simulates the scenario where the client has monthly
obligations and one month is fulfilled.

C<MONTHLY_TWO_MET> simulates the scenario where the client has monthly
obligations and two months are fulfilled.

C<MONTHLY_THREE_MET> simulates the scenario where the client has monthly
obligations and three months are fulfilled.

C<NOT_FOUND> simulates the scenario where no data is found.

=back

=head3 Response Data

For full details of the response data, see the HMRC API specification. In summary,
the data contains a single element `obligations` pointing to an array of
VAT return obligations:

    {
      obligations => [
        {
          periodKey => "#001"
          start     => "2017-04-06",
          end       => "2017-07-05",
          due       => "2017-08-12",
          status    => "F",
          received  => "2017-08-05", # only present if 'Fulfilled'
        },
        {
          periodKey => "#004"
          start     => "2018-01-06",
          end       => "2018-04-05",
          due"      => "2018-05-12",
          status    => "O",
        },
      ]
    }

=head3 Example usage

    my $result = $vat->obligations({
        from => '2018-01-01',
        to   => '2018-12-31',
    });
    foreach my $obligation( @{$result->data->{obligations}} ) {
        print "-- VAT Return Obligation --\n"
        print "           ID: $obligation->{periodKey}\n";
        print " Period Start: $obligation->{start}\n";
        print "   Period End: $obligation->{end}\n";
        print "     Due Date: $obligation->{due}\n";
        print "Received Date: ";
        if ($obligation->{status} eq 'F') {
            print "$obligation->{received}\n";
        }
        else {
            print "Still Outstanding\n";
        }
    }


=cut

=head2 liabilities({ from => 'YYYY-MM-DD', to => 'YYYY-MM-DD', [test_mode => $test_mode] })

Retrieve a set of VAT payment liabilities. Returns a WebService::HMRC::Response
object reference. Requires permission for the C<read:vat> service scope.

=head3 Parameters

=over

=item from

Return liabilities from this date, specified as YYYY-MM-DD.
Required parameter. The date must be before today's date, otherwise the api
will return an error.

=item to

Return liabilities up to this date, specified as YYYY-MM-DD.
Required parameter.

=item test_mode

Optional parameter used only for testing against the HMRC sandbox api.

If set to C<SINGLE_LIABILITY>, returns a single valid liability when used
with dates from 2017-01-02 and to 2017-02-02.

If set to C<MULTIPLE_LIABILITIES>, returns multiple valid liabilities when
used with dates from 2017-04-05 and to 2017-12-21.

This parameter should not be used in production systems - it causes dummy
test data to be returned.

=back

=head3 Response Data

For full details of the response data, see the HMRC API specification. In summary,
the data contains a single element `liabilities` pointing to an array of
VAT payment liabilities:

    {
      liabilities => [
        {
          taxPeriod => {
            from => "2017-04-06",
            to   => "2017-07-06"
          },
          type => "VAT ...",
          originalAmount => 6000.00,
          outstandingAmount => 100.00,
          due => "2017-07-06"
        },
      ]
    }

=head3 Example usage

    my $result = $vat->liabilities({
        from => '2018-01-01',
        to   => '2018-03-31',
    });
    foreach my $liability( @{$result->data->{liabilities}} ) {
        print "-- VAT Payment Liability --\n"
        print "           Type: $liability->{type}\n";
        print "   Period Start: $liability->{taxPeriod}->{from}\n";
        print "     Period End: $liability->{taxPeriod}->{to}\n";
        print "Original Amount: $liability->{originalAmount}\n";
        print "     Due Amount: $liability->{outstandingAmount}\n";
        print "       Due Date: $liability->{due}\n";
    }


=cut

=head2 payments({ from => 'YYYY-MM-DD', to => 'YYYY-MM-DD', [test_mode => $test_mode]  })

Retrieve a set of payments received by HMRC in respect of VAT over the
specified date range. Returns a WebService::HMRC::Response object reference.
Requires permission for the C<read:vat> service scope.

=head3 Parameters

=over

=item from

Return payments from this date, specified as YYYY-MM-DD.
Required parameter.

=item to

Return payments up to this date, specified as YYYY-MM-DD.
Required parameter.

=item test_mode

Optional parameter used only for testing against the HMRC sandbox api.

If set to C<SINGLE_PAYMENT>, returns a single valid payment when used with
dates from 2017-01-02 and to 2017-02-02.

If set to C<MULTIPLE_PAYMENTS>, returns multiple valid payments when used
with dates from 2017-02-27 and to 2017-12-21.

This parameter should not be used in production systems - it causes dummy
test data to be returned.

=back

=head3 Response Data

For full details of the response data, see the HMRC API specification. In
summary, the data contains a single element `payments` pointing to an array
of payments received by HMRC:

    {
      payments => [
        {
          amount => 100.00,
          received => "2017-04-06"
        },
      ]
    }

=head3 Example usage

    my $result = $vat->payments(
        from => '2018-01-01',
        to   => '2018-03-31',
    );
    foreach my $payment( @{$result->data->{payments}} ) {
        print "-- VAT Payments Received by HMRC --\n"
        print "Date Received: $payment->{received}\n";
        print "       Amount: $payment->{amount}\n";
    }


=cut

=head2 get_return({ period_key => $period_key })

Retrieve a previously submitted VAT return corresponding to the supplied
period_key. Returns a WebService::HMRC::Response object reference.
Requires permission for the C<read:vat> service scope.

=head3 Parameters

=over

=item period_key

The ID code for the period that uniquely identifies a submitted
VAT return, as contained within the response from an obligations()
method call.

=back

=head3 Response Data

For full details of the response data, see the HMRC API specification. In
summary, the data is a hashref comprising the following elements which
correspond to fields on the traditional paper VAT100 form:

    {
      periodKey => "#001",
      vatDueSales => 100.00,
      vatDueAcquisitions => 100.00,
      totalVatDue => 200,
      vatReclaimedCurrPeriod => 100.00,
      netVatDue => 100,
      totalValueSalesExVAT => 500,
      totalValuePurchasesExVAT => 500,
      totalValueGoodsSuppliedExVAT => 500,
      totalAcquisitionsExVAT => 500
    }

Note that some fields are returned as integers where only whole pounds
are required to be submitted by HMRC.


=cut

=head2 submit_return($hashref)

Retrieve a previously submitted VAT return corresponding to the supplied
period_key. Returns a WebService::HMRC::Response object reference.
Requires permission for the C<write:vat> service scope.

=head3 Parameters

For full details of the required parameters and the rules for calculating
each value, see the HMRC API specification and VAT documentation. The
brief descriptions here are intended as a simplified overview to help
understand the code and are not a comprehensive or necessarily accurate
representation of VAT law.

In summary, the data is a hashref comprising the following elements which
correspond to fields on the traditional paper VAT100 form. All parameters
are required.

=over

=item period_key

The ID code for the period that uniquely identifies a submitted
VAT return, as contained within the response from an obligations()
method call.

=item vatDueSales

VAT due on sales and other outputs. Corresponds to box 1 on the traditional
VAT100 paper form.

=item vatDueAcquisitions

VAT due on acquisitions from other EC member states as part of the reverse-
charge scheme. Corresponds to box 2 on the traditional VAT100 paper form.

=item totalVatDue

The sum of vatDueSales and vatDueAcquisitions. Corresponds to box 3 on the
traditional VAT100 paper form.

=item vatReclaimedCurrPeriod

VAT reclaimed on purchases and other inputs, including acquisitions from other
EC member states. This corresponds to box 4 on the traditional VAT100 paper
form.

=item netVatDue

The absolute (unsigned, always positive) difference between totalVatDue and
vatReclaimedCurrPeriod. This corresponds to box 5 on the traditional VAT100
paper form.

=item totalValueSalesExVAT

Total value of sales and all other outputs, excluding any VAT, rounded to
an integer value. This corresponds to box 6 on the traditional VAT100 paper
form.

=item totalValuePurchasesExVAT

Total value of purchases and all other inputs, excluding any VAT and including
any exempt purchases. This corresponds to box 7 on the traditional VAT100
paper form.

=item totalValueGoodsSuppliedExVAT

Total value of all supplies of goods (but not services) to other EC member
states and costs directly related to that supply (such as freight or
insurance), excluding any VAT. This corresponds to box 8 on the traditional
VAT100 paper form.

=item totalAcquisitionsExVAT

Total value of acquisitions of goods from other EC member states and
directly related costs (such as freight or insurance), excluding any VAT.
This corresponds to box 9 on the traditional VAT100 paper form.

=item finalised

Boolean declaration that the data being submitted has been finalised and
approved by the user. Must be true for successful submission.

This parameter is converted to a JSON()->true or JSON()->false
boolean value before encoding to json for submission.

Defaults to false.

=back

=head3 Response Headers

For full details of the response headers, see the HMRC API specification. In
summary the headers confirm receipt of the submission, comprising:

=over

=item Receipt ID

Unique reference number returned for a submission.

=item Receipt-Timestamp

ISO8601 format timestamp of the form '2018-02-14T09:32:15Z'.

=item X-CorrelationId

Unique ID for this operation.

=back

=head3 Response Data

For full details of the response data, see the HMRC API specification. In
summary, the response data is a hashref with keys:

=over

=item processingDate

ISO8601 timestamp indicating the time that the submission was processed.

=item formBundleNumber

Unique number representing the "Form Bundle" in which the submitted data
has been stored by HMRC.

=item paymentIndicator

Set to 'DD' if payment is due to HMRC and they hold a Direct Debit
instruction for the client, 'BANK' if a repayment is due from HMRC and
they hold bank details to make the repayment, otherwise this element is
not present in the returned data.

=item chargeRefNumber

Present only is payment is due to HMRC.

=back

=head3 Example

    my $result = $vat->submit_return({
        periodKey => "#001",
        vatDueSales => 100.00,
        vatDueAcquisitions => 0.00,
        totalVatDue => 100.00,
        vatReclaimedCurrPeriod => 50.00,
        netVatDue => 50.00,
        totalValueSalesExVAT => 500,
        totalValuePurchasesExVAT => 250,
        totalValueGoodsSuppliedExVAT => 0,
        totalAcquisitionsExVAT => 0,
        finalised => 1,
    })
    if($result->is_success) {
        print "==== VAT Return Submitted ====\n";
        printf "        Receipt ID: %s\n", $result->header('Receipt-ID');
        printf " Receipt Timestamp: %s\n", $result->header('Receipt-Timestamp');
        printf "       Tracking ID: %s\n", $result->header('X-CorrelationId');
        printf "   Processing Date: %s\n", $result->data->{processingDate};
        printf "       Form Bundle: %s\n", $result->data->{formBundleNumber};
        if($result->data->{paymentIndicator}) {
            printf " Payment Indicator: %s\n", $result->data->{paymentIndicator};
        }
        if($result->data->{paymentIndicator}) {
            printf "  Charge Reference: %s\n", $result->data->{chargeRefNumber};
        }
    }


=cut

=head1 AUTHORISATION

Access to the HMRC Making Tax Digital VAT APIs requires that an application
be registered with HMRC and enabled for this service. Additionally permission
must be granted by a registered user for the application to access the
C<read:vat> or C<write:vat> service scope, as noted for each method.

Authorisation is provided by means of an C<access token>.

For more details on obtaining and using access tokens, See
L<WebService::HMRC::Authenticate>.

Further information, application credentials and documentation may be obtained
from the
L<HMRC Developer Hub|https://developer.service.hmrc.gov.uk/api-documentation>.

=head1 TESTING

The basic tests are run as part of the installation instructions shown above
use an invalid uri as an endpoint. This tests basic interaction with the
module's method and does not require an internet connection.

Developer pre-release tests may be run with the following command:

    prove -l xt/

With a working internet connection, HMRC application credentials and an access
token granted by a test user enrolled for the Making Tax Digital for Business
VAT service, interaction with the real HMRC sandbox api can be tested. The
test user's VAT registration number must also be provided.

The credentials are specified as environment variables when running the tests:

    HMRC_ACCESS_TOKEN=[MY-ACCESS-TOKEN] \
    HMRC_VRN=[USER-VAT-REGISTRATION-NUMBER] \
    make test TEST_VERBOSE=1

=head1 AUTHOR

Nick Prater, <nick@npbroadcast.com>

=head1 BUGS

Please report any bugs or feature requests to C<bug-webservice-hmrc-vat at rt.cpan.org>, or through
the web interface at L<http://rt.cpan.org/NoAuth/ReportBug.html?Queue=WebService-HMRC-VAT>.  I will be notified, and then you'll
automatically be notified of progress on your bug as I make changes.

=head1 SUPPORT AND DOCUMENTATION

You can find documentation for this module with the perldoc command.

    perldoc WebService::HMRC::VAT

The C<README.pod> file supplied with this distribution is generated from the
L<WebService::HMRC::VAT> module's pod by running the following
command from the distribution root:

    perldoc -u lib/WebService/HMRC/VAT.pm > README.pod

You can also look for information at:

=over 4

=item * RT: CPAN's request tracker (report bugs here)

L<http://rt.cpan.org/NoAuth/Bugs.html?Dist=WebService-HMRC-VAT>

=item * CPAN Ratings

L<http://cpanratings.perl.org/d/WebService-HMRC-VAT>

=item * Search CPAN

L<http://search.cpan.org/dist/WebService-HMRC-VAT/>

=item * Github

L<https://github.com/nick-prater/WebService-HMRC-VAT>

=back

=head1 ACKNOWLEDGEMENTS

This module was originally developed for use as part of the
L<LedgerSMB|https://ledgersmb.org/> open source accounting software.

=head1 LICENSE AND COPYRIGHT

Copyright 2018 Nick Prater.

This program is free software; you can redistribute it and/or modify it
under the terms of the the Artistic License (2.0). You may obtain a
copy of the full license at:

L<http://www.perlfoundation.org/artistic_license_2_0>

Any use, modification, and distribution of the Standard or Modified
Versions is governed by this Artistic License. By using, modifying or
distributing the Package, you accept this license. Do not use, modify,
or distribute the Package, if you do not accept this license.

If your Modified Version has been derived from a Modified Version made
by someone other than you, you are nevertheless required to ensure that
your Modified Version complies with the requirements of this license.

This license does not grant you the right to use any trademark, service
mark, tradename, or logo of the Copyright Holder.

This license includes the non-exclusive, worldwide, free-of-charge
patent license to make, have made, use, offer to sell, sell, import and
otherwise transfer the Package with respect to any patent claims
licensable by the Copyright Holder that are necessarily infringed by the
Package. If you institute patent litigation (including a cross-claim or
counterclaim) against any party alleging that the Package constitutes
direct or contributory patent infringement, then this Artistic License
to you shall terminate on the date that such litigation is filed.

Disclaimer of Warranty: THE PACKAGE IS PROVIDED BY THE COPYRIGHT HOLDER
AND CONTRIBUTORS "AS IS' AND WITHOUT ANY EXPRESS OR IMPLIED WARRANTIES.
THE IMPLIED WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR
PURPOSE, OR NON-INFRINGEMENT ARE DISCLAIMED TO THE EXTENT PERMITTED BY
YOUR LOCAL LAW. UNLESS REQUIRED BY LAW, NO COPYRIGHT HOLDER OR
CONTRIBUTOR WILL BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, OR
CONSEQUENTIAL DAMAGES ARISING IN ANY WAY OUT OF THE USE OF THE PACKAGE,
EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.


=cut