NAME
    WebService::NetSuite - A perl interface to the NetSuite SuiteTalk (Web
    Services) API

SYNOPSIS
        use WebService::NetSuite;
  
        my $ns = WebService::NetSuite->new({
            nsemail         => 'blarg@foo.com',
            nspassword      => 'foobar123',
            nsroleName      => 'Administrator',
            nsaccountName   => 'My NS Account',
        });

        # old 'new' method still supported, but discouraged:
        #my $ns = WebService::NetSuite->new({
        #    nsrole     => 3,
        #    nsemail    => 'blarg@foo.com',
        #    nspassword => 'foobar123',
        #    nsaccount  => 123456,
        #    sandbox    => 1,
        #});

        my $customer_id = $ns->add( 'customer',
            { firstName  => 'Gonzo',
              lastName   => 'Muppet',
              email      => 'gonzo@muppets.com',
              entityId   => 'muppet_database_id',
              subsidiary => 1,
              isPerson   => 1,
        });

DESCRIPTION
    This module is a client to the NetSuite SuiteTalk web service API.

    Initial content shamelessly stolen from
    https://github.com/gitpan/NetSuite

    Refactored and released as WebService::NetSuite for the 2013 target and
    updated access methods using the passport data structure instead of
    login/logout.

    This reboot of the original NetSuite module is still rough and under
    construction.

    NetSuite Help Center -
    https://system.sandbox.netsuite.com/app/help/helpcenter.nl

    You'll need a NetSuite login to get to the help center unfortunately.
    Silly NetSuite.

  new(%options)
    The new method creates the WebService::NetSuite object and the
    underlying SOAP object that is used to communicate with NetSuite. The
    new symtax automatically determines the NetSuite host to communicate
    with based on your email, password, and account name:

        NEW SYNTAX:

        my $ns = WebService::NetSuite->new({
            nsemail         => 'blarg@foo.com',
            nspassword      => 'foobar123',
            nsroleName      => 'Administrator',
            nsaccountName   => 'My NS Account',
            sandbox         => 0,
            debug           => 1,
            debugFile       => 'NetSuite.dbg',
        });

        OLD SYNTAX:

        my $ns = WebService::NetSuite->new({
            nsrole          => 3,
            nsemail         => 'blarg@foo.com',
            nspassword      => 'foobar123',
            nsaccount       => 123456,
            sandbox         => 0,
            debug           => 1,
            debugFile       => 'NetSuite.dbg',
        });

  add(recordType, hashReference)
    The add method submits a new record to NetSuite. It requires a record
    type, and hash reference containing the data of the record.

    For a boolean value, the request uses a numeric zero to represent false,
    and the textual word "true" to represent true. I believe this is an
    error with NetSuite; identified in their last release.

    For a record reference field, like entityStatus, simply pass the numeric
    internalId of the field. If you are unsure what the internalIds are for
    a value, check the getSelectValue method.

    For an enumerated value, simply submit a string.

    For a list value, pass an array of hashes.

        my $customer = {
            isPerson => 0, # meaning false
            companyName => 'Wolfe Electronics',
            entityStatus => 13, # notice I only pass in the internalId
            emailPreference => '_hTML', # enumerated value
            unsubscribe => 0,
            addressbookList => [
              {
                  defaultShipping => 'true',
                  defaultBilling => 0,
                  isResidential => 0,
                  phone => '650-627-1000',
                  label => 'United States Office',
                  addr1 => '2955 Campus Drive',
                  addr2 => 'Suite 100',
                  city => 'San Mateo',
                  state => 'CA',
                  zip => '94403',
                  country => '_unitedStates',
              },
            ],
        };

        my $internalId = $ns->add('customer', $customer);
        print "I have added a customer with internalId $internalId\n";

    If successful this method will return the internalId of the newly
    generated record. Otherwise, the error details are sent to the
    errorResults method.

    If you wanted to ensure a record was submitted successfully, I recommend
    the following syntax:

        if (my $internalId = $ns->add('customer', $customer)) {
            print "I have added a customer with internalId $internalId\n";
        }
        else {
            print "I failed to add the customer!\n";
        }

  update(recordType, hashReference)
    The update method will request an update of an existing record. The only
    difference with this operation is that the internalId of the record
    being updated must be present inside the hash reference.

        my $customer = {
            internalId => 1234, # the internaldId of the record being updated
            phone => '555-555-5555',
        };

        my $internalId = $ns->update('customer', $customer);
        print "I have updated a customer with internalId $internalId\n";

    If successful this method will return the internalId of the updated
    record Otherwise, the error details are sent to the errorResults method.

  delete(recordType, hashOrInternalId)
    The delete method very simply deletes a record. It requires the record
    type and either a hashref indicating the criteria or internalId number
    for the record.

        The first 2 examples are exactly the same:

        1) my $internalId = $ns->delete('customer', 1234);
           print "I have deleted a customer with internalId $internalId\n";

        2) my $internalId = $ns->delete('customer', {internalId => 1234});
           print "I have deleted a customer with internalId $internalId\n";

        3) my $internalId = $ns->delete('customer', {externalId => 5678});
           print "I have deleted a customer with internalId $internalId\n";

    If successful this method will return the internalId of the deleted
    record Otherwise, the error details are sent to the errorResults method.

  search(searchType, hashReference, configReference)
    The search method submits a query to NetSuite. If the query is
    successful, a true value (1) is returned, otherwise it is undefined.

    To conduct a very basic search for all customers, excluding inactive
    accounts, I would write:

        my $query = {
            basic => [
                { name => 'isInactive', value => 0 } # 0 means false
            ]
        };
    
        $ns->search('customer', $query);

    Notice that the query is a hash reference of search types. Foreach
    search type in the hash there is an array of hashes for each field in
    the criteria.

    Once the query is constructed, I designate the search to use and the
    query. And submit it to NetSuite.

    This query structure may seem confusing, especially in a simply example.
    But within NetSuite there are several different searches you can
    perform. Some examples of these searchs are:

    customer contact supportCase employee calendarEvent item opportunity
    phoneCall task transaction

    Then within each search, you can also join with other searches to
    combine information. To demonstrate a more complex search, we will take
    this example.

    Let's imagine you wanted to see transactions, specifically sales orders,
    invoices, and cash sales, that have transpired over the last year.

        my $query = {
            basic => [
                { name => 'mainline', value => 'true' },
                { name => 'type', attr => { operator => 'anyOf' }, value => [
                        { value => '_salesOrder' },
                        { value => '_invoice' },
                        { value => '_cashSale' },
                    ]   
                },
                { name => 'tranDate', value => 'previousOneYear', attr => { operator => 'onOrAfter' } },
            ],
        };

    From that list, you want to see if the customer associated with each
    transaction has a valid email address on file, and is not a lead or a
    prospect. The joined query would look like this:

        my $query = {
            basic => [
                { name => 'mainline', value => 'true' },
                { name => 'type', attr => { operator => 'anyOf' }, value => [
                        { value => '_salesOrder' },
                        { value => '_invoice' },
                        { value => '_cashSale' },
                    ]   
                },
                { name => 'tranDate', value => 'previousOneYear', attr => { operator => 'onOrAfter' } },
            ],
            customerJoin => [
                { name => 'email', attr => { operator => 'notEmpty' } },
                { name => 'entityStatus', attr => { operator => 'anyOf' }, value => [
                        { attr => { internalId => '13' } },
                        { attr => { internalId => '15' } },
                        { attr => { internalId => '16' } },
                    ]                                  
                },
            ],
        };

    Notice that each hash reference within either the basic or customerJoin
    arrays has a "name" and "value" key. In some cases you also have an
    "attr" key. This "attr" key is another hash reference that contains the
    operator for a field, or the internalId for a field.

    Also notice that for enumerated search fields, like "entityStatus" or
    "type", the "value" key contains an array of hashes. Each of these
    hashes represent one of many possible collections.

    To take this a step further, we may want to search for some custom
    fields that exists in a customer's record. These custom fields are
    located in the "customFieldList" field of a record and can be queries
    like so:

        my $query = {
            basic => [
                { name => 'customFieldList', value => [
                        {
                            name => 'customField',
                            attr => {
                                internalId => 'custentity1',
                                operator => 'anyOf',
                                'xsi:type' => namespace('core') . ':SearchMultiSelectCustomField'
                            },
                            value => [
                                { attr => { internalId => 1 } },
                                { attr => { internalId => 2 } },
                                { attr => { internalId => 3 } },
                                { attr => { internalId => 4 } },
                            ]
                        },
                    ],
                },
            ],
        };

    Notice that we have added a new layer to the "attr" key called
    'xsi:type'. That is because this module cannot determine the custom
    field types for YOUR particular NetSuite account in real time. Thus, you
    have to provide them within the query.

    If the search is successful, a true value (1) is returned, otherwise it
    is undefined. If successful, the results are passed to the searchResults
    method, otherwise call the errorResults method.

    Also, for this method, you are given special access to the header of the
    search request. This allows you to designate the number of records to be
    returned in each set, as well as whether to return just basic
    information about the results, or extended information about the
    results.

        # perform a search and only return 10 records per page
        $ns->search('customer', $query, { pageSize => 10 });
    
        # perform a search and only provide basic information about the results
        $ns->search('customer', $query, { bodyFieldsOnly => 0 });

  searchResults
    The searchResults method returns the results of a successful search
    request. It is a hash reference that contains the record list and
    details of the search.

        {
            'recordList' => [
                {
                    'accessRoleName' => 'Customer Center',
                    'priceLevelInternalId' => '3',
                    'unbilledOrders' => '2512.7',
                    'entityStatusName' => 'CUSTOMER-Closed Won',
                    'taxItemInternalId' => '-112',
                    'lastPageVisited' => 'login-register',
                    'isInactive' => 'false',
                    'shippingItemName' => 'UPS Ground',
                    'entityId' => 'A Wolfe',
                    'entityStatusInternalId' => '13',
                    'accessRoleInternalId' => '14',
                    'recordExternalId' => 'entity-5',
                    'webLead' => 'No',
                    'territoryName' => 'Default Round-Robin',
                    'recordType' => 'customer',
                    'emailPreference' => '_default',
                    'taxItemName' => 'CA-SAN MATEO',
                    'taxable' => 'true',
                    'partnerName' => 'E Auctions Online',
                    'companyName' => 'Wolfe Electronics',
                    'shippingItemInternalId' => '92',
                    'leadSourceName' => 'Accessory Sale',
                    'creditHoldOverride' => '_auto',
                    'title' => 'Perl Developer',
                    'priceLevelName' => 'Employee Price',
                    'partnerInternalId' => '170',
                    'giveAccess' => 'true',
                    'visits' => '150',
                    'stage' => '_customer',
                    'termsName' => 'Due on receipt',
                    'defaultAddress' => 'A Wolfe<br>2955 Campus Drive<br>Suite 100
    <br>San Mateo CA 94403<br>United States',
                    'lastVisit' => '2008-03-22T16:40:00.000-07:00',
                    'isPerson' => 'false',
                    'recordInternalId' => '-5',
                    'fax' => '650-627-1001',
                    'salesRepInternalId' => '23',
                    'dateCreated' => '2006-07-22T00:00:00.000-07:00',
                    'termsInternalId' => '4',
                    'salesRepName' => 'Clark Koozer',
                    'unsubscribe' => 'false',
                    'categoryInternalId' => '2',
                    'phone' => '650-555-9788',
                    'shipComplete' => 'false',
                    'lastModifiedDate' => '2008-01-28T19:28:00.000-08:00',
                    'territoryInternalId' => '-5',
                    'categoryName' => 'Individual',
                    'firstVisit' => '2007-03-24T16:13:00.000-07:00',
                    'leadSourceInternalId' => '100102'
                },
            ],
            'totalPages' => '79', # the total number of pages in the set
            'totalRecords' => '790', # the total records returned by the search
            'pageSize' => '10', # the number of records per page
            'pageIndex' => '1', # the current page
            'statusIsSuccess' => 'true'
        }

    The "recordList" field is an array of hashes containing a record's
    values. Refer to the get method for details on the understanding of a
    record's data structure.

  searchMore(pageIndex)
    If your initial search returns several pages of results, you can jump to
    another result page quickly using the searchMore method.

    For example, if after performing an initial search you are given 1 of
    100 records, when there are 500 total records. You could quickly jump to
    the 301-400 block of records by entering the pageIndex value.

        $ns->search('customer', $query);
    
        # determine my result set
        my $totalPages = $ns->searchResults->{totalPages};
        my $pageIndex = $ns->searchResults->{pageIndex};
        my $totalRecords = $ns->searchResults->{totalRecords};
    
        # output a message
        print "I found $totalRecords records!\n";
        print "Displaying page $pageIndex of $totalPages\n";
    
        my $jumpToPage = 3;
        $ns->searchMore($jumpToPage);
        print "Jumping to page $jumpToPage\n";
        print "Now displaying page $jumpToPage of $totalPages\n";

  searchNext
    If your initial search returns several pages of results, you can
    automatically jump to the next page of results using the searchNext
    function. This is most useful when downloading sets of more than 1000
    records. (Which is the limit of an initial search).

        $ns->search('transaction', $query);
        if ($ns->searchResults->{totalPages} > 1) {
            while ($ns->searchResults->{pageIndex} != $ns->searchResults->{totalPages}) {
                for my $record (@{ $ns->searchResults->{recordList} }) {
                    my $internalId = $record->{recordInternalId};
                    print "Found record with internalId $internalId\n";
                }
                $ns->searchNext;
            }
        }

  get(recordType, hashOrInternalId)
    The get method returns the most complete information for a record. It
    takes a hash which describes the criteria or an internalId.

    The first 2 examples are identical:

    1) $ns->get('customer', 1234)

    2) $ns->get('customer', {internalId => 1234})

    3) $ns->get('customer', {externalId => 5678})

        # to see an individual field in the response
        if ($ns->get('customer', 1234)) {
            my $firstName = $ns->getResults->{firstName};
            print "I got a customer with the first name $firstName\n";
        }
    
        # to output the complete data structure
        my $getSuccess = $ns->get('customer', 1234);
        if ($getSuccess) {
            print Dumper($ns->getResults);
        }

    If the operation in successful, a true value (1) is returned, otherwise
    it is undefined.

    The results will be passed to the getResults method, otherwise call the
    errorResults method.

  getResults
    The getResults method returns a hash reference containing all of the
    information for a given record. (Some fields were omitted)

        {
            'recordInternalId' => '1234',
            'recordExternalId' => 'entity-5',
            'recordType' => 'customer',
            'isInactive' => 'false',
            'entityStatusInternalId' => '13',
            'entityStatusName' => 'CUSTOMER-Closed Won',
            'entityId' => 'A Wolfe',
            'emailPreference' => '_default',
            'fax' => '650-627-1001',
            'contactList' => [
                {
                    'contactInternalId' => '25',
                    'contactName' => 'Amy Nguyen'
                },
            ],
            'creditCardsList' => [
                {
                    'ccDefault' => 'true',
                    'ccMemo' => 'This is the preferred credit card.',
                    'paymentMethodName' => 'Visa',
                    'paymentMethodInternalId' => '5',
                    'ccNumber' => '************1111',
                    'ccExpireDate' => '2010-01-01T00:00:00.000-08:00',
                    'ccName' => 'A Wolfe'
                }
            ],
            'addressbookList' => [
                {
                    'country' => '_unitedStates',
                    'defaultShipping' => 'true',
                    'internalId' => '244715',
                    'defaultBilling' => 'true',
                    'phone' => '650-627-1000',
                    'state' => 'CA',
                    'addrText' => 'A Wolfe<br>2955 Campus Drive<br>Suite 100<br>San Mateo CA 94403<br>United States',
                    'addr2' => 'Suite 100',
                    'zip' => '94403',
                    'city' => 'San Mateo',
                    'isResidential' => 'false',
                    'addressee' => 'A Wolfe',
                    'addr1' => '2955 Campus Drive',
                    'override' => 'false',
                    'label' => 'Default'
                }
            ],
            'dateCreated' => '2006-07-22T00:00:00.000-07:00',
            'lastModifiedDate' => '2008-01-28T19:28:00.000-08:00',
        };

    It is important to note how some of this data is returned.

    Notice that the internalId for the record is labeled "recordInternalId"
    instead of just "internalId". This is the same for the
    "recordExternalId".

    For a boolean value, the response the string "true" or "false.

    For a record reference field, like entityStatus, the name of this value
    and its internalId are returned as two seperate values: entityStatusName
    and entityStatusInternalId. This appending of the words "Name" and
    "InternalId" after the field name is the same for all reference fields.

    For an enumerated value, a string is returned.

    For a list, the value is an array of hashes. Even if the list contains
    only a single hash reference, it will still be returned as an array.

    The easiest way to access an understand this function, is to dump the
    response and determine the best way to interate through your data. For
    example, if I wanted to see if the customer had a default credit card
    selected, I might write:

        if ($ns->get('customer', 1234)) {
            if (defined $ns->getResults->{creditCardsList}) {
                if (scalar @{ $ns->getResults->{creditCardsList} } == 1) {
                    print "This customer has a default credit card!\n";
                }
                else { 
                    for my $creditCard (@{ $ns->getResults->{creditCardsList} }) {
                        if ($creditCard->{ccDefault} eq 'true') {
                            print "This customer has a default credit card!\n";
                        }
                    }
                }
            }
            else {
                "There are no credit cards on file!\n";
            }
        }
        else {
            # my get request failed, better check the errorResults method
        }

    Or, if I was more concerned with checking this customers last activity,
    I might write:

        $ns->get('customer', 1234);
    
        # assuming the request was successful
        my $internalId = $ns->getResults->{recordInternalId};
        my $lastModifiedDate = $ns->getResults->{lastModifiedDate};
        print "Customer $internalId was last updated on $lastModifiedDate.\n";

  getSelectValue
    The getSelectValue method returns a list of internalId numbers and names
    for a record reference field. For instance, if you wanted to know all of
    the acceptable values for the "terms" field of a customer you could
    submit a request like:

        $ns->getSelectValue('customer_terms');

    If successful, a call to the getResults method, will return a hash
    reference that looks like this:

        {
            'recordRefList' => [
              {
                  'recordRefInternalId' => '5',
                  'recordRefName' => '1% 10 Net 30'
              },
              {
                  'recordRefInternalId' => '6',
                  'recordRefName' => '2% 10 Net 30'
              },
              {
                  'recordRefInternalId' => '4',
                  'recordRefName' => 'Due on receipt'
              },
              {
                  'recordRefInternalId' => '1',
                  'recordRefName' => 'Net 15'
              },
              {
                  'recordRefInternalId' => '2',
                  'recordRefName' => 'Net 30'
              },
              {
                  'recordRefInternalId' => '3',
                  'recordRefName' => 'Net 60'
              }
            ],
            'totalRecords' => '6',
            'statusIsSuccess' => 'true'
        }

    If the request fails, the error details are sent to the errorResults
    method.

    From these results, we now know that the "terms" field of a customer can
    be submitted using any of the recordRefInternalIds. Thus, to update a
    customer's terms, we might write:

        my $customer = {
            internalId => 1234,
            terms => 4, # Due on receipt
        }

        $ns->update('customer', $customer);

    For a complete list of acceptable values for this operation, visit the
    coreTypes XSD file for web services version 2.6. Look for the
    "GetSelectValueType" simpleType.

    <https://webservices.netsuite.com/xsd/platform/v2_6_0/coreTypes.xsd>

  getCustomization
    The getCustomization retrieves the metadata for Custom Fields, Lists,
    and Record Types. For instance, if you wanted to know all of the custom
    fields for the body of a transaction, you might write:

        $ns->getCustomization('transactionBodyCustomField');

    If successful, a call to the getResults method, will return a hash
    reference that looks like this:

        {
            'recordList' => [
              {
                  'fieldType' => '_phoneNumber',
                  'sourceFromName' => 'Phone',
                  'bodyPrintStatement' => 'false',
                  'bodyAssemblyBuild' => 'false',
                  'bodySale' => 'true',
                  'bodyItemReceiptOrder' => 'false',
                  'isMandatory' => 'false',
                  'recordType' => 'transactionBodyCustomField',
                  'bodyPurchase' => 'false',
                  'bodyPickingTicket' => 'true',
                  'bodyExpenseReport' => 'false',
                  'name' => 'Entity',
                  'bodyItemFulfillmentOrder' => 'false',
                  'bodyPrintPackingSlip' => 'false',
                  'isFormula' => 'false',
                  'sourceFromInternalId' => 'STDENTITYPHONE',
                  'bodyItemFulfillment' => 'false',
                  'label' => 'Customer Phone',
                  'bodyJournal' => 'false',
                  'showInList' => 'false',
                  'recordInternalId' => 'CUSTBODY1',
                  'help' => 'This is the customer\'s phone number from the
    customer record.  It is generated dynamically every time the form is accessed
     - so that changes in the customer record will be reflected the next time the
     transaction is viewed/edited/printed.<br>Note: This is an example of a
     transaction body field, sourced from a customer standard field.',
                  'storeValue' => 'false',
                  'isParent' => 'false',
                  'defaultChecked' => 'false',
                  'bodyInventoryAdjustment' => 'false',
                  'bodyOpportunity' => 'false',
                  'bodyPrintFlag' => 'true',
                  'checkSpelling' => 'false',
                  'displayType' => '_disabled',
                  'bodyItemReceipt' => 'false',
                  'sourceListInternalId' => 'STDBODYENTITY',
                  'bodyStore' => 'false'
              },
              'totalRecords' => '1',
              'statusIsSuccess' => 'true'
        };

    If the request fails, the error details are sent to the errorResults
    method.

    For a complete list of acceptable values for this operation, visit the
    coreTypes XSD file for web services version 2.6. Look for the
    "RecordType" simpleType.

    <https://webservices.netsuite.com/xsd/platform/v2_6_0/coreTypes.xsd>

  attach(attachRequest)
=head2 detach(detachRequest)
    At this time, only a basic reference is supported, Contact reference is
    not supported yet.

    As an example, to attach a file to an expenseReport, you would do the
    following:

        sub nsRecRef {
            my ($rectype, $id) = @_;
            return  { type => $rectype, internalId => $id };
        }
 
        my $attachRequest = {
            attachTo        => nsRecRef('expenseReport', $erId),
            attachedRecord  => nsRecRef('file',          $fid)
        };
        $ns->attach($attachRequest) or nsfatal 'error attaching';

        The detach operation is coded exactly the same.

  errorResults
    The errorResults method is populated when a request returns an erroneous
    response from NetSuite. These errors can occur at anytime and with any
    operation. Always assume your operations will fail, and build your code
    accordingly.

    The hash reference that is returned looks like this:

        {
            'message' => 'You have entered an invalid email address or account
    number. Please try again.',
            'code' => 'INVALID_LOGIN_CREDENTIALS'
        };

    If there is something FUNDAMENTALLY wrong with your request (like you
    have included an invalid field), your errorResults may look like this:

        {
            'faultcode' => 'soapenv:Server.userException',
            'detailDetail' => 'partners-java002.svale.netledger.com',
            'faultstring' => 'com.netledger.common.schemabean.NLSchemaBeanException:
    <<somefield>> not found on {urn:relationships_2_6.lists.webservices.netsuite.com}Customer'
        };

    Thus, a typical error-prepared script might look like this:

        $ns->login or die "Can't connect to NetSuite!\n";
    
        if ($ns->search('customer', $query)) {
            for my $record (@{ $ns->searchResults->{recordList} }) {
                if ($ns->get('customer', $record->{recordInternalId})) {
                    print Dumper($ns->getResults);
                }
                else {
                    # If an error is encountered while running through
                    # a list, print a notice and break the loop
                    print "An error occured!\n";
                    last;
                }
            }
        }
        else {
        
            # I really want to know why my search would fail
            # lets output the error and message
            my $message = $ns->errorResults->{message};
            my $code = $ns->errorResults->{code};
        
            print "Unable to perform search!\n";
            print "($code): $message\n";
        
        }
    
        $ns->logout; # no error handling here, if this fails, oh well.

    For a complete listing of errors and associated messages, consult the
    SuiteTalk (Web Services) Records Guide.

    <http://www.netsuite.com/portal/developers/resources/suitetalk-documenta
    tion.shtml>

AUTHOR
    Fred Moyer, fred@redhotpenguin.com

LICENCE AND COPYRIGHT
    Copyright 2013, iParadigms LLC.

    Original Netsuite module copyright (c) 2008, Jonathan Lloyd. All rights
    reserved.

    This module is free software; you can redistribute it and/or modify it
    under the same terms as Perl itself. See perlartistic.

ACKNOWLEDGEMENTS
    Initial content shamelessly stolen from
    https://github.com/gitpan/NetSuite

    Thanks to iParadigms LLC for sponsoring the reboot of this module.