Compare commits

..

778 Commits

Author SHA1 Message Date
ricky a97d867610 Fixed the UI, now working 2026-03-05 07:53:23 -05:00
ricky 021a9ec0c6 Added Issue safe attributes for line items 2026-03-05 07:28:30 -05:00
ricky f056c27bc5 Merge branch 'master' into lineitems 2026-03-05 07:08:30 -05:00
ricky 2520892e2c 2026.3.1 2026-03-04 20:14:16 -05:00
ricky b96678a2e9 fixed accident deleteion details_cache_key 2026-03-04 20:09:13 -05:00
ricky bccfcd9dbc cache qbo details to reduce api calls 2026-03-04 20:06:22 -05:00
ricky 8ba99b7db2 Fixed eager loading issues 2026-03-04 19:18:06 -05:00
ricky aff7d0c48e removed uneeded logging of issue and project contents. 2026-03-04 18:37:30 -05:00
ricky e9b3b1c838 Merge branch 'master' into dev 2026-03-04 17:42:42 -05:00
ricky 2fc2f94cd1 Fixed combining of estimate pdf 2026-03-04 13:23:59 -05:00
ricky 681747e08b Got the UI working 2026-03-04 07:55:35 -05:00
ricky 9f9810686f removed logging 2026-03-03 20:36:23 -05:00
ricky f041e1bce4 Added logging for completed pull 2026-03-03 20:05:19 -05:00
ricky d44d5e2fb7 Fixed log prefix 2026-03-03 19:54:50 -05:00
ricky 4403267abb Moved QBO fetch from customer model into service 2026-03-03 19:49:36 -05:00
ricky be400c2b2a Added logging for errors when editing 2026-03-03 19:22:15 -05:00
ricky 23e565a304 raise exceptions if not initialized properly 2026-03-02 22:57:13 -05:00
ricky 2e2b17fac3 log should be private 2026-03-02 22:54:26 -05:00
ricky 28db5cb8c8 removed unused code 2026-03-02 22:50:43 -05:00
ricky 0df15693d2 removed unused begin 2026-03-02 22:49:53 -05:00
ricky f8b1c72394 show all customer when search is blank 2026-03-02 22:49:18 -05:00
ricky 899237c5ab Reduced blanket rescues, added respond_to_missing?, and extracted push into CustomerPushService 2026-03-02 22:41:22 -05:00
ricky f02b50ae26 Added time stamps to each qbo entity model 2026-03-02 07:10:13 -05:00
ricky 485a977d1a Use Safe Navigation Operator &. 2026-03-01 21:31:28 -05:00
ricky 03d5a5d148 Always show sync status 2026-03-01 21:25:07 -05:00
ricky 0deab9dbd3 2026.3.0 2026-03-01 19:35:55 -05:00
ricky 899c9878c4 Fix: only attach invoices if document is updated 2026-03-01 19:27:23 -05:00
ricky eb6beea5fa Initial line item idea 2026-03-01 15:04:25 -05:00
ricky b95a3b6623 Refactor: Update billing error messages in locale for consistency and clarity 2026-03-01 12:35:54 -05:00
ricky ef3f00c445 Refactor: Replace BillingValidator with inline validations in bill method and update error messages in locale 2026-03-01 12:29:20 -05:00
ricky 46f06df995 Removed unused service 2026-03-01 12:14:53 -05:00
ricky b15b88f48d Fix: Correct I18n reference in last_sync method for proper translation 2026-03-01 01:07:43 -05:00
ricky 7b7b07b5fa fixed file name 2026-03-01 01:01:26 -05:00
ricky 16ca1caabc Refactor: Enhance QboWebhookProcessor with logging for signature validation 2026-03-01 00:58:46 -05:00
ricky 69d266bdca formatting 2026-03-01 00:45:27 -05:00
ricky 3728ec2a12 Refactor: Improve address formatting in CustomersController and enhance HTML rendering for billing and shipping addresses 2026-03-01 00:40:11 -05:00
ricky cefa36c880 Removed unsued Customer destroy method 2026-03-01 00:28:57 -05:00
ricky ed111fefe7 Refactor: Update QBO connection handling to use QboConnectionService for consistency across services and controllers 2026-03-01 00:27:06 -05:00
ricky 5a662f67b8 Removed sync from migration 2026-02-28 23:45:55 -05:00
ricky 6e90548dbb Removed unused method 2026-02-28 22:56:03 -05:00
ricky f921f227e2 Refactor: Introduce PdfServiceBase for shared PDF fetching functionality and update Invoice/Estimate controllers to utilize new services 2026-02-28 21:59:01 -05:00
ricky a34ae46358 Refactor: Simplify invoice PDF fetching and introduce InvoicePdfService for better organization 2026-02-28 21:36:16 -05:00
ricky e4cfb0674e Refactor: Enhance estimate loading and syncing logic in EstimateController 2026-02-28 21:20:35 -05:00
ricky 348c521491 Added comments 2026-02-28 20:03:19 -05:00
ricky 6cee8c1d81 Fix: Remove logging of fetched page information in SyncServiceBase 2026-02-28 19:59:50 -05:00
ricky d4a0aa1db5 Refactor: Introduce SyncServiceBase for shared functionality across sync services 2026-02-28 19:47:45 -05:00
ricky 12884a211e Added comments 2026-02-28 18:04:38 -05:00
ricky 4ed71f5667 Fix: Remove unnecessary QBO configuration check in billing enqueue method 2026-02-28 18:01:32 -05:00
ricky 8303dec501 Fix: Raise error if no QBO configuration is found in sync jobs 2026-02-28 17:58:11 -05:00
ricky 9b07ae7073 Fix: Ensure retry_on configuration specifies wait time and attempts for error handling 2026-02-28 17:51:48 -05:00
ricky baf321d4d6 Fix: Update retry_on configuration to specify wait time for error handling 2026-02-28 17:51:31 -05:00
ricky 0a2d38a927 Update plugin version to 2026.2.16
Updated to use active jobs & services for all background work
2026-02-28 09:29:50 -05:00
ricky b80dbaa015 Fix: Update last_update query to use correct timestamp field for customer sync 2026-02-28 09:11:01 -05:00
ricky 9e399b934b Fix: Update last_update query to use correct timestamp field for employee sync 2026-02-28 09:10:55 -05:00
ricky cc6fd07435 Update notice for missing estimate to include syncing information with QuickBooks 2026-02-28 08:53:46 -05:00
ricky 7a50df24d9 Add logging to get_estimate method for better debugging 2026-02-28 08:53:38 -05:00
ricky ca02ead9f9 Use delete not destroy 2026-02-28 08:35:51 -05:00
ricky 9089adaba0 removed uneeded comments 2026-02-28 08:35:35 -05:00
ricky dc6eba8566 Refactor logging in PdfPatch to use custom log method for better clarity and consistency 2026-02-28 08:29:10 -05:00
ricky 19911b7940 Refactor EstimateSyncJob to support syncing by ID and document number; add EstimateSyncService for handling estimate synchronization 2026-02-28 08:24:25 -05:00
ricky a80f59cc45 Refactor sync_by_id method in Estimate model to use EstimateSyncJob for syncing 2026-02-28 07:50:23 -05:00
ricky eee99e4d83 Implement CustomerSyncService for customer synchronization and update CustomerSyncJob to support syncing by ID 2026-02-28 07:50:07 -05:00
ricky b3f01bd372 Refactor persist method in InvoiceSyncService to use 'local' variable for clarity and add logging for updates 2026-02-28 07:41:22 -05:00
ricky d1ba93d61a Refactor persist method in EmployeeSyncService to use 'local' variable for clarity 2026-02-28 07:41:05 -05:00
ricky 9a688c4841 Set primary_key 2026-02-27 23:39:34 -05:00
ricky e94352e2c4 Added comment 2026-02-27 23:19:44 -05:00
ricky ea0f42b68e Added comment 2026-02-27 23:18:40 -05:00
ricky 5a31c194a5 Added comments 2026-02-27 23:15:37 -05:00
ricky 6f8af9bba8 Implement Employee synchronization; add EmployeeSyncJob and EmployeeSyncService for improved background processing and logging 2026-02-27 23:07:12 -05:00
ricky 03109d5775 Refactor invoice processing and synchronization; implement InvoiceSyncJob and related services for improved background processing and logging 2026-02-27 22:33:04 -05:00
ricky a1cbf9a0a9 Refactor logging in CustomerSyncJob to use a centralized log method; enhance consistency and readability of log messages 2026-02-27 22:32:42 -05:00
ricky 9c0f153518 Refactor logging across controllers and jobs to use a centralized log method; improve consistency and readability of log messages 2026-02-27 22:32:07 -05:00
ricky f32b48296d Refactor estimate synchronization to use EstimateSyncJob; remove direct sync logic from Estimate model for improved background processing 2026-02-27 08:29:52 -05:00
ricky 3d37f01bff Added timestamps to estiamtes and invoices 2026-02-27 08:08:39 -05:00
ricky 889e9bf31f Refactor customer synchronization to use CustomerSyncJob; remove direct sync logic from Customer model for improved background processing 2026-02-27 08:00:38 -05:00
ricky 208e839e6a Refactor CustomerToken model for improved token management; streamline token generation and expiration handling, and enhance association with issues 2026-02-26 21:05:02 -05:00
ricky 4f55751500 Refactor QuickBooks webhook handling to use ActiveJob for processing; improve security with signature verification and streamline entity processing 2026-02-26 20:30:20 -05:00
ricky a64016eb95 Refactor QBO billing to use ActiveJob; remove threaded billing and add manual job enqueue support 2026-02-26 19:48:29 -05:00
ricky 5d858ae186 Enhance customer search functionality by ordering results and refining search method 2026-02-25 22:05:52 -05:00
ricky b38f850df3 2026.2.15 2026-02-25 21:13:55 -05:00
ricky 138e55933b Fixed creation of new customers. 2026-02-25 15:32:45 -05:00
ricky 5fbc169ade Restored old search 2026-02-25 08:08:02 -05:00
ricky d6737a6747 2026.2.14 2026-02-22 19:11:14 -05:00
ricky 65db8f00a8 Improve customer search with Full-Text index and phonetic matching 2026-02-22 19:07:20 -05:00
ricky 0197dc2a30 removed unused method 2026-02-22 13:34:23 -05:00
ricky cd1caa502d Merge branch 'master' into dev 2026-02-22 13:32:01 -05:00
ricky 4b45d24a75 Enhance Customer model with redmine's built in searchable and event capabilities 2026-02-22 13:31:28 -05:00
ricky 64a4526aa4 2026.2.13 2026-02-21 19:08:32 -05:00
ricky 3514401808 Add unique IDs to search forms for customers and estimates 2026-02-21 19:07:40 -05:00
ricky 3deafd8a6d Fixed search event_url 2026-02-21 11:35:15 -05:00
ricky a54de28db5 Extending customers to Redmine's built in search 2026-02-21 11:20:20 -05:00
ricky 6434eea906 2026.2.12 2026-02-21 08:24:36 -05:00
ricky 9b656534ae Sanitize search, no little bobby tables 2026-02-21 08:23:58 -05:00
ricky 659a1fbcf0 2026.2.11 2026-02-20 19:11:31 -05:00
ricky 4dc1f5d0bd Enhance billing functionality in IssuePatch with detailed logging and self-references 2026-02-20 09:47:47 -05:00
ricky 02f34582f4 2026.2.10
Addressed the Bullet (the N+1 query detector) warning to include customers
2026-02-16 18:56:09 -05:00
ricky 2f9ef6304f scope.includes(:customer) 2026-02-16 18:53:29 -05:00
ricky 886d5f4ace 2026.2.9 2026-02-16 08:15:46 -05:00
ricky 1ade938eb3 Fixed Querying issues by customer name 2026-02-16 08:13:57 -05:00
ricky 3111f391f3 Filter by customer works now 2026-02-15 21:34:22 -05:00
ricky d2b9113914 2026.2.8 2026-02-14 18:57:22 -05:00
ricky 447e048819 updated screensots 2026-02-14 09:32:40 -05:00
ricky e7dfc3f2ad added sync estimates by id 2026-02-14 08:25:02 -05:00
ricky 139f5dd618 render partial on footer 2026-02-13 22:39:54 -05:00
ricky 9c11704d03 Added label_create_estimate 2026-02-13 20:45:59 -05:00
ricky 2ae53adf08 added needed trailing space 2026-02-13 20:37:42 -05:00
ricky 877c1b78a5 removed old comment 2026-02-13 18:38:42 -05:00
ricky 1d47703206 fixed indentiation 2026-02-13 18:37:12 -05:00
ricky a069556ed9 Alphabetized: All keys are now in A-Z order for easier maintenance.
Branding: Changed "Quickbooks" to QuickBooks (capitalized B) to match official branding.

Grammar:

warn_ru_sure: Changed "You sure?" to "Are you sure?"

label_no_customers: Refined to "There are no customers matching the search term(s)."

label_billing_error: Added punctuation and clarified the phrasing.

Cleanup: Removed unnecessary spaces inside quotes (e.g., " Make" and "Matching ").
2026-02-13 18:33:48 -05:00
ricky 359c582e22 Fixed partial billing and added flash messages 2026-02-13 18:25:07 -05:00
ricky e63b9e4217 use safe_join 2026-02-13 07:32:20 -05:00
ricky 6fd355d8cc 2026.2.7 2026-02-12 19:00:55 -05:00
ricky e6b57392d1 Merge branch '422' 2026-02-12 18:59:32 -05:00
ricky 331c1eabeb seems to work without overiding the main issues _form 2026-02-11 19:59:41 -05:00
ricky 167385bb99 override issues form to add passing @project to issues hook 2026-02-11 19:31:01 -05:00
ricky 11b9876d4f Removed unused controller_issues_new_before_save that was used for finding the 422 error 2026-02-11 08:09:00 -05:00
ricky 9cf72821b0 2026.2.6 2026-02-11 08:05:28 -05:00
ricky 57adcce431 Refactor JavaScript path handling for issue form updates to help prevent 422 errors on new issue creation 2026-02-11 07:59:18 -05:00
ricky 7fdb15f7e8 more logging 2026-02-10 22:09:16 -05:00
ricky 6e11e05a24 2026.2.5 2026-02-09 21:54:01 -05:00
ricky a6751d3f41 simplified appointment link javascript 2026-02-09 21:43:29 -05:00
ricky 8944e92ffc Use html data attributes 2026-02-09 20:52:27 -05:00
ricky f0c0a42c96 2026.2.4 2026-02-09 20:04:47 -05:00
ricky a4b51457bb moved controller_issues_new_before_save nito issues_hook_listener 2026-02-09 20:04:16 -05:00
ricky fb4a883b43 Added logging 2026-02-09 14:31:56 -05:00
ricky c24ec93335 force a tracker if still nil 2026-02-09 14:21:50 -05:00
ricky df49964bf9 added tracker nil check 2026-02-09 14:19:31 -05:00
ricky 502ba94465 Readded missing controller_issues_new_before_save 2026-02-09 14:12:01 -05:00
ricky ff038fe5ae removed method: :get from estimate link 2026-02-09 10:41:48 -05:00
ricky 3eed122598 fixed typo 2026-02-09 10:38:10 -05:00
ricky d8d34540a9 use link_to, as button_to doesn't allow for openign links in a new tab 2026-02-09 10:35:12 -05:00
ricky c01cc5ca97 2026.2.3 2026-02-09 09:36:50 -05:00
ricky 6a2f7a1146 initialize string link 2026-02-09 09:21:42 -05:00
ricky f4c844f097 2026.2.2 2026-02-08 19:59:28 -05:00
ricky 1135c69e1b Don't change link text 2026-02-08 19:53:50 -05:00
ricky ef86d222cb Removed change link button 2026-02-08 19:52:17 -05:00
ricky be88a601ae Merge branch 'dev' into js 2026-02-08 19:50:55 -05:00
ricky e6c4e81df2 Remove unused add_appointment method from CustomersController 2026-02-08 19:48:01 -05:00
ricky f4a979672f update link on change 2026-02-08 19:45:58 -05:00
ricky 8a4d64ffc0 Update appointment link with selected invoice links 2026-02-08 18:25:20 -05:00
ricky ac05d38763 2026.2.1 2026-02-08 13:15:20 -05:00
ricky 548dc4fba8 Implement issue creation error handling; add project validation and refactored issue hooks 2026-02-08 13:07:28 -05:00
ricky 7a73b7e8a9 refactor error handling in issue creation; remove unused reload_new_issue method 2026-02-08 10:58:39 -05:00
ricky b38bd951f7 fiex typo tracker not project 2026-02-08 10:02:42 -05:00
ricky 0e3318efdd Added prefilters to help locate 422 on issue creation.
This is an effort to figure out why I get 422 Unprocessable Entity errors sometimes when creating new issues.
2026-02-08 09:58:34 -05:00
ricky d063494bd2 removed empty link string 2026-02-06 23:00:26 -05:00
ricky e35a2148eb 2026.2.0 2026-02-06 19:37:32 -05:00
ricky c8f115ae02 removed unused js 2026-02-06 19:28:42 -05:00
ricky d59e52b111 removed subject from logs 2026-02-06 18:58:16 -05:00
ricky 2c3548d1ac Added logging 2026-02-06 18:55:49 -05:00
ricky d80007bc84 Titleize the subject before save , but keep words containing numbers mixed with letters capitalized 2026-02-06 18:52:11 -05:00
ricky 5d7d9a81bb inital start of using javascript to update the appointment link with selected document links 2026-02-06 17:58:02 -05:00
ricky b030f85b74 Botton_to didn't work for google cal link 2026-02-04 21:39:18 -05:00
ricky 2f0ee6a6d6 refoactored hooks and patches under RedmineQbo module 2026-02-04 07:18:25 -05:00
ricky 637cfa89b4 Refactored view hooks 2026-02-03 22:52:34 -05:00
ricky c36f4c905b moved issue hooks into issue_patch 2026-02-03 22:42:08 -05:00
ricky 83fb20044d updated plugin description 2026-01-31 22:10:23 -05:00
ricky 928e632dd3 converted links to buttons 2026-01-31 21:02:22 -05:00
ricky 8b9cf5066e only show checkboxes if more then 1 invoices 2026-01-31 20:44:15 -05:00
ricky 45bfce87d8 2026.1.9 2026-01-31 13:02:21 -05:00
ricky 6f33e9d23d Inverted conditionals in estimates and invoices lists to use 'unless' 2026-01-31 12:54:07 -05:00
ricky 92460392b9 Refactor estimates and invoices rendering to use local variables for the abilty for other plugins to display filtered lists 2026-01-31 12:45:47 -05:00
ricky f1bdf59697 Add customer details translation and fix link in appointment action 2026-01-30 23:05:23 -05:00
ricky 60e2f1d2b0 2026.1.8 2026-01-30 21:30:28 -05:00
ricky 6c9ae82f81 fixed typo 2026-01-30 21:22:08 -05:00
ricky 42e4494f6e fixed entity, they need strings 2026-01-30 21:17:17 -05:00
ricky 7e0b2c9d09 Removed unused folders 2026-01-30 20:51:09 -05:00
ricky 5ca68b01b6 removed unsued tests 2026-01-30 20:46:04 -05:00
ricky ebd4fa7363 Add flash error messages for forbidden access in permission checks 2026-01-30 20:40:57 -05:00
ricky e6818958ae Add I18n support for flash messages and update locale file with new notices 2026-01-30 20:19:12 -05:00
ricky 5b31459629 use a symbol for empty 2026-01-30 19:28:33 -05:00
ricky 92de2928f6 Merge branch 'master' into symbols 2026-01-30 17:12:11 -05:00
ricky a8af180de2 2026.1.7 2026-01-30 17:09:34 -05:00
ricky e621dc9e3a nil not nill 2026-01-30 17:00:29 -05:00
ricky c3d7c1c867 Use symbols 2026-01-30 16:58:14 -05:00
ricky defeec7f8e 2026.1.6 2026-01-30 07:53:51 -05:00
ricky 37c302e274 use symbol keys 2026-01-30 07:53:29 -05:00
ricky 006e907b35 need to supply selected id 2026-01-30 07:53:14 -05:00
ricky f1f77a8022 use locale 2026-01-30 07:42:09 -05:00
ricky ff358d806e 2026.1.5 2026-01-30 07:22:11 -05:00
ricky b80e1d4e28 loose the hash rocket, use symbol keys 2026-01-29 22:30:59 -05:00
ricky f24128ef75 Cleaned up Issue Show Hook 2026-01-29 21:53:18 -05:00
ricky d3a8c05f50 Added missing to_s method 2026-01-29 21:36:50 -05:00
ricky f023cd246d Titlize new issue subjects 2026-01-29 21:18:15 -05:00
ricky b7e18a3c3f Cleaned up Issue Form Hook 2026-01-29 21:12:32 -05:00
ricky 67f2dbf4d8 Removed reduntant onchange js call to update issue 2026-01-28 17:35:12 -05:00
ricky 924aa7657b Fixed typo 2026-01-28 10:11:28 -05:00
ricky 16fe07f177 Fixed typo 2026-01-28 08:50:37 -05:00
ricky 9257b2f938 2026.1.4 2026-01-28 08:42:33 -05:00
ricky 0227681e92 Added hook list to README 2026-01-28 08:41:55 -05:00
ricky c034696810 Disabled showing assinee to customer 2026-01-28 08:38:02 -05:00
ricky ffdabccd84 Updated README with information about plugin hooks 2026-01-28 08:18:29 -05:00
ricky 1f03908040 Refactor employee and estimate models by removing redundant comments; update locale file to remove item references; delete unused QboItemTest file. 2026-01-28 07:45:20 -05:00
ricky 43a5317b4e Removed DEVELOPMENT from plugin name 2026-01-28 07:03:08 -05:00
ricky 4c49ec6890 2026.1.3 2026-01-27 21:17:17 -05:00
ricky ef7faee685 Process updates from the hooks 2026-01-27 20:50:54 -05:00
ricky 02b48d2de4 Dont add reference for customers to projects 2026-01-27 18:07:37 -05:00
ricky e670d99766 Removed project level intergration 2026-01-27 18:01:42 -05:00
ricky 241dd594d0 Fixed pdf hooks 2026-01-26 23:22:23 -05:00
ricky b603cb634a Version Bump 2026.1.2 2026-01-26 22:59:48 -05:00
ricky 1308a05011 Removed Load Customer Link 2026-01-26 21:00:16 -05:00
ricky 334ed60bf7 removed extra comma 2026-01-26 20:14:01 -05:00
ricky d63bf809f2 Merge branch 'redmine-6' into dev-6 2026-01-26 10:16:40 -05:00
ricky 31406af681 Version bump 2026.1.1 2026-01-26 10:09:24 -05:00
ricky 479be461a6 Dynamically load updateIssueFrom javascript path 2026-01-26 10:08:53 -05:00
ricky c1af031d22 Link to sandbox if enabled 2026-01-23 21:40:08 -05:00
ricky a741cd0217 update readme 2026-01-22 21:54:03 -05:00
ricky 4ae9374401 Merge branch 'redmine-6' into dev-6 2026-01-22 21:50:40 -05:00
ricky b096244454 Better display of customer note 2026-01-22 21:50:17 -05:00
ricky 4983cd661c Fixed missing locale 2026-01-22 21:49:33 -05:00
ricky 5f6fb4af27 Only show select all when more than one invoice 2026-01-22 21:29:08 -05:00
ricky 2f2c74403f Bug fixes 2026-01-22 21:20:47 -05:00
ricky 43579d73e5 Merge remote-tracking branch 'origin/hooks' into dev-6 2026-01-22 20:58:06 -05:00
ricky a90d6b839f Updated readme 2026-01-22 20:57:09 -05:00
ricky e76f977ca8 Removed Vehicles 2026-01-22 20:54:06 -05:00
ricky 7f821d241c Removed typo 2026-01-22 20:47:31 -05:00
ricky 1bc9227c7f finish redmine-6 merge into dev 2026-01-22 20:43:06 -05:00
ricky 3c2f1d0edd Merge branch 'redmine-6' into HEAD 2026-01-22 20:39:20 -05:00
ricky 35e303d54b Migrated hard coded strings to locales 2026-01-21 20:49:54 -05:00
ricky 2aeb3fa028 Updated copyright dates 2026-01-21 20:40:06 -05:00
ricky c85e45b544 Print attached estimate 2026-01-21 20:29:38 -05:00
ricky 6cd7825430 Added pdf method 2026-01-21 20:28:54 -05:00
ricky 14f411c2e1 Moved hooks & patches into sperate folders 2026-01-21 19:35:08 -05:00
ricky 623510b474 Finaly got the issue form javascrip reloading to work 2026-01-21 12:20:43 -05:00
ricky 20d9f0a84e update readme 2026-01-20 21:41:58 -05:00
ricky f741ce5dc9 Fixed displaying of notes 2026-01-20 21:40:17 -05:00
ricky 72ec89292f Added onclick to load customer link 2026-01-20 21:15:25 -05:00
ricky b54eb86b7f Moved javascript into ViewLayoutsHookListener 2026-01-20 21:02:40 -05:00
ricky f74f3ad72e Select All Invoices For Bulk PDF 2026-01-20 20:49:03 -05:00
ricky 0647b7708f Udated README & Copyright Date 2026-01-20 13:49:38 -05:00
ricky 7d644f0619 Dynamically load all Hooks & Patches 2026-01-20 08:12:34 -05:00
ricky b712c328ba Remove logging 2026-01-20 08:11:53 -05:00
ricky 5649ba05cd Added Checkbox Controller javascript 2026-01-19 19:34:59 -05:00
ricky bcdd515cf1 Remove view_layouts_base_body_bottom 2026-01-19 19:34:10 -05:00
ricky 704dff2a72 Multiple Invoices to PDF 2026-01-19 19:33:29 -05:00
ricky 55d00f9005 Added sandbox to settings 2025-07-08 21:01:48 -04:00
ricky eba3f529f8 Set version & requirements 2025-07-08 20:19:19 -04:00
ricky f0a3b0193c Development mode
Use the QBO sandbox & Display DEVEOMPENT in the plugin name
2025-06-18 08:30:21 -04:00
ricky 19733c3f8c Added gem rexml 2025-06-16 23:41:30 -04:00
ricky f22795ac90 Moved strings for notices to en.yml 2025-06-16 22:56:54 -04:00
ricky 166a9ee31b Removed has_many purchases, table doesn't exist anymore 2025-06-16 22:54:47 -04:00
ricky 4d85c24872 generate redirect_uri protocol based one site settings 2025-06-16 22:30:30 -04:00
ricky 43c7374c42 Load oauth key & secret when constructing client, not on application start up 2025-06-16 22:16:58 -04:00
ricky 60857e9dca generate redirect_uri protocol based one site settings 2025-06-16 22:15:42 -04:00
ricky d38f0d6ac1 Merge branch 'master' of https://github.com/rickbarrette/redmine_qbo 2025-06-15 19:03:59 -04:00
ricky f6da031e72 include Redmine::I18n 2025-06-15 18:59:00 -04:00
ricky 9779437c00 Log token refresh 2024-12-19 09:41:53 -05:00
ricky 1a37926628 Log error not info 2024-12-19 09:36:23 -05:00
ricky dac9a7c756 Started Support for Redmine 6 2024-12-12 06:00:36 -05:00
ricky 9ac1261ed0 Sort by id not doc_number
This fixes the bug where documents were displayed out of order
2024-08-30 09:19:58 -04:00
ricky 9b69d3f728 Added link to customer profile for appointments 2024-08-26 11:00:50 -04:00
ricky a5de879260 Fixed formatting 2024-08-26 08:41:56 -04:00
ricky 6464e1cbc6 Added actions 2024-08-26 07:58:40 -04:00
ricky 7f3a94229a Create Estimate 2024-08-26 07:58:05 -04:00
ricky 395e0117fb Update _actions.html.erb 2024-08-26 07:57:10 -04:00
ricky e04d363e42 Added label for actions 2024-08-26 07:56:06 -04:00
ricky 3b6c0d4a70 Removed Action links 2024-08-26 07:52:27 -04:00
ricky d1f6ccd9cb Create _actions.html.erb 2024-08-26 07:51:38 -04:00
ricky 74f7ba41df Add Appointment Link 2024-08-21 21:39:50 -04:00
ricky 4fb424faa8 Only sync by doc number if not in database 2024-08-20 07:14:37 -04:00
ricky 63218e7f42 Fixed formating 2024-08-19 23:28:54 -04:00
ricky 7f0bb3cae7 Removed extra end 2024-08-19 23:26:43 -04:00
ricky ad7417c233 Moved work into thread to repsond quickly 2024-08-19 23:21:56 -04:00
ricky cf0be2336b Removed sync button from sidebar 2024-08-19 23:12:20 -04:00
ricky 6e08746611 2.1.1 Force Estimate sync by Doc Number when searching 2024-08-19 22:51:53 -04:00
ricky 7eb26facaf Use the first result 2024-08-19 22:49:20 -04:00
ricky 9115cc662c Forgot params[:search] 2024-08-19 22:39:50 -04:00
ricky 9e7c1dbfb2 removed () 2024-08-19 22:38:16 -04:00
ricky e99f5d2e52 Added webhook view 2024-08-19 22:36:44 -04:00
ricky 039d1ca993 Use Logger.info 2024-08-19 22:31:41 -04:00
ricky dd9ac3c481 Added Estimate.sync_by_doc_number 2024-08-19 22:30:34 -04:00
ricky 4f789080e7 2.1.0 Bumped wrong versoin 2024-08-19 20:18:22 -04:00
ricky 80fc858a35 send back status 200 if request succeeded 2024-08-19 20:14:02 -04:00
ricky 6f8d280657 5.2.0 FIXED QBO Authentication 2024-08-19 20:06:13 -04:00
ricky 5782cbc166 Added https 2024-08-19 20:04:09 -04:00
ricky 0729d2ac41 added https to redirect_uri 2024-08-19 20:02:22 -04:00
ricky 6c6de0ba86 Added log 2024-08-19 19:59:26 -04:00
ricky 11dbcaf80c Use Setting.host_name & path 2024-08-19 19:53:51 -04:00
ricky 95592e542f Use qbo_oauth_callback_path 2024-08-19 19:30:51 -04:00
ricky 472bdec4fa Use qbo_authenticate_path 2024-08-19 19:17:45 -04:00
ricky c7a313e9ed Add customer name to details 2024-04-03 11:47:38 -04:00
ricky c14b590083 2024 Copy Right Update 2024-03-29 08:10:05 -04:00
ricky 040c920481 2.0.5 2024-03-29 07:58:26 -04:00
ricky 8c63817950 Use free_form_number 2024-03-28 14:13:39 -04:00
ricky e2f43d398f Nil Checks 2024-03-28 14:01:18 -04:00
ricky 7ba4829066 Update Customer Phone Numbers On Sync 2024-03-28 13:51:29 -04:00
ricky 938999db91 Added quickbooks to customer's name 2024-03-28 12:54:36 -04:00
ricky 0b60a8e41b 2.0.4 2024-01-07 20:53:07 -05:00
ricky 817a43e849 Fixed update 2024-01-07 20:47:26 -05:00
ricky 047296329e 2.0.32.0.3 2023-12-31 16:42:47 -05:00
ricky c8cb74f3d4 Merge branch 'redmine-5' 2023-12-31 16:35:26 -05:00
ricky aceb6cb6b5 fixed typo 2023-12-31 16:26:02 -05:00
ricky b531076c18 Merge branch 'dev' into hooks 2023-12-30 23:43:10 -05:00
ricky 9e342ced28 Merge branch 'master' into hooks 2023-12-30 23:40:08 -05:00
ricky 9fd1bc9dff Merge branch 'redmine-5' 2023-12-30 23:35:25 -05:00
ricky 0537d9bd86 Merge branch 'master' into hooks 2023-12-30 23:18:55 -05:00
ricky 04391f1c6e 2.0.2 2023-12-30 23:07:17 -05:00
ricky e2bf42e66b Fixed invoice pdf 2023-12-30 23:04:43 -05:00
ricky 0c72ca9294 missed this authenticated_request 2023-12-30 23:01:32 -05:00
ricky 2985fad77c Fixed typo 2023-12-30 23:01:01 -05:00
ricky 02b5fb4d0e Fixed returned variable handling 2023-12-30 22:53:08 -05:00
ricky bf417c163c Rework performing authenticated requests 2023-12-30 22:33:28 -05:00
ricky b35974e455 2.0.1 2023-12-30 20:55:58 -05:00
ricky 6d0abf865e 2023 2023-12-30 20:54:40 -05:00
ricky 275af9be82 Fixed formatting 2023-12-30 20:53:39 -05:00
ricky f4e44a1975 Remove listen (was used for development env) 2023-12-30 20:29:20 -05:00
ricky 81f322b616 Call refresh_token to set token time stamps 2023-12-30 20:26:30 -05:00
ricky f094ef57ec Setter for notes 2023-12-30 20:08:36 -05:00
ricky 2e32d8f6e5 Fixed get_base 2023-12-30 19:55:01 -05:00
ricky 3e352f270d Added Item 2023-12-30 19:41:02 -05:00
ricky 45056e8ff4 Remove unsed columns 2023-12-30 19:28:04 -05:00
ricky b13abe51bf Display token expiration times 2023-12-30 19:27:37 -05:00
ricky c3513427de Used realm_id not comany_id 2023-12-30 19:13:08 -05:00
ricky 7a6b6882d2 Update get_base 2023-12-30 18:53:01 -05:00
ricky d6ec34cef9 added realm_id 2023-12-30 12:56:46 -05:00
ricky 84dfdd707a fixed token names 2023-12-30 12:46:52 -05:00
ricky 517a239485 Started reworking Oauth token sorage 2023-12-30 12:39:51 -05:00
ricky 47868051f8 Rails 6.1 Deprecates update_attributes 2023-12-29 20:25:26 -05:00
ricky 96e4e9df66 Fixed typo with params 2023-12-29 20:17:46 -05:00
ricky 7d510e4028 Added notes to allowed params 2023-12-29 20:06:15 -05:00
ricky 6760b29148 Log the time stamp 2023-12-29 19:20:31 -05:00
ricky 122063b1d5 Fixed customer typo 2023-12-29 19:14:38 -05:00
ricky b304c3a175 Fixed employee typo 2023-12-29 19:09:24 -05:00
ricky 5b89d73c20 Remove QboItem.sync 2023-12-29 18:56:20 -05:00
ricky 8380dda25a render 403 when forbidden 2023-12-29 17:06:00 -05:00
ricky 7839116134 Version 2.0.0 2023-12-27 16:31:11 -05:00
ricky b3a809ab1c Redmine 5.1 Update 2023-12-27 15:03:08 -05:00
ricky 3a0e58c3da Append last 4 of phone number to customers name 2023-05-22 07:34:18 -04:00
ricky 26433c9020 Added hour totals to customer job history 2023-01-14 06:38:48 -05:00
ricky a531ef4f87 Added hour totals to customer job history 2023-01-14 06:20:41 -05:00
ricky 6dbf84f401 Added nil check to address_to_s 2022-05-04 12:37:31 -04:00
ricky 3220ff728f Fixed process_estimate that I accidently broke 2022-04-06 12:45:40 -04:00
ricky 1fae647381 Merge branch 'master' into dev 2022-03-19 05:49:54 -04:00
ricky d1764e2203 Removed Comment 2022-03-19 05:49:18 -04:00
ricky d8d1942673 Version 1.1.6 2022-03-19 05:47:07 -04:00
ricky 8e329b2dd2 Fix broken invoice processing 2022-03-19 05:29:59 -04:00
ricky 3622f8cad7 Added space 2022-03-19 05:16:26 -04:00
ricky f830881883 Don't need to pass customer, only issue 2022-03-17 06:52:46 -04:00
ricky fb87e8a33a Added hook :show_issue_view_right 2022-03-17 06:45:57 -04:00
ricky 8bdec410c4 Fixed render of hook :show_customer_view_right 2022-03-17 06:12:35 -04:00
ricky dec9eee90b Merge branch 'master' into dev 2022-03-15 12:29:11 -04:00
ricky 0513763607 Version 1.1.5 2022-03-14 19:41:53 -04:00
ricky b7e3ea9e3d estimate not e 2022-03-14 19:39:36 -04:00
ricky 3ea2cd14d1 Fixed accidental removal of qbo prefix 2022-03-14 19:33:20 -04:00
ricky 7b7875991f created get_estimate to remove redundant code 2022-03-14 19:28:40 -04:00
ricky b1a106d4d8 No id required 2022-03-14 19:27:31 -04:00
ricky 0281d86f1a No id required 2022-03-14 19:25:20 -04:00
ricky 2231156873 use estimate_doc_path not a hard coded path 2022-03-14 19:01:20 -04:00
ricky 2745ecf242 Added call_hook :show_customer_view_right 2022-03-14 07:52:33 -04:00
ricky 13472c3b3a Remove nhtsa_vin gem 2022-03-14 07:26:07 -04:00
ricky b686110145 Initial Vehicle Delete 2022-03-14 07:22:43 -04:00
ricky d91e7892c3 Added PDF hooks 2022-03-13 22:36:34 -04:00
ricky f26224de56 Created hook process_invoice_custom_fields 2022-03-13 20:06:13 -04:00
ricky ecc8930bec Try to bill completed issue, TODO handle errors 2022-03-13 17:51:52 -04:00
ricky 5814740a5d Allow customer to view estimate 2022-03-13 17:49:58 -04:00
ricky 25159c760a FIX - forgot to drop qbo from time_entries.billed 2022-03-13 01:17:18 -05:00
ricky 3ff9132acb Updated readme 2022-03-13 01:00:34 -05:00
ricky b5f00f254c Added a copy link button 2022-03-13 00:53:32 -05:00
ricky 70f2c473d5 Moved buttons to watcher link location 2022-03-13 00:37:51 -05:00
ricky b3b11d726d Version 1.1.4 2022-03-12 16:03:14 -05:00
ricky f97d5bc731 Moving fat into CustomerToken 2022-03-12 16:01:13 -05:00
ricky 49507d06c7 Updated TODO list 2022-03-12 00:03:26 -05:00
ricky 5d928c486f Getter convenience method for tokens 2022-03-12 00:01:40 -05:00
ricky 0485e9d64c Allow attachment viewing w/ valid customer token 2022-03-11 23:16:23 -05:00
ricky cc0839204e Ignore workspace files 2022-03-11 21:14:12 -05:00
ricky 760a85a1da removed link_to user & version 2022-03-10 06:53:05 -05:00
ricky c821774e9b Version 1.1.3 2022-03-09 23:17:38 -05:00
ricky 47a19a7e77 Removed Qbo index & sync html, methods, & routes
This stuff isn't really used for anything
2022-03-09 23:06:54 -05:00
ricky a75f1abd71 Removed extra white spaces 2022-03-09 22:53:03 -05:00
ricky 09c497ff96 Removed old #attr_accessible... comments 2022-03-09 22:31:19 -05:00
ricky cae1d9de02 Removed old commented out code 2022-03-09 22:27:33 -05:00
ricky 1050a4f6a7 Use rails router, not hard coded links 2022-03-09 22:18:47 -05:00
ricky be4ef44c13 Added strings for Flash messages, not used yet 2022-03-09 22:10:16 -05:00
ricky 89e4132fc1 Update readme & comments 2022-03-09 21:51:06 -05:00
ricky 8d8201822b No longer need to prepend("Qbo") 2022-03-09 21:45:07 -05:00
ricky 726eb4632e use plural for routes 2022-03-09 21:43:08 -05:00
ricky ffcb2ee608 Remove payment orphans 2022-03-09 21:40:40 -05:00
ricky 24f8be6e80 Drop the Qbo prefix 2022-03-09 21:31:56 -05:00
ricky 08fa4aefc4 Added button text to locale file 2022-03-09 19:09:37 -05:00
ricky 13bbd5dfc1 Use rails router, not hard coded links 2022-03-09 19:05:12 -05:00
ricky 8e6eeab680 Moved HTML code into partial _from_hook.html.erb 2022-03-09 18:55:44 -05:00
ricky 70d9d5063a Moved issues show details to issue folder 2022-03-09 18:25:07 -05:00
ricky 374429f161 Added 2016 2022-03-08 20:50:39 -05:00
ricky c69666e747 Added first year 2022-03-08 20:49:55 -05:00
ricky 7dc04b4a07 QboItem doesn’t exist anymore 2022-03-08 20:30:52 -05:00
ricky 7b5e54aaba Merge branch 'master' of https://github.com/rickbarrette/redmine_qbo 2022-03-07 07:52:26 -05:00
ricky 30b704c90f Don't generate new customer view token every time 2022-03-07 07:52:17 -05:00
ricky 2f98b5afaa Update README.md
formatting
2022-03-06 19:04:29 -05:00
ricky 3c3b43cfc5 Readme update
formatting
2022-03-06 19:02:31 -05:00
ricky 09f2a534be Readme update
Cleaning things up
2022-03-06 19:01:42 -05:00
ricky 7b5b673ebf Version 1.1.2 2022-03-06 18:20:30 -05:00
ricky c72d0a83ca New line @ EOF & formating 2022-03-06 17:50:22 -05:00
ricky 3159289ac0 Removed unused code, only need to bill time
removed line items, payments, drop used db tables
2022-03-06 17:26:57 -05:00
ricky a9cc5fac73 Removed unsed code & cleaned up comments 2022-03-06 17:05:04 -05:00
ricky fe06fccacd Only show sync button if User is an admin 2022-03-06 16:49:07 -05:00
ricky 8b4a46f7eb H3 not H2 2022-03-06 16:27:39 -05:00
ricky cf362caaf2 Cleaning up html formatting 2022-03-06 16:24:17 -05:00
ricky de1be7d296 Make last 8 bold 2022-03-06 16:17:08 -05:00
ricky d8e3e1a72f Styling & formatting 2022-03-06 13:46:55 -05:00
ricky 64a7ad844f Styling & formatting 2022-03-06 13:46:26 -05:00
ricky 9201c4ca96 render 404 on all exceptions 2022-03-06 09:23:05 -05:00
ricky dab6b6f723 don't show doors if vehicle.doors is nil 2022-03-06 09:17:06 -05:00
ricky 495243d177 Add trim & doors to vehicle 2022-03-06 08:59:48 -05:00
ricky 332f07c93d Version 1.1.1 2022-03-06 07:51:39 -05:00
ricky 54d4be9762 Only show sidebar views when user is logged in 2022-03-06 07:32:01 -05:00
ricky f1e3c29c97 Added Load Customer Link on Issue Form 2022-03-05 08:26:57 -05:00
ricky 66d393a465 Dynamically load hooks/patches & require redmine4+ 2022-03-02 07:15:54 -05:00
ricky 218d3392f0 Moved string to en.yml 2022-02-24 18:34:17 -05:00
ricky 0136d91cc3 Comments & formatting 2022-02-24 04:58:41 -05:00
ricky a95f0350d8 Fixed missing transation error 2022-02-23 20:20:47 -05:00
ricky 55c04b6585 update Issue form on customer name change 2022-02-23 18:36:08 -05:00
ricky ea21bc362a autocomplete off for forms & search 2022-02-23 18:24:57 -05:00
ricky 117d92b879 Fixed customer sorting & removed customer filter 2022-02-21 20:23:57 -05:00
ricky 440c8e4618 list issues in desc order 2022-02-21 19:33:38 -05:00
ricky 1344526f7f update txn_date too 2022-02-21 08:51:24 -05:00
ricky 19acfbc76f Added logging 2022-02-21 08:51:01 -05:00
ricky 9dfb27f0a4 Prevent webhook loops 2022-02-21 07:54:24 -05:00
ricky 51cd830710 Updated screenshots 2022-02-21 06:53:50 -05:00
ricky 956ba2ad46 Version 1.1.0 2022-02-21 06:25:51 -05:00
ricky 3ae3107760 desc sort issues, estimates, & invoices 2022-02-21 06:22:13 -05:00
ricky 925d4b8bcf Updated comments & removed unused code 2022-02-21 06:06:24 -05:00
ricky ca6dbfd12d Removed duplicate private declaration 2022-02-21 05:26:54 -05:00
ricky 9ea03d0c6d Removed sync on view 2022-02-21 05:23:11 -05:00
ricky 6ad4929d53 Sync Estimates & Invoices on database update 2022-02-21 05:17:35 -05:00
ricky 446f419af0 Sync invoice when viewing 2022-02-20 19:37:03 -05:00
ricky f3c5de82e0 Bug fix 2022-02-20 18:02:37 -05:00
ricky 56e24752cf Import invoice fix 2022-02-20 17:53:01 -05:00
ricky 255af13b20 Add txn_date to invoice & estimate databse 2022-02-20 17:40:41 -05:00
ricky 02b4f1eb43 Added Invoice date 2022-02-20 17:26:24 -05:00
ricky 8c735d3921 Added Estimate date 2022-02-20 17:19:56 -05:00
ricky 70e6038215 Moved customer issue counts 2022-02-20 15:22:31 -05:00
ricky fc7501c4fe address not a 2022-02-20 14:57:21 -05:00
ricky 45b60cfea1 PhysicalAddress to_s 2022-02-20 14:52:11 -05:00
ricky 09313ad471 exclude before filter for customer/view 2022-02-20 13:42:06 -05:00
ricky 1b15aecbff Disable autocomplete suggestions for search 2022-02-20 08:11:57 -05:00
ricky 2bea7dbc8d fixed customer link - missing view issues/history 2022-02-20 07:56:25 -05:00
ricky 3468b5f236 Open links in new window 2022-02-19 22:53:39 -05:00
ricky 1c431d14dc remove gem faraday_middleware & set oauth2 1.4.7 2022-02-19 22:48:44 -05:00
ricky 7234a70265 Added allowed params for qbo controller 2022-02-19 21:47:12 -05:00
ricky a459d84b00 Added Estimate & Invoice List to Customer view 2022-02-19 21:19:08 -05:00
ricky 49d2ed8244 Readme update 2022-02-19 20:56:54 -05:00
ricky d6aebfcb99 Moved strings to Ruby I18n en.yml 2022-02-19 20:46:39 -05:00
ricky 2085eb7869 Add gem dependancy faraday_middleware 1.2.0 to fix NoMethodError: undefined method `dependency' for Gzip:Class 2022-02-18 22:42:20 -05:00
ricky c101a86f02 Open Estimate PDF in new windows 2022-02-17 09:19:32 -05:00
ricky 2d32769a59 Error not fatal 2022-02-13 13:19:04 -05:00
ricky a2f755388e Fixed merge error 2022-02-13 12:34:03 -05:00
ricky 8a8f1af2bd Updated Copyright 2022 2022-02-13 11:57:43 -05:00
ricky 4582b8c5b9 Added estimate search in side bar 2022-02-13 10:03:43 -05:00
ricky f66fbf6656 specified the rails version in migrations 2022-02-12 16:14:36 -05:00
ricky 41d49ccce5 Setting.host_name for hooks not a hardcoded URL 2022-02-12 09:24:21 -05:00
ricky c85f450742 Merge branch 'master' into redmine-4 2022-02-12 09:19:36 -05:00
ricky e314dae10d Logging & exception handler for updating invoices 2022-02-12 09:12:11 -05:00
ricky b1192a1912 Use host name from settings for auth & added exception handing in QboInvoice 2022-02-06 16:37:00 -05:00
ricky 7cc8a946fd Inital update for redmine 4.0+ compatibility 2022-02-01 20:53:20 -05:00
ricky 4b34852c72 Don't include open issues in closed issue list 2020-03-30 20:56:38 -04:00
ricky 5d7fc9dabd Don't include open issues in closed issue list 2020-03-28 20:39:04 -04:00
ricky db61952e67 added notes to regaurding customer autocomplete field 2020-03-28 20:19:58 -04:00
ricky 016dca242c Added comments to clearify methods 2020-03-28 20:16:54 -04:00
ricky 983811af97 Added comments to clearify methods 2020-03-28 20:08:44 -04:00
ricky d18a9726ac Added onchange event to customer_id 2020-03-28 19:56:42 -04:00
ricky cdef838d3e comment out missing method call 2020-03-26 20:31:00 -04:00
ricky 7703d724e1 Update issues_save_hook_listener.rb
titleize all subjects
2020-03-26 20:22:07 -04:00
ricky 94b5efbd00 forgot a comma 2020-03-26 12:53:59 -04:00
ricky f43020b864 Update issues_form_hook_listener.rb
Added onchange listener to customer form field
2020-03-26 12:50:32 -04:00
ricky 0d0f808305 Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2020-01-24 13:55:15 -05:00
ricky 279e8b15e0 Added open/closed issue lists for customer/vehicles 2020-01-24 13:54:26 -05:00
ricky 099f729303 Update README.md 2020-01-06 08:52:36 -05:00
ricky 5150a31cdb Removed unused gems 2020-01-05 12:33:52 -05:00
ricky b5d17dc862 Removed oauth1 database columns & updated settings 2020-01-05 12:21:36 -05:00
ricky e6c5feb3f3 Remove uneeded columns from qbos table 2020-01-05 12:03:19 -05:00
ricky 5573e941c6 Cleaning up uneeded code 2020-01-05 11:58:27 -05:00
ricky 29dbca20e0 Update copyright 2020-01-05 11:44:02 -05:00
ricky d6c114d52b Version 0.8.1 2020-01-05 09:20:23 -05:00
ricky 87b8daf283 Fixed expired token check 2020-01-05 09:17:50 -05:00
ricky 719abe20a6 Update qbo.rb 2020-01-05 01:22:27 -05:00
ricky 4a5b83265d Update qbo.rb 2020-01-05 01:20:23 -05:00
ricky 8d103d3fc6 Update qbo.rb 2020-01-05 01:12:48 -05:00
ricky 9310f207a3 Update qbo.rb 2020-01-05 01:08:42 -05:00
ricky 000b67b329 Added employee 2020-01-05 00:08:00 -05:00
ricky ebee9395ba Update qbo_employee.rb 2020-01-05 00:06:45 -05:00
ricky 2cd6731f0c Update qbo_employee.rb 2020-01-05 00:05:25 -05:00
ricky ebdbd25082 Update qbo.rb 2020-01-04 23:45:00 -05:00
ricky 18ada91fcd Update qbo.rb 2020-01-04 23:38:46 -05:00
ricky 1cf3926585 Update qbo.rb 2020-01-04 23:33:33 -05:00
ricky e776deeece Update qbo.rb 2020-01-04 23:07:53 -05:00
ricky 8c2f30949a Update qbo.rb 2020-01-04 23:04:09 -05:00
ricky 015a989f72 Update qbo.rb 2020-01-04 22:57:53 -05:00
ricky 0d4d5a6136 Update qbo.rb 2020-01-04 22:52:01 -05:00
ricky 0364989fe1 Update qbo_controller.rb 2020-01-04 22:46:20 -05:00
ricky fb47eaba0e Update qbo.rb 2020-01-04 22:45:31 -05:00
ricky 725d511be5 Update qbo.rb 2020-01-04 22:30:43 -05:00
ricky fd85f296de Update 030_update_qbos_token.rb 2020-01-04 22:16:07 -05:00
ricky 9549bb8fe2 Update qbo.rb 2020-01-04 20:24:29 -05:00
ricky 6a1c8b0551 Update README.md 2020-01-04 16:09:03 -05:00
ricky 086632e804 Version 0.8.0 (OAuth 2.0) 2020-01-04 16:01:16 -05:00
ricky d37ff922fc Update customer.rb 2020-01-04 15:51:04 -05:00
ricky 3483efa100 Oauth2 Upgrade 2020-01-04 15:41:15 -05:00
ricky f65eea2820 Oauth2 Upgrade 2020-01-04 15:37:17 -05:00
ricky a4111e0a11 oAuth2 Upgrade 2020-01-04 15:36:36 -05:00
ricky ebe5373d82 Oauth2 Upgrade 2020-01-04 15:34:57 -05:00
ricky 5b8c7d42c5 Oauth2 Upgrade 2020-01-04 15:34:04 -05:00
ricky b8fc57d583 Oauth2 Upgrade 2020-01-04 15:31:19 -05:00
ricky 7c42197cb1 Oauth2 Upgrade 2020-01-04 15:30:03 -05:00
ricky cc0ffce892 Oauth2 Upgrade 2020-01-04 15:29:18 -05:00
ricky 0fd2abbec3 Oauth2 Upgrade 2020-01-04 15:28:43 -05:00
ricky 215b219a6d Oauth2 Upgrade 2020-01-04 15:28:04 -05:00
ricky ea71542d81 Oauth2 Upgrade 2020-01-04 15:26:35 -05:00
ricky 5dbf486b50 Update qbo_controller.rb 2020-01-04 15:21:14 -05:00
ricky b734125d6b Update qbo_controller.rb 2020-01-04 15:19:31 -05:00
ricky 06e6295c6e Update qbo.rb 2020-01-04 14:32:32 -05:00
ricky fd383ad9d4 Update qbo.rb 2020-01-04 14:29:23 -05:00
ricky 4eb6c533f1 Update qbo.rb 2020-01-04 14:21:07 -05:00
ricky 5af7d73768 Update qbo_controller.rb 2020-01-04 14:18:42 -05:00
ricky 1d0ae34261 Update qbo.rb 2020-01-04 14:13:29 -05:00
ricky 21656b3e14 Update qbo.rb 2020-01-04 14:09:37 -05:00
ricky 131976cd71 Update issue_patch.rb 2020-01-04 14:08:48 -05:00
ricky 88c1b9c9a2 Update issue_patch.rb 2020-01-04 14:06:42 -05:00
ricky 5ea9aed3cb Update qbo.rb 2020-01-04 14:05:00 -05:00
ricky 41e10d9b0e Update qbo.rb 2020-01-04 13:59:09 -05:00
ricky 45859bef3e Update qbo.rb 2020-01-04 13:56:38 -05:00
ricky f5c40738dc Update issue_patch.rb 2020-01-04 13:54:29 -05:00
ricky bfa37ee634 Update issue_patch.rb 2020-01-04 13:51:31 -05:00
ricky 787b55f3d7 Update qbo.rb 2020-01-04 13:46:13 -05:00
ricky 61f882e98c Update qbo.rb 2020-01-04 13:45:00 -05:00
ricky 37db0d3d72 Update qbo.rb 2020-01-04 13:31:40 -05:00
ricky 4f2dec3069 Update qbo.rb 2020-01-04 13:29:36 -05:00
ricky 35a7c3cfeb Create 030_update_qbos_token.rb 2020-01-04 13:28:22 -05:00
ricky cbbaf5a95c Update qbo.rb 2020-01-04 13:22:27 -05:00
ricky 647923e5e6 Update 029_update_qbos_types.rb 2020-01-04 13:19:21 -05:00
ricky 70ca4e9964 Create 029_update_qbos_types.rb 2020-01-04 13:17:15 -05:00
ricky 7fb40ad4a8 Update qbo_controller.rb 2020-01-04 13:09:09 -05:00
ricky 36083d23a0 Update qbo_controller.rb 2020-01-04 13:07:36 -05:00
ricky 2ec57f2bbf Update qbo_controller.rb 2020-01-04 13:06:23 -05:00
ricky 278708e566 Update qbo_controller.rb 2020-01-04 12:47:30 -05:00
ricky 23f2b92e8d Update qbo_controller.rb 2020-01-04 12:45:16 -05:00
ricky 5d92eeddfb Update qbo_controller.rb 2020-01-04 12:44:05 -05:00
ricky 384a8c033c Update qbo_controller.rb 2020-01-04 12:39:23 -05:00
ricky 32b12b60f9 Update qbo_controller.rb 2020-01-04 12:33:54 -05:00
ricky 93db447239 Update qbo_controller.rb 2020-01-04 12:29:54 -05:00
ricky 19a6180e15 Update qbo.rb 2020-01-04 12:26:04 -05:00
ricky 3408ee173c Update qbo.rb 2020-01-04 12:25:12 -05:00
ricky b817e842dd Update qbo.rb 2020-01-04 12:23:59 -05:00
ricky 51c3b8338e Update qbo.rb 2020-01-04 12:23:05 -05:00
ricky c6a3edfbc1 Update qbo_controller.rb 2020-01-04 12:21:45 -05:00
ricky 21d8d90465 Update qbo.rb 2020-01-04 12:21:24 -05:00
ricky 04c0fa57c6 Update qbo_controller.rb 2020-01-04 12:17:26 -05:00
ricky f5ad761712 Update qbo_controller.rb 2020-01-04 12:14:54 -05:00
ricky 9b80485915 Update qbo.rb 2020-01-04 12:14:05 -05:00
ricky 87de865c00 oauth2 2020-01-04 11:51:25 -05:00
ricky 1ea27e8511 Only show error flash if errors are not empty 2019-11-12 12:50:44 -05:00
ricky 8f0ca00b09 Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2019-11-12 12:45:20 -05:00
ricky 859a1d505b always show errors 2019-11-12 12:45:14 -05:00
ricky cd109653a2 Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2019-11-12 12:37:12 -05:00
ricky cab723bbcd Removed styles & removed after find call 2019-11-12 12:36:25 -05:00
ricky 3dd712629b added local vin validation 2019-11-12 12:35:54 -05:00
ricky cdf2603e12 Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2019-11-12 12:16:30 -05:00
ricky 5df9d324bc Update vehicle.rb 2019-11-12 12:15:59 -05:00
ricky f78c0338b4 Made decode_vin not private 2019-11-12 12:08:04 -05:00
ricky fe6aa7908f Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2019-11-12 11:51:54 -05:00
ricky aa45338e36 Update vehicle.rb 2019-11-11 10:10:23 -05:00
ricky 213dca2621 Fixed redicect to :home 2019-07-08 11:30:53 -04:00
ricky fee710d717 Rediect to vehicle if vin is already taken 2019-06-26 11:39:07 -04:00
ricky 65eac58f6c Update customer.rb 2019-06-25 12:48:35 -04:00
ricky b4f5112fc3 Create 028_add_customers_mobile_phone_number.rb 2019-06-25 12:46:03 -04:00
ricky fa5dcbf9a9 Fixed redirect to for new vehicles 2019-06-25 12:37:49 -04:00
ricky e0aebb1c23 Update customer.rb 2019-06-25 11:59:08 -04:00
ricky 6d176acc2b Update customer.rb 2019-06-25 11:54:26 -04:00
ricky 9e9b29fef9 Update 027_add_customers_phone_number.rb 2019-06-25 11:53:40 -04:00
ricky 1af846537d Update customer.rb 2019-06-25 11:50:14 -04:00
ricky d6c5daff49 Added phone number 2019-06-25 11:15:57 -04:00
ricky 61c76ad80a Create 027_add_customers_phone_number.rb 2019-06-25 10:59:34 -04:00
ricky 0d514790fd Moved issue.status.is_closed? check back into issue save hook 2019-03-26 12:15:03 -04:00
ricky 748d431d35 Removed controller_issues_edit_before_save hook 2019-03-26 12:11:54 -04:00
ricky 87b8d99c41 Working on issue.bill_time 2019-03-26 12:09:38 -04:00
ricky a0da53b6cf Fixed formatting & removed search from heading 2018-10-15 21:10:41 -04:00
ricky 02d630c631 Fixed Invoice Link to be HTML safe 2018-10-15 21:07:36 -04:00
ricky 15b214c800 Moved html into partial view 2018-10-15 20:57:03 -04:00
ricky 1b5e185087 Added nil check 2018-10-15 19:42:41 -04:00
ricky 102309600e Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2018-10-15 19:36:09 -04:00
ricky 6acc7db91b Changed Customer search to shown on all pages 2018-10-15 19:34:51 -04:00
ricky 02898883a8 Update line_item.rb 2018-10-14 23:35:13 -04:00
ricky ce02b70bc3 Update line_items_controler.rb 2018-10-14 23:34:17 -04:00
ricky d4d4a555f8 Forgot to add sidebar files 2018-10-14 23:32:07 -04:00
ricky c2663cd0a0 License update 2018-10-14 23:31:17 -04:00
ricky d48609361f Added customer search to sidebar 2018-10-14 23:27:19 -04:00
ricky 70995f6e55 Fixed formating 2018-10-04 20:59:46 -04:00
ricky 05a0472939 Changed redirect to :back 2018-10-04 20:57:24 -04:00
ricky cff9f3fde3 Fixed new vehicle add missing customer 2018-09-22 00:29:54 -04:00
ricky e24b704571 Added qbo partial views 2018-09-19 22:57:45 -04:00
ricky 4d99f54c79 Added autocomplete to vehicle customer field 2018-09-19 22:33:36 -04:00
ricky e65725c334 Update init.rb 2018-09-19 10:15:13 -04:00
ricky 4829daab7c Update new.html.erb 2018-09-19 10:10:54 -04:00
ricky 260e9f3e4a Update _list.html.erb 2018-09-19 10:07:52 -04:00
ricky e3ce2445b8 Update index.html.erb 2018-09-19 10:05:45 -04:00
ricky 2b333667ed Update init.rb 2018-09-19 09:52:58 -04:00
ricky 1077cf214c Update show.html.erb 2018-09-19 09:51:27 -04:00
ricky f27fdf5274 Update index.html.erb 2018-09-19 09:50:58 -04:00
ricky 1dbcca4ca0 Create _search.html.erb 2018-09-19 09:49:51 -04:00
ricky 558e2359f7 Added partial customers/search & fixed formatting 2018-09-19 09:45:36 -04:00
ricky f99ef648b3 Rename _search.erb to _search.html.erb 2018-09-19 09:24:05 -04:00
ricky 5e4e3329c8 Create _search.erb 2018-09-19 09:23:37 -04:00
ricky b0a66aba0a Fixed custom field redering 2018-07-31 15:22:46 -04:00
ricky f2dd500536 Merge branch 'line_items' of github.com:rickbarrette/redmine_qbo into line_items 2018-03-31 08:04:17 -04:00
ricky 7412ac4f91 Added boolean billed 2018-03-31 08:03:26 -04:00
ricky 2acb3efe5a Create line_items_controler.rb 2018-03-31 08:00:56 -04:00
ricky 2cc0d06bc5 Merge branch 'master' into line_items 2018-03-29 22:42:54 -04:00
ricky 4070cb7c49 Merge branch 'master' of github.com:rickbarrette/redmine_qbo 2018-03-29 22:42:03 -04:00
ricky fcd196355a Create line_item.rb 2018-03-29 10:17:01 -04:00
ricky ea502d5b7b Create 026_create_line_items.rb 2018-03-29 10:05:37 -04:00
ricky 1f33009f89 Update LICENSE 2018-03-28 15:24:44 -04:00
ricky 3509ae9725 Removed squish_vin and added last 8 of vin to vehicle name 2018-03-27 09:40:01 -04:00
ricky 49858c45c9 do a full search by setting the full parameter to true. 2018-03-27 09:38:44 -04:00
ricky b78cd44cc9 don't bill time if not assigned to anyone EE 2018-03-05 08:58:33 -05:00
ricky 39fcd6d4dd Removed Drive & Doors 2018-03-03 13:55:22 -05:00
ricky 8838d36793 Fixed vin decoding 2018-03-03 13:54:19 -05:00
ricky 63fa94e6f2 Merge branch 'master' into nhtsa_vin 2018-03-03 13:14:05 -05:00
ricky 17183f9643 Update vehicle.rb 2018-03-03 13:11:58 -05:00
ricky 667d0bfa97 Remove Edmunds API & Added NhtsaVin 2018-03-03 12:38:56 -05:00
ricky 88a6be0d27 Remove Edmunds API Key Setting 2018-03-03 12:26:25 -05:00
ricky c3eaddff97 Start work to switch from edmunds_vin to nhtsa_vin 2018-03-03 12:22:44 -05:00
ricky f03adad463 Update issues_form_hook_listener.rb 2017-11-19 22:17:25 -05:00
ricky bd03e3ac32 Did things 2017-11-19 22:14:40 -05:00
ricky 299a28a0d2 Create controller_issues_listener.rb 2017-11-19 22:01:20 -05:00
ricky cee8ddced1 Added safe_attributes for ProjecT 2017-11-17 20:54:14 -05:00
ricky 738cd21b1f Update issues_form_hook_listener.rb 2017-11-17 10:56:13 -05:00
ricky b8186e4b52 More customer/project relations 2017-11-17 10:51:25 -05:00
ricky d98a8b8cc4 Fixed Project releationships and database migration 2017-11-17 10:14:05 -05:00
ricky dba6c4b131 Update vehicle.rb 2017-11-13 22:19:20 -05:00
ricky 118812f16f Update vehicle.rb 2017-11-13 22:17:03 -05:00
ricky 0b96a1412c Rename app/views/qbo/list_simple.html.erb to app/views/issues/_list_simple.html.erb 2017-11-13 22:07:12 -05:00
ricky 29de191d26 Rename app/views/issues/list_simple.html.erb to app/views/qbo/list_simple.html.erb 2017-11-13 22:06:45 -05:00
ricky f86af9ca71 Create list_simple.html.erb 2017-11-13 22:04:04 -05:00
ricky d25de7b30f Merge remote-tracking branch 'origin/dev' 2017-11-13 21:37:28 -05:00
ricky 273bd3d6be Update application.js 2017-11-13 21:01:17 -05:00
ricky ac446723f1 Update application.js 2017-11-13 20:59:45 -05:00
ricky c21bc1333f Update projects_form_hook_listener.rb 2017-11-13 20:38:33 -05:00
ricky 4e4255995e Update projects_form_hook_listener.rb 2017-11-13 20:37:59 -05:00
ricky c68b540597 Update projects_form_hook_listener.rb 2017-11-13 20:36:44 -05:00
ricky 1358871ccc Update init.rb 2017-11-13 20:31:16 -05:00
ricky 908511f299 Update init.rb 2017-11-13 20:30:28 -05:00
ricky 6260de21f9 Update query_patch.rb 2017-11-13 20:28:19 -05:00
ricky e2f276097c Update query_patch.rb 2017-11-13 20:27:36 -05:00
ricky 205bb67a6a Update query_patch.rb 2017-11-13 20:20:46 -05:00
ricky 05edafec4c Update query_patch.rb 2017-11-13 20:13:26 -05:00
ricky 4a073d3a71 Update query_patch.rb 2017-11-13 20:12:23 -05:00
ricky f2cbf31e17 Create projects_form_hook_listener.rb 2017-11-13 19:53:37 -05:00
ricky 22b22780ea Create project_patch.rb
Add relationships to projects
2017-11-13 19:47:39 -05:00
ricky 71cfa28817 Create 025_update_projects.rb
Added customer & vehicle reference to a project.
2017-11-13 19:40:14 -05:00
ricky 8b2d88f80b Create qbo_controller.rb 2017-06-14 09:43:52 -04:00
ricky eaf0a57e51 Create qbo_controller.rb 2017-06-13 21:58:42 -04:00
ricky 512f5ad7ba Create index.html.erb 2017-06-13 21:55:13 -04:00
ricky 8d2351d3f9 Create vehicle.rb 2017-06-13 21:51:36 -04:00
ricky c5a20c9e7f Create vehicle.rb 2017-06-13 21:50:50 -04:00
ricky 4a3b663333 Create vehicle.rb 2017-06-13 21:49:10 -04:00
ricky e43635b5d8 Update vehicle.rb 2017-06-13 12:31:07 -04:00
ricky 7044377f16 Update vehicle.rb 2017-06-13 12:27:22 -04:00
ricky 7ced1bf942 Create vehicle.rb 2017-06-13 12:22:48 -04:00
ricky 7ca3315ce5 Update vehicle.rb 2017-06-13 11:44:46 -04:00
ricky 2b8c4b4d4d Update vehicle.rb 2017-06-06 08:58:11 -04:00
ricky a359e8815b Update issues_form_hook_listener.rb 2017-06-06 08:53:36 -04:00
ricky 01cf82813c Update vehicle.rb 2017-06-06 08:32:21 -04:00
ricky 625e400c48 Update vehicle.rb 2017-06-06 08:27:55 -04:00
ricky 56793cee7c Update vehicle.rb 2017-06-06 08:25:35 -04:00
ricky 3ba5337812 Update vehicle.rb
Updated regex to remove invalid chars
2017-06-06 08:22:50 -04:00
ricky 129e3d4821 Update issues_form_hook_listener.rb 2017-06-06 08:12:08 -04:00
ricky 4d524a7d61 Update issues_form_hook_listener.rb 2017-06-06 08:10:28 -04:00
ricky 429fb920fb Update index.html.erb 2017-04-04 22:29:06 -04:00
ricky 77c7f0b6fe Update index.html.erb 2017-04-04 22:28:20 -04:00
ricky 1a043bea76 Added Permission Check 2017-04-04 22:26:58 -04:00
ricky e4d770c272 Update index.html.erb 2017-04-04 22:23:33 -04:00
ricky fce3931858 Added new customer button 2017-04-04 22:22:29 -04:00
ricky 43cdade6e1 Merge branch 'master' into dev 2017-04-04 09:32:50 -04:00
ricky 4374f9436c Removed .service from get_base call 2017-04-04 09:31:02 -04:00
ricky 7c63c3c816 Fixed typo 2017-04-04 08:49:08 -04:00
ricky b3f491a60b Fixed logic 2017-04-04 08:48:36 -04:00
ricky 4adcbba840 Update vehicle.rb 2017-04-04 08:47:15 -04:00
ricky baccb42455 Merge branch 'dev' of github.com:rickbarrette/redmine_qbo into dev 2017-04-04 08:46:44 -04:00
ricky d0842dd803 Merge branch 'dev' of github.com:rickbarrette/redmine_qbo into dev 2017-04-04 08:45:45 -04:00
ricky 02aabe6045 Forgot End 2017-04-04 08:45:34 -04:00
ricky 0e47f9eb5f Some Cleanup & Fixed to_s to report vin
to_s to report vin when year,make,model are nil
2017-04-04 08:44:14 -04:00
ricky f1d2d63f20 Removed un-needed initializer 2017-04-04 08:37:49 -04:00
ricky f322f9f7ab Update Copyright 2017-04-04 08:34:03 -04:00
ricky 6db8b76902 Update Copyright 2017-04-04 08:33:37 -04:00
ricky 61adce1299 0.5.0 2017-04-03 22:59:09 -04:00
ricky daffb3719e Copyright Update & Formating 2017-04-03 22:57:25 -04:00
ricky 1b8626d28f Update init.rb 2017-04-03 22:53:34 -04:00
ricky b119344fad Copyright Update 2017-04-03 22:52:37 -04:00
ricky 4381d403d4 Copyright Update 2017-04-03 22:52:14 -04:00
ricky 26bfaca1d6 Copyright Update 2017-04-03 22:52:00 -04:00
ricky 0c68d8094a Copyright Update 2017-04-03 22:51:43 -04:00
ricky 6230175ba5 Copyright Update 2017-04-03 22:51:25 -04:00
ricky 5dc4dc5637 Copyright Update 2017-04-03 22:51:03 -04:00
ricky ac15307fb8 Copyright Update 2017-04-03 22:50:50 -04:00
ricky ec5ce497d8 Copyright Update 2017-04-03 22:50:38 -04:00
ricky 01fe52157d Update issue_patch.rb 2017-04-03 22:50:19 -04:00
ricky 75737cf2fd Copyright Update 2017-04-03 22:50:05 -04:00
ricky 7824edf5aa Copyright Update 2017-04-03 22:49:52 -04:00
ricky 6b70b447a5 Copyright Update 2017-04-03 22:48:07 -04:00
ricky 5a6b679099 Copyright Update 2017-04-03 22:47:49 -04:00
ricky 72835dcf65 Copyright Update 2017-04-03 22:47:36 -04:00
ricky b9e2349983 Update qbo.rb 2017-04-03 22:47:21 -04:00
ricky ef13ec7e11 Update qbo_employee.rb 2017-04-03 22:47:10 -04:00
ricky 00b40da8c4 Copyright Update 2017-04-03 22:46:48 -04:00
ricky 2be25adf18 Copyright Update 2017-04-03 22:46:35 -04:00
ricky 5ab9a777f6 Copyright Update 2017-04-03 22:46:19 -04:00
ricky 7fbb1d6ba3 Copyright Update 2017-04-03 22:46:04 -04:00
ricky 786c80609c Copyright Update 2017-04-03 22:45:21 -04:00
ricky efb554824d Update vehicles_controller.rb 2017-04-03 22:44:40 -04:00
ricky c615abc896 Update qbo_controller.rb 2017-04-03 22:44:29 -04:00
ricky 8ecc3414da Update payments_controller.rb 2017-04-03 22:44:17 -04:00
ricky 505def8d23 Update invoice_controller.rb 2017-04-03 22:44:03 -04:00
ricky da155de514 Copyright Update 2017-04-03 22:43:42 -04:00
ricky 7d727e1ad8 Copyright Update 2017-04-03 22:43:25 -04:00
ricky 3dcb5155fc Add Blank Option 2017-04-03 22:40:07 -04:00
ricky 4424593e63 Add Blank to Select 2017-04-03 22:38:45 -04:00
ricky 8eae838ef8 Update filter_estimates_by_customer.js.erb 2017-04-03 22:34:19 -04:00
ricky d5e8b4bbc4 Update qbo_estimate.rb 2017-04-03 22:26:53 -04:00
ricky fc8efa53e9 Merge branch 'dev' of github.com:rickbarrette/redmine_qbo into dev 2017-04-03 22:25:29 -04:00
ricky 15ea3aeaa2 Update qbo_estimate.rb 2017-04-03 22:24:04 -04:00
ricky 35bf300f2d Show only estimates attached to the customer 2017-04-03 22:07:06 -04:00
ricky 72bf10680f Added ajax to update estimates 2017-04-03 21:57:56 -04:00
ricky bd8706deee Create filter_estimates_by_customer.js.erb 2017-04-03 21:55:47 -04:00
ricky e8619529d4 Added routes for filtering estimates & invoices 2017-04-03 21:52:39 -04:00
ricky fd3c8e15e6 Added filter methods for estimates & invoices 2017-04-03 21:49:57 -04:00
ricky 166c1d3002 Add files via upload 2017-04-02 17:13:21 -04:00
ricky 773d60fb23 Delete plugin_issue_view.png 2017-04-02 17:12:53 -04:00
ricky cc46902095 Update qbo_estimate.rb 2017-04-02 08:45:02 -04:00
ricky acb2628c7a Update qbo_estimate.rb 2017-04-02 08:41:36 -04:00
ricky e4914590f8 Moved Invoice Sync up in the order 2017-04-02 08:31:22 -04:00
ricky e3a8e464ae Update qbo_estimate.rb 2017-04-02 08:21:36 -04:00
ricky 8a6bb45b6a Fixed Custom Field Logic 2017-04-02 08:06:07 -04:00
ricky 3decf83a7b Update qbo_invoice.rb 2017-04-02 07:54:33 -04:00
ricky 1b7b286d1b Update qbo_invoice.rb 2017-04-02 07:31:28 -04:00
ricky a8804f6704 Increment the sync token 2017-04-02 07:28:53 -04:00
ricky 5d03e261d1 Added customer association 2017-04-02 07:19:12 -04:00
ricky 1ae766b8bd Added customer association 2017-04-02 07:13:28 -04:00
ricky 119c36569f Added association for invoices & estimates 2017-04-02 07:08:24 -04:00
ricky 3be69d5efd Update 024_update_invoices_and_estimates.rb 2017-04-02 07:00:43 -04:00
ricky b55dd99efd Merge branch 'master' into dev 2017-04-02 06:59:01 -04:00
ricky eff1f97ab2 Create 024_update_invoices_and_estimates.rb 2017-04-02 06:56:17 -04:00
ricky 06050bd139 Removed unused method update_vehicles 2017-03-31 16:23:42 -04:00
ricky a48840ddfb Fixed typo 2017-03-31 16:12:21 -04:00
ricky 9b9aabee11 Format the VIN 2017-03-31 16:11:19 -04:00
ricky 7782627286 Split the VIN 2017-03-31 16:07:09 -04:00
ricky 41a113dc59 Do not hide notes 2017-03-31 16:02:48 -04:00
ricky b84e249dfb 0.4.3 2017-03-23 06:05:43 -04:00
ricky 6b45f767a4 Merge pull request #8 from rickbarrette/permissions
Permissions
2017-03-23 06:03:06 -04:00
ricky a34b6a07fc fixed typos 2017-03-23 05:56:26 -04:00
ricky 2ce811bbbf Update auth_helper.rb 2017-03-23 05:50:31 -04:00
ricky 02153de8b0 Added before filters add_customer, view_customer 2017-03-23 05:47:37 -04:00
ricky 68be20459b Added global_check_permission 2017-03-23 05:45:45 -04:00
ricky bbd03cc337 Update init.rb 2017-03-23 05:42:54 -04:00
ricky 4fc71a93f2 Update init.rb 2017-03-23 05:42:09 -04:00
ricky 8e7e1908e4 Update customers_controller.rb 2017-03-23 05:39:55 -04:00
ricky 89fba883ef Update customers_controller.rb 2017-03-23 05:38:06 -04:00
ricky 15f317fba1 Update customers_controller.rb 2017-03-23 05:36:51 -04:00
ricky 894ee9abfd added check_permission 2017-03-23 05:33:58 -04:00
ricky ca17807117 Update payments_controller.rb 2017-03-23 05:29:54 -04:00
ricky a70ba2f164 Update payments_controller.rb 2017-03-23 05:27:38 -04:00
ricky 78ac97298c Update payments_controller.rb 2017-03-23 05:25:57 -04:00
ricky 72cd349c1b Update payments_controller.rb 2017-03-23 05:23:44 -04:00
ricky 6fc1d27dca Update auth_helper.rb 2017-03-23 05:21:56 -04:00
ricky 525c6b99d6 Update auth_helper.rb 2017-03-23 05:19:13 -04:00
ricky 3eaff0ab30 Update auth_helper.rb 2017-03-23 05:14:47 -04:00
ricky 85b40bc9cf Update payments_controller.rb 2017-03-23 05:11:15 -04:00
ricky 37a2b95447 Update payments_controller.rb 2017-03-23 05:10:05 -04:00
ricky 33feb91713 added permission_checker 2017-03-23 05:08:33 -04:00
ricky f7357f30ce Update payments_controller.rb 2017-03-23 05:03:58 -04:00
ricky c0ae01018b Update payments_controller.rb 2017-03-23 05:01:01 -04:00
ricky 4353e910c8 Update payments_controller.rb 2017-03-23 04:57:22 -04:00
ricky bef9774c4e Update payments_controller.rb 2017-03-23 04:52:19 -04:00
ricky 863437b1b7 Added before filter to check permissions 2017-03-23 04:50:17 -04:00
ricky 7cfa15910a Update init.rb 2017-03-23 04:41:31 -04:00
ricky 2154a3d001 Update init.rb 2017-03-22 23:09:05 -04:00
ricky fdab090a3d Update init.rb 2017-03-22 23:06:12 -04:00
ricky 3f32b7fef1 Update init.rb 2017-03-22 22:53:21 -04:00
ricky 14422bc549 Update init.rb 2017-03-22 22:52:24 -04:00
ricky 6bb66597e8 Added some permissions
view_customers, add_customers, view_payments, add_payments
2017-03-22 22:44:09 -04:00
166 changed files with 3582 additions and 2575 deletions
+2
View File
@@ -1,3 +1,5 @@
.bundle .bundle
.config .config
.dockerrc
.vscode
Gemfile.lock Gemfile.lock
+3 -5
View File
@@ -1,15 +1,13 @@
source 'https://rubygems.org' source 'https://rubygems.org'
gem 'quickbooks-ruby' gem 'quickbooks-ruby'
gem 'quickbooks-ruby-base' gem 'oauth2'
gem 'oauth-plugin'
gem 'oauth'
gem 'roxml' gem 'roxml'
gem 'edmunds_vin'
gem 'will_paginate' gem 'will_paginate'
gem 'rails-jquery-autocomplete' gem 'rails-jquery-autocomplete'
gem 'jquery-rails', '~> 3.1.4'
gem 'jquery-ui-rails' gem 'jquery-ui-rails'
gem 'rexml'
gem 'combine_pdf'
group :assets do group :assets do
gem 'coffee-rails' gem 'coffee-rails'
+1 -1
View File
@@ -1,6 +1,6 @@
The MIT License (MIT) The MIT License (MIT)
Copyright (c) 2016 Rick Barrette Copyright (c) 2016 - 2026 Rick Barrette
Permission is hereby granted, free of charge, to any person obtaining a copy Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal of this software and associated documentation files (the "Software"), to deal
+84 -60
View File
@@ -1,86 +1,110 @@
#Redmine Quickbooks Online # Redmine QuickBooks Online
A plugin for Redmine to connect to Quickbooks Online A plugin for Redmine to connect to QuickBooks Online.
The goal of this project is to allow Redmine to connect with Quickbooks Online to create `Time Activity Entries` for completed work when an Issue is closed. The goal of this project is to allow Redmine to connect with QuickBooks Online to create Time Activity Entries for billable hours logged when an Issue is closed.
`Note: Although the core functionality is complete, this project is still under heavy development. I am still working on refining everthing and adding other features. Tags should be stable` ## Disclaimer
`Note: I am currently using this in a live production enviroment with no issues` **Note:** Although the core functionality is complete, this project is still under development and the master branch may be unstable. Tags should be stable and are recommended.
####Features ## Compatibility
* Issues can be assigned to a `Customer` via drop down in the edit Issue form
* The `Employee` for the Issue is assigned via the assigned Redmine User | Plugin Version | Redmine Version |
- This is set via a drop down in the user admistration page. | :--- | :--- |
* IF an `Issue` has been assined a `Customer` when an Issue is closed the following will happen: | Version 2026.1.0+ | Redmine 6.1 |
- A new `Time Activity` will be billed agaist the `Customer` assinged to the issue for each Redmine Time Entery. | Version 2.0.0+ | Redmine 5 |
+ Time Entries will be totalled up by Activity name. This will allow billing for diffrent activities without having to create seperate Issues. | Version 1.0.0+ | Redmine 4 |
+ The Time Activity names are used to lookup `Items` in Quickbooks. | Version 0.8.1 | Redmine 3 |
+ IF there isn'tany Items that match the Activity name it will be skipped, and will not be billed to the `Customer`
- Labor Rates are set by the `Item` in Quickbooks ## Features
* `Payments` Can be created via the Redmine application menu
* `Customers` Can be created via the Redmine application menu * **Customer Assignment:** Issues can be assigned to a Customer via a dropdown in the edit Issue form.
- `Customers` can be searched * Once a customer is attached to an Issue, you can attach an Estimate to the issue via a dropdown menu.
- Basic information for the `Customer` can be viewed/edit via the Customer page * **Employee Mapping:** An Employee is assigned to a Redmine User via a dropdown in the User Administration page.
* Webhook Support * **Automatic Billing:** If an Issue has been assigned a Customer, the following happens when the Issue is closed:
- `Invoices` are automaticly attached to an Issue if a line item has a hashtag number in a `Line Item` * A new Time Activity will be billed against the Customer assigned to the issue for each Redmine Time Entry.
+ `Invoice` Custom Fields are matched Issue Custom Fileds and are automaticly updated in Quickbooks. For example, this is usefull for extracting the Mileage In / Out from the Issue and updating the Invoice with the information. * Time Entries will be totalled up by Activity name. This allows billing for different activities without having to create separate Issues.
- `Customers` are automaticly updated in local database * The Time Activity names are used to dynamically lookup Items in QuickBooks.
* If there are no Items that match the Activity name, it will be skipped and will not be billed to the Customer.
* Labor Rates are set by the corresponding Item in QuickBooks.
* **Customer Management:** Customers can be created via the New Customer Page.
* Customers can be searched by name or phone number.
* Basic information for the Customer can be viewed/edited via the Customer page.
* **Webhook Support:**
* **Invoices:** Automatically attached to an Issue if a line item contains a hashtag number (e.g., `#123`).
* **Custom Fields:** Invoice Custom Fields are matched to Issue Custom Fields and are automatically updated in QuickBooks. (Useful for extracting Mileage In/Out from the Issue to update the Invoice).
* **Sync:** Customers are automatically updated in the local database.
* **Plugin View Hooks** Allows intergration of other features supported by companion plugins, for example [redmine_qbo_vehicles](https://github.com/rickbarrette/redmine_qbo_vehicles) adds customer vehicle interation
## Prerequisites ## Prerequisites
* Sign up to become a developer for Intuit https://developer.intuit.com/ * Sign up to become a developer for Intuit: https://developer.intuit.com/
* Create your own aplication to obtain your API keys * Create your own application to obtain your API keys.
* Set up webhook service to https://redmine.yourdomain.com/qbo/webhook * Set up the webhook service to `https://redmine.yourdomain.com/qbo/webhook`
- See https://developer.intuit.com/docs/0100_accounting/0300_developer_guides/webhooks
##The Install ## Installation
1. To install, clone this repo into your plugin folder 1. **Clone the plugin:**
Clone this repo into your plugin folder and checkout a tagged version.
```bash
cd path/to/redmine/plugins
git clone git@github.com:rickbarrette/redmine_qbo.git
cd redmine_qbo
git checkout <tag>
```
`git clone git@github.com:rickbarrette/redmine_qbo.git` 2. **Install dependencies:** *Crucial for Redmine 6 / Rails 7 compatibility.*
2. Migrate your database Bash
`rake redmine:plugins:migrate RAILS_ENV=production` ```
bundle install
```
3. Navigate to the plugin configuration page and suppy your own OAuth key & secret. 3. **Migrate your database:**
4. After saving your key & secret, you need to click on the Authenticate link on the plugin configuration page to authenticate with QBO. Bash
5. Assign an Employee to each of your users via the User Administration Page ```
bundle exec rake redmine:plugins:migrate RAILS_ENV=production
```
## Automatic Deploy 4. **Restart Redmine:** You must restart your Redmine server instance for the plugin and hooks to load.
5. **Configuration:**
* Navigate to the plugin configuration page (`Administration > Plugins > Configure`).
* Supply your own OAuth Key & Secret.
* After saving the Key & Secret, click the **Authenticate** link on the configuration page to connect to QBO.
6. **User Mapping:**
* Assign an Employee to each of your users via the **User Administration Page**.
If you want the redmine server to be automaticly restarted after a git pull event add this hook to your git hook directory
https://gist.github.com/rickbarrette/3c999c7f37e321f9c60380de99e494f5
## Usage ## Usage
To enable automatic `Time Activity` entries for an Issue , you need only to assign a `Customer` to an Issue via drop downs in the issue creation/update form. To enable automatic Time Activity entries for an Issue, you simply need to assign a Customer to an Issue via the dropdowns in the issue creation/update form.
Note: After the inital synchronization, this plugin will recieve push notifications via Intuit's webhook service. **Note:** After the initial synchronization, this plugin will receive push notifications via Intuit's webhook service.
## TODO ## Companion Plugin Hooks
* Abiltiy to add line items to a ticket in a dynamic table so they can be added to the invoice upon closing of the issue * :pdf_left, { issue: issue }
* Customer Deletion * :pdf_right, { issue: issue }
* Email Customer updates, provding a link that would: bypass the login page, go directly to the issue directing them to, and allow them to view only that issue. * :process_invoice_custom_fields, { issue: issue, invoice: invoice }
* Add Setting for Sandbox Mode * :show_customer_view_right, {customer: @customer}
* Refactor Models prefixed with Qbo...
* Seperate Vehicles into a seperate plugin
* Make HTML Pretty
* Intergrate Customer Search into Redmine Search
* Fix Issue sort by Customer
* MORE Stuff...
## License ## License
The MIT License (MIT) > The MIT License (MIT)
>
Copyright (c) 2016 rick barrette > Copyright (c) 2016 - 2026 Rick Barrette
>
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: > Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
>
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. > The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
>
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. > THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
Binary file not shown.

After

Width:  |  Height:  |  Size: 724 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 520 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 53 KiB

After

Width:  |  Height:  |  Size: 672 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 148 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 538 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 72 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 185 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 49 KiB

+134 -54
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -10,7 +10,6 @@
# This controller class will handle map management # This controller class will handle map management
class CustomersController < ApplicationController class CustomersController < ApplicationController
unloadable
include AuthHelper include AuthHelper
helper :issues helper :issues
@@ -27,102 +26,149 @@ class CustomersController < ApplicationController
include SortHelper include SortHelper
helper :timelog helper :timelog
before_filter :require_user, :except => :view before_action :add_customer, only: :new
skip_before_filter :verify_authenticity_token, :check_if_login_required, :only => [:view] before_action :view_customer, except: [:new, :view]
skip_before_action :verify_authenticity_token, :check_if_login_required, only: [:view]
default_search_scope :names autocomplete :customer, :name, full: true, extra_data: [:id]
autocomplete :customer, :name, :full => false, :extra_data => [:id] def allowed_params
params.require(:customer).permit(:name, :email, :primary_phone, :mobile_phone, :phone_number, :notes)
end
def filter_vehicles_by_customer # getter method for a customer's invoices
@filtered_vehicles = Vehicle.all.where(customer_id: params[:selected_customer]) # used for customer autocomplete field / issue form
def filter_invoices_by_customer
@filtered_invoices = Invoice.all.where(customer_id: params[:selected_customer])
end
# getter method for a customer's estimates
# used for customer autocomplete field / issue form
def filter_estimates_by_customer
@filtered_estimates = Estimate.all.where(customer_id: params[:selected_customer])
end end
# display a list of all customers # display a list of all customers
def index def index
if params[:search] if params[:search]
@customers = Customer.search(params[:search]).paginate(:page => params[:page]) @customers = Customer.search(params[:search]).order(:name).paginate(page: params[:page])
if only_one_non_zero?(@customers) if only_one_non_zero?(@customers)
redirect_to @customers.first redirect_to @customers.first
end end
end end
end end
# initialize a new customer
def new def new
@customer = Customer.new @customer = Customer.new
end end
# create a new customer
def create def create
@customer = Customer.new(params[:customer]) @customer = Customer.new(allowed_params)
if @customer.save @customer.save
flash[:notice] = "New Customer Created" log "Customer ##{@customer.id} created successfully."
flash[:notice] = t :notice_customer_created
redirect_to @customer redirect_to @customer
else rescue => e
flash[:error] = @customer.errors.full_messages.to_sentence log "Failed to create customer: #{e.message}"
flash[:error] = e.message
redirect_to new_customer_path redirect_to new_customer_path
end end
end
# display a specific customer # display a specific customer
def show def show
begin
@customer = Customer.find_by_id(params[:id]) @customer = Customer.find_by_id(params[:id])
@vehicles = @customer.vehicles.paginate(:page => params[:page]) return render_404 unless @customer
@issues = @customer.issues
rescue ActiveRecord::RecordNotFound @open_issues = @customer.issues
.joins(:status)
.includes(:status, :project, :tracker, :priority)
.where(issue_statuses: { is_closed: false })
.order(id: :desc)
@closed_issues = @customer.issues
.joins(:status)
.includes(:status, :project, :tracker, :priority)
.where(issue_statuses: { is_closed: true })
.order(id: :desc)
@hours = TimeEntry
.joins(:issue)
.where(issues: { id: @open_issues.select(:id) })
.sum(:hours)
@closed_hours = TimeEntry
.joins(:issue)
.where(issues: { id: @closed_issues.select(:id) })
.sum(:hours)
rescue => e
Rails.logger.error "Failed to load customer ##{params[:id]}: #{e.message}\n#{e.backtrace.join("\n")}"
flash[:error] = e.message
render_404 render_404
end end
end
# return an HTML form for editing a customer # return an HTML form for editing a customer
def edit def edit
begin
@customer = Customer.find_by_id(params[:id]) @customer = Customer.find_by_id(params[:id])
rescue ActiveRecord::RecordNotFound return render_404 unless @customer
rescue => e
log "Failed to edit customer"
flash[:error] = e.message
render_404 render_404
end end
end
# update a specific customer # update a specific customer
def update def update
begin
@customer = Customer.find_by_id(params[:id]) @customer = Customer.find_by_id(params[:id])
if @customer.update_attributes(params[:customer]) @customer.update(allowed_params)
flash[:notice] = "Customer updated" flash[:notice] = t :notice_customer_updated
redirect_to @customer redirect_to @customer
else rescue => e
log "Failed to update customer: #{e.message}"
flash[:error] = e.message
redirect_to edit_customer_path redirect_to edit_customer_path
flash[:error] = @customer.errors.full_messages.to_sentence if @customer.errors
end
rescue ActiveRecord::RecordNotFound
render_404
end
end end
def destroy # creates new customer view tokens, removes expired tokens & redirects to newly created customer view with new token.
begin def share
Customer.find_by_id(params[:id]).destroy issue = Issue.find(params[:id])
flash[:notice] = "Customer deleted successfully" token = issue.share_token
redirect_to action: :index redirect_to view_path(token.token)
rescue ActiveRecord::RecordNotFound rescue ActiveRecord::RecordNotFound
flash[:error] = t(:notice_issue_not_found)
render_404 render_404
end end
end
# Customer view for an issue # displays an issue for a customer with a provided security CustomerToken
def view def view
User.current = User.anonymous
User.current = User.find_by lastname: 'Anonymous' # Load only active, non-expired token
@token = CustomerToken.active.find_by(token: params[:token])
return render_403 unless @token
@token = CustomerToken.where("token = ? and expires_at > ?", params[:token], Time.now) # Load associated issue
@token = @token.first @issue = @token.issue
if @token return render_403 unless @issue
# Optional: enforce token belongs to the issue's customer
return render_403 unless @issue.customer_id == @token.issue.customer_id
# Store token in session for subsequent requests if needed
session[:token] = @token.token session[:token] = @token.token
@issue = Issue.find @token.issue_id
@journals = @issue.journals. load_issue_data
preload(:details). rescue ActiveRecord::RecordNotFound
preload(:user => :email_address). render_403
reorder(:created_on, :id).to_a end
private
def load_issue_data
@journals = @issue.journals.preload(:details).preload(user: :email_address).reorder(:created_on, :id).to_a
@journals.each_with_index { |j, i| j.indice = i + 1 } @journals.each_with_index { |j, i| j.indice = i + 1 }
@journals.reject!(&:private_notes?) unless User.current.allowed_to?(:view_private_notes, @issue.project) @journals.reject!(&:private_notes?) unless User.current.allowed_to?(:view_private_notes, @issue.project)
Journal.preload_journals_details_custom_fields(@journals) Journal.preload_journals_details_custom_fields(@journals)
@@ -132,18 +178,25 @@ class CustomersController < ApplicationController
@changesets = @issue.changesets.visible.preload(:repository, :user).to_a @changesets = @issue.changesets.visible.preload(:repository, :user).to_a
@changesets.reverse! if User.current.wants_comments_in_reverse_order? @changesets.reverse! if User.current.wants_comments_in_reverse_order?
@relations = @issue.relations.select {|r| r.other_issue(@issue) && r.other_issue(@issue).visible? } @relations = @issue.relations.select { |r| r.other_issue(@issue)&.visible? }
@allowed_statuses = @issue.new_statuses_allowed_to(User.current) @allowed_statuses = @issue.new_statuses_allowed_to(User.current)
@priorities = IssuePriority.active @priorities = IssuePriority.active
@time_entry = TimeEntry.new(:issue => @issue, :project => @issue.project) @time_entry = TimeEntry.new(issue: @issue, project: @issue.project)
@relation = IssueRelation.new @relation = IssueRelation.new
else
render_403
end
end end
private # redmine permission - add customers
def add_customer
global_check_permission(:add_customers)
end
# redmine permission - view customers
def view_customer
global_check_permission(:view_customers)
end
# checks to see if there is only one item in an array
# @return true if array only has one item
def only_one_non_zero?( array ) def only_one_non_zero?( array )
found_non_zero = false found_non_zero = false
array.each do |val| array.each do |val|
@@ -155,4 +208,31 @@ class CustomersController < ApplicationController
found_non_zero found_non_zero
end end
# format a quickbooks address to a human readable string
def address_to_s(address)
return if address.nil?
lines = [
address.line1,
address.line2,
address.line3,
address.line4,
address.line5
].compact_blank
city_line = [
address.city,
address.country_sub_division_code,
address.postal_code
].compact_blank.join(" ")
lines << city_line unless city_line.blank?
lines.join("\n")
end
def log(msg)
Rails.logger.info "[CustomersController] #{msg}"
end
end end
+63 -11
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,20 +8,72 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class EstimateController < ApplicationController class EstimateController < ApplicationController
unloadable
include AuthHelper include AuthHelper
before_filter :require_user before_action :require_user, unless: -> { session[:token].nil? }
skip_before_action :verify_authenticity_token, :check_if_login_required, unless: -> { session[:token].nil? }
before_action :load_estimate, only: [:show, :doc]
# # Displays the estimate PDF in the browser or redirects with an error if not found.
# Downloads and forwards the estimate pdf def doc
# render_pdf(@estimate)
end
# Displays the estimate PDF in the browser or redirects with an error if not found.
def show def show
base = QboEstimate.get_base.service render_pdf(@estimate)
estimate = base.fetch_by_id(params[:id])
@pdf = base.pdf(estimate)
send_data @pdf, filename: "estimate #{estimate.doc_number}.pdf", :disposition => 'inline', :type => "application/pdf"
end end
private
# Loads the estimate based on ID or doc number, with a fallback to sync if not found locally.
def load_estimate
log "Attempting to load Estimate with params: #{params.inspect}"
@estimate = find_estimate || sync_and_find_estimate
unless @estimate
redirect_back fallback_location: root_path, flash: { error: I18n.t(:notice_estimate_not_found) }
end
end
# Attempts to find the estimate locally by ID or doc number.
def find_estimate
return Estimate.find_by(doc_number: params[:search]) if params[:search].present?
return Estimate.find_by(id: params[:id]) if params[:id].present?
end
# If the estimate is not found locally, attempts to sync it from the source and find it again.
def sync_and_find_estimate
if params[:search].present?
log "Estimate #{params[:search]} not found locally. Syncing by doc number."
Estimate.sync_by_doc_number(params[:search])
return Estimate.find_by(doc_number: params[:search])
end
if params[:id].present?
log "Estimate #{params[:id]} not found locally. Syncing by ID."
Estimate.sync_by_id(params[:id])
return Estimate.find_by(id: params[:id])
end
nil
rescue StandardError => e
log "Estimate sync failed: #{e.message}"
nil
end
# Renders the estimate PDF or redirects with an error if rendering fails.
def render_pdf(estimate)
pdf, ref = EstimatePdfService.new(qbo: QboConnectionService.current!).fetch_pdf(doc_ids: [estimate.id])
send_data( pdf, filename: "estimate #{ref}.pdf", disposition: :inline, type: "application/pdf" )
rescue StandardError => e
log "PDF render failed for Estimate #{estimate&.id}: #{e.message}"
redirect_back fallback_location: root_path, flash: { error: I18n.t(:notice_estimate_not_found) }
end
# Logs messages with a consistent prefix for easier debugging.
def log(msg)
Rails.logger.info "[EstimateController] #{msg}"
end
end end
+21 -12
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,20 +8,29 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class InvoiceController < ApplicationController class InvoiceController < ApplicationController
unloadable
include AuthHelper include AuthHelper
before_filter :require_user, :unless => proc {|c| session[:token].nil? } before_action :require_user, unless: -> { session[:token].nil? }
skip_before_filter :verify_authenticity_token, :check_if_login_required, :unless => proc {|c| session[:token].nil? } skip_before_action :verify_authenticity_token, :check_if_login_required, unless: -> { session[:token].nil? }
# # Displays the invoice PDF in the browser or redirects with an error if not found.
# Downloads and forwards the invoice pdf
#
def show def show
base = QboInvoice.get_base log "Processing request for #{request.original_url}"
invoice = base.fetch_by_id(params[:id])
@pdf = base.pdf(invoice) invoice_ids = Array(params[:invoice_ids] || params[:id])
send_data @pdf, filename: "invoice #{invoice.doc_number}.pdf", :disposition => 'inline', :type => "application/pdf" pdf, ref = InvoicePdfService.new(qbo: QboConnectionService.current!).fetch_pdf(doc_ids: invoice_ids)
send_data pdf, filename: "invoice #{ref}.pdf", disposition: :inline, type: "application/pdf"
rescue StandardError => e
log "Invoice PDF failure: #{e.message}"
redirect_back fallback_location: root_path, flash: { error: I18n.t(:notice_invoice_not_found) }
end
private
# Logs messages with a consistent prefix for easier debugging.
def log(msg)
Rails.logger.info "[InvoiceController] #{msg}"
end end
end end
+47 -120
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -9,141 +9,68 @@
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboController < ApplicationController class QboController < ApplicationController
unloadable
require 'openssl'
include AuthHelper include AuthHelper
before_filter :require_user, :except => :qbo_webhook before_action :require_user, except: :webhook
skip_before_filter :verify_authenticity_token, :check_if_login_required, :only => [:qbo_webhook] skip_before_action :verify_authenticity_token, :check_if_login_required, only: :webhook
# # Initiates the OAuth authentication process by redirecting the user to the QuickBooks authorization URL. The callback URL is generated based on the application's settings and routes.
# Called when the QBO Top Menu us shown
#
def index
@qbo = Qbo.first
@customer_count = Customer.count
@qbo_item_count = QboItem.count
@qbo_employee_count = QboEmployee.count
@qbo_invoice_count = QboInvoice.count
@qbo_estimate_count = QboEstimate.count
end
#
# Called when the user requests that Redmine to connect to QBO
#
def authenticate def authenticate
callback = qbo_oauth_callback_url redirect_to QboOauthService.authorization_url(callback_url: callback_url)
token = Qbo.get_oauth_consumer.get_request_token(:oauth_callback => callback)
session[:qb_request_token] = Marshal.dump(token)
redirect_to("https://appcenter.intuit.com/Connect/Begin?oauth_token=#{token.token}") and return
end end
# # Handles the OAuth callback from QuickBooks. Exchanges the authorization code for access and refresh tokens, saves the connection details, and redirects to the sync page with a success notice. If any error occurs during the process, logs the error and redirects back to the plugin settings page with an error message.
# Called by QBO after authentication has been processed
#
def oauth_callback def oauth_callback
at = Marshal.load(session[:qb_request_token]).get_access_token(:oauth_verifier => params[:oauth_verifier]) QboOauthService.exchange!(code: params[:code], callback_url: callback_url, realm_id: params[:realmId])
#There can only be one... redirect_to qbo_sync_path, flash: { notice: I18n.t(:label_connected) }
Qbo.destroy_all
# Save the authentication information rescue StandardError => e
qbo = Qbo.new log "OAuth failure: #{e.message}"
qbo.qb_token = at.token redirect_to plugin_settings_path(:redmine_qbo), flash: { error: I18n.t(:label_error) }
qbo.qb_secret = at.secret
qbo.token_expires_at = 6.months.from_now.utc
qbo.reconnect_token_at = 5.months.from_now.utc
qbo.company_id = params['realmId']
if qbo.save!
redirect_to qbo_sync_path, :flash => { :notice => "Successfully connected to Quickbooks" }
else
redirect_to plugin_settings_path(:redmine_qbo), :flash => { :error => "Error" }
end
end end
# Manual Billing # Manual billing endpoint to trigger the billing process for a specific issue. Validates the issue and its associations, enqueues a job to bill the issue's time entries, and redirects back to the issue with a notice. If validation fails, redirects back with an error message.
def bill def bill
i = Issue.find_by_id params[:id] issue = Issue.find_by(id: params[:id])
if i.customer raise I18n.t(:notice_error_issue_not_found) unless issue
i.bill_time raise I18n.t(:notice_billing_error_no_customer) unless issue.customer
redirect_to i, :flash => { :notice => "Successfully Billed #{i.customer.name}" } raise I18n.t(:notice_billing_error_no_employee) unless issue.assigned_to&.employee_id.present?
else raise I18n.t(:notice_billing_error_no_qbo) unless Qbo.exists?
redirect_to i, :flash => { :error => "Cannot bill without a customer assigned" }
end BillIssueTimeJob.perform_later(issue.id)
redirect_to issue, flash: { notice: "#{I18n.t(:label_billing_enqueued)} #{issue.customer.name}"}
rescue StandardError => e
redirect_to issue || root_path, flash: { error: e.message }
end end
# Quickbooks Webhook Callback # Manual sync endpoint to trigger a full synchronization of QuickBooks entities with the local database. Enqueues all relevant sync jobs and redirects to the home page with a notice that syncing has started.
def qbo_webhook
# check the payload
signature = request.headers['intuit-signature']
key = Setting.plugin_redmine_qbo['settingsWebhookToken']
data = request.body.read
hash = Base64.encode64(OpenSSL::HMAC.digest(OpenSSL::Digest::Digest.new('sha256'), key, data)).strip()
# proceed if the request is good
if hash.eql? signature
if request.headers['content-type'] == 'application/json'
data = JSON.parse(data)
else
# application/x-www-form-urlencoded
data = params.as_json
end
# Process the information
entities = data['eventNotifications'][0]['dataChangeEvent']['entities']
entities.each do |entity|
id = entity['id'].to_i
name = entity['name']
# TODO rename all other models!
name.prepend("Qbo") if not name.eql? "Customer"
# Magicly initialize the correct class
obj = name.constantize
# for merge events
obj.destroy(entity['deletedId']) if entity['deletedId']
#Check to see if we are deleting a record
if entity['operation'].eql? "Delete"
obj.destroy(id)
#if not then update!
else
obj.sync_by_id(id)
end
end
# Record that last time we updated
Qbo.update_time_stamp
# The webhook doesn't require a response but let's make sure we don't send anything
render :nothing => true
else
render nothing: true, status: 400
end
end
#
# Synchronizes the QboCustomer table with QBO
#
def sync def sync
# Update info in background QboSyncDispatcher.full_sync!
Thread.new do redirect_to :home, flash: { notice: I18n.t(:label_syncing) }
if Qbo.exists?
Customer.sync
QboItem.sync
QboEmployee.sync
QboEstimate.sync
QboInvoice.sync
# Record the last sync time
Qbo.update_time_stamp
end
ActiveRecord::Base.connection.close
end end
redirect_to qbo_path(:redmine_qbo), :flash => { :notice => "Successfully synced to Quickbooks" } # Endpoint to receive QuickBooks webhook notifications. Validates the request and processes the payload to sync relevant data to Redmine. Responds with appropriate HTTP status codes based on success or failure of processing.
def webhook
QboWebhookProcessor.process!(request: request)
head :ok
rescue StandardError => e
log "Webhook failure: #{e.message}"
head :unauthorized
end
private
# Constructs the OAuth callback URL based on the application's settings and routes. This URL is used during the OAuth flow to redirect users back to the application after authentication with QuickBooks.
def callback_url
"#{Setting.protocol}://#{Setting.host_name}#{qbo_oauth_callback_path}"
end
# Logs messages with a consistent prefix for easier debugging and monitoring.
def log(msg)
Rails.logger.info "[QboController] #{msg}"
end end
end end
-126
View File
@@ -1,126 +0,0 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
# This controller class will handle map management
class VehiclesController < ApplicationController
unloadable
include AuthHelper
before_filter :require_user
# display a list of all vehicles
def index
if params[:customer_id]
begin
@vehicles = Customer.find_by_id(params[:customer_id]).vehicles.paginate(:page => params[:page])
rescue ActiveRecord::RecordNotFound
render_404
end
end
if params[:search]
@vehicles = Vehicle.search(params[:search]).paginate(:page => params[:page])
if only_one_non_zero?(@vehicles)
redirect_to @vehicles.first
end
end
end
# return an HTML form for creating a new vehicle
def new
@vehicle = Vehicle.new
@customers = Customer.all.order(:name)
@customer = params[:customer_id] if params[:customer_id]
end
# create a new vehicle
def create
@vehicle = Vehicle.new(params[:vehicle])
if @vehicle.save
flash[:notice] = "New Vehicle Created"
redirect_to @vehicle
else
flash[:error] = @vehicle.errors.full_messages.to_sentence
redirect_to new_vehicle_path
end
end
# display a specific vehicle
def show
begin
@vehicle = Vehicle.find_by_id(params[:id])
rescue ActiveRecord::RecordNotFound
render_404
end
end
# return an HTML form for editing a vehicle
def edit
begin
@vehicle = Vehicle.find_by_id(params[:id])
@customer = @vehicle.customer.id
@customers = Customer.all.order(:name)
rescue ActiveRecord::RecordNotFound
render_404
end
end
# update a specific vehicle
def update
@customer = params[:customer]
begin
@vehicle = Vehicle.find_by_id(params[:id])
if @vehicle.update_attributes(params[:vehicle])
flash[:notice] = "Vehicle updated"
redirect_to @vehicle
else
flash[:error] = @vehicle.errors.full_messages.to_sentence if @vehicle.errors
redirect_to edit_vehicle_path
end
rescue ActiveRecord::RecordNotFound
render_404
end
end
# delete a specific vehicle
def destroy
begin
Vehicle.find_by_id(params[:id]).destroy
flash[:notice] = "Vehicle deleted successfully"
redirect_to action: :index
rescue ActiveRecord::RecordNotFound
render_404
end
end
# returns a dynamic list of vehicles owned by a customer
def update_vehicles
@vehicles = Customer.find_by(customer_id: params[:customer_id].to_i).vehicles
respond_to do |format|
format.html { render(:text => "not implemented") }
format.js
end
end
private
def only_one_non_zero?( array )
found_non_zero = false
array.each do |val|
if val!=0
return false if found_non_zero
found_non_zero = true
end
end
found_non_zero
end
end
+39 -2
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -13,7 +13,44 @@ module AuthHelper
def require_user def require_user
return unless session[:token].nil? return unless session[:token].nil?
if !User.current.logged? if !User.current.logged?
render :file => "public/401.html.erb", :status => :unauthorized, :layout =>true flash[:error] = t :notice_forbidden
render_403
end end
end end
def allowed_to?(action)
return false if User.current.nil?
project = Project.find(params[:project_id])
return false if project.nil?
return true if User.current.allowed_to?(action, project)
false
end
def check_permission(permission)
if !allowed_to?(permission)
flash[:error] = t :notice_forbidden
render_403
end
end
def global_check_permission(permission)
if !globaly_allowed_to?(permission)
flash[:error] = t :notice_forbidden
render_403
end
end
def globaly_allowed_to?( action)
return false if User.current.nil?
projects = Project.all
projects.each { |p|
if User.current.allowed_to?(action, p)
return true
end
}
false
end
end end
+116
View File
@@ -0,0 +1,116 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class BillIssueTimeJob < ActiveJob::Base
queue_as :default
retry_on StandardError, wait: 5.minutes, attempts: 5
# Perform billing of unbilled time entries for a given issue by creating corresponding TimeActivity records in QuickBooks Online, and then marking those entries as billed in Redmine. This job is typically triggered after an invoice is created or updated to ensure all relevant time is captured for billing.
def perform(issue_id)
issue = Issue.find(issue_id)
log "Starting billing for issue ##{issue.id}"
issue.with_lock do
unbilled_entries = issue.time_entries.where(billed: [false, nil]).lock
return if unbilled_entries.blank?
totals = aggregate_hours(unbilled_entries)
return if totals.blank?
log "Aggregated hours for billing: #{totals.inspect}"
qbo = QboConnectionService.current!
raise "No QBO configuration found" unless qbo
qbo.perform_authenticated_request do |access_token|
create_time_activities(issue, totals, access_token, qbo)
end
# Only mark billed AFTER successful QBO creation
unbilled_entries.update_all(billed: true)
end
log "Completed billing for issue ##{issue.id}"
Qbo.update_time_stamp
rescue => e
log "Billing failed for issue ##{issue_id} - #{e.message}"
raise e
end
private
# Aggregate time entries by activity name and sum their hours
def aggregate_hours(entries)
entries.includes(:activity)
.group_by { |e| e.activity&.name }
.transform_values { |rows| rows.sum(&:hours) }
.compact
end
# Create TimeActivity records in QBO for each activity type with the appropriate hours and link them to the issue's assigned employee and customer
def create_time_activities(issue, totals, access_token, qbo)
log "Creating TimeActivity records in QBO for issue ##{issue.id}"
time_service = Quickbooks::Service::TimeActivity.new( company_id: qbo.realm_id, access_token: access_token)
item_service = Quickbooks::Service::Item.new( company_id: qbo.realm_id, access_token: access_token )
totals.each do |activity_name, hours_float|
next if activity_name.blank?
next if hours_float.to_f <= 0
item = find_item(item_service, activity_name)
next unless item
hours, minutes = convert_hours(hours_float)
time_entry = Quickbooks::Model::TimeActivity.new
time_entry.description = build_description(issue)
time_entry.employee_id = issue.assigned_to.employee_id
time_entry.customer_id = issue.customer_id
time_entry.billable_status = "Billable"
time_entry.hours = hours
time_entry.minutes = minutes
time_entry.name_of = "Employee"
time_entry.txn_date = Date.today
time_entry.hourly_rate = item.unit_price
time_entry.item_id = item.id
log "Creating TimeActivity for #{activity_name} (#{hours}h #{minutes}m)"
time_service.create(time_entry)
end
end
# Convert a decimal hours float into separate hours and minutes components for QBO TimeActivity
def convert_hours(hours_float)
total_minutes = (hours_float.to_f * 60).round
hours = total_minutes / 60
minutes = total_minutes % 60
[hours, minutes]
end
# Build a descriptive string for the TimeActivity based on the issue's tracker, ID, subject, and completion status
def build_description(issue)
base = "#{issue.tracker} ##{issue.id}: #{issue.subject}"
return base if issue.closed?
"#{base} (Partial @ #{issue.done_ratio}%)"
end
# Find an item in QBO by name, escaping single quotes to prevent query issues. Returns nil if not found.
def find_item(item_service, name)
safe = name.gsub("'", "\\\\'")
item_service.query("SELECT * FROM Item WHERE Name = '#{safe}'").first
end
def log(msg)
Rails.logger.info "[BillIssueTimeJob] #{msg}"
end
end
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,30 +8,29 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class Payment class CustomerSyncJob < ApplicationJob
unloadable queue_as :default
retry_on StandardError, wait: 5.minutes, attempts: 5
include ActiveModel::Model # Perform a full sync of all customers, or an incremental sync of only those updated since the last sync
def perform(full_sync: false, id: nil)
qbo = QboConnectionService.current!
raise "No QBO configuration found" unless qbo
attr_accessor :errors, :customer_id, :account_id, :payment_method_id, :total_amount log "Starting #{full_sync ? 'full' : 'incremental'} sync for customer ##{id || 'all'}..."
validates_presence_of :customer_id, :account_id, :payment_method_id, :total_amount
validates :total_amount, numericality: true
def save service = CustomerSyncService.new(qbo: qbo)
payment = Quickbooks::Model::Payment.new
payment.customer_id = @customer_id.to_i if id.present?
payment.deposit_to_account_id = @account_id.to_i service.sync_by_id(id)
payment.payment_method_id = @payment_method_id.to_i else
payment.total = @total_amount service.sync(full_sync: full_sync)
Qbo.get_base(:payment).service.update(payment) end
end end
def save! private
save
end
# Dummy stub to make validtions happy. def log(msg)
def update_attribute Rails.logger.info "[CustomerSyncJob] #{msg}"
end end
end end
+36
View File
@@ -0,0 +1,36 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class EmployeeSyncJob < ApplicationJob
queue_as :default
retry_on StandardError, wait: 5.minutes, attempts: 5
# Performs a sync of employees from QuickBooks Online.
def perform(full_sync: false, id: nil)
qbo = QboConnectionService.current!
raise "No QBO configuration found" unless qbo
log "Starting #{full_sync ? 'full' : 'incremental'} sync for employee ##{id || 'all'}..."
service = EmployeeSyncService.new(qbo: qbo)
if id.present?
service.sync_by_id(id)
else
service.sync(full_sync: full_sync)
end
end
private
def log(msg)
Rails.logger.info "[EmployeeSyncJob] #{msg}"
end
end
+38
View File
@@ -0,0 +1,38 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class EstimateSyncJob < ApplicationJob
queue_as :default
retry_on StandardError, wait: 5.minutes, attempts: 5
# Performs a sync of estimates from QuickBooks Online.
def perform(full_sync: false, id: nil, doc_number: nil)
qbo = QboConnectionService.current!
raise "No QBO configuration found" unless qbo
log "Starting #{full_sync ? 'full' : 'incremental'} sync for estimate ##{id || doc_number || 'all'}..."
service = EstimateSyncService.new(qbo: qbo)
if id.present?
service.sync_by_id(id)
elsif doc_number.present?
service.sync_by_doc(doc_number)
else
service.sync(full_sync: full_sync)
end
end
private
def log(msg)
Rails.logger.info "[EstimateSyncJob] #{msg}"
end
end
+36
View File
@@ -0,0 +1,36 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class InvoiceSyncJob < ApplicationJob
queue_as :default
retry_on StandardError, wait: 5.minutes, attempts: 5
# Performs a sync of invoices from QuickBooks Online.
def perform(full_sync: false, id: nil)
qbo = QboConnectionService.current!
raise "No QBO configuration found" unless qbo
log "Starting #{full_sync ? 'full' : 'incremental'} sync for invoice ##{id || 'all'}..."
service = InvoiceSyncService.new(qbo: qbo)
if id.present?
service.sync_by_id(id)
else
service.sync(full_sync: full_sync)
end
end
private
def log(msg)
Rails.logger.info "[InvoiceSyncJob] #{msg}"
end
end
@@ -1,19 +1,24 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. #The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class HeaderFooterHookListener < Redmine::Hook::ViewListener
def view_layouts_base_html_head(context = {}) class QboSyncDispatcher
#nothing
end
def view_layouts_base_body_bottom(context = {}) SYNC_JOBS = [
return "<div id='qbo_footer' align='center'><b>Last Sync: </b> #{Qbo.last_sync if Qbo.exists?}</div>" CustomerSyncJob,
EstimateSyncJob,
InvoiceSyncJob,
EmployeeSyncJob
].freeze
# Dispatches all synchronization jobs to perform a full sync of QuickBooks entities with the local database. Each job is enqueued with the `full_sync` flag set to true.
def self.full_sync!
SYNC_JOBS.each { |job| job.perform_later(full_sync: true) }
end end
end end
+42
View File
@@ -0,0 +1,42 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboWebhookProcessor
# Processes the incoming QuickBooks webhook request by validating the signature and enqueuing a background job to handle the webhook payload. Raises an error if the signature is invalid.
def self.process!(request:)
body = request.raw_post
signature = request.headers['intuit-signature']
secret = Setting.plugin_redmine_qbo['settingsWebhookToken']
raise "Invalid signature" unless valid_signature?(body, signature, secret)
WebhookProcessJob.perform_later(body)
end
private
# Validates the QuickBooks webhook request by computing the HMAC signature and comparing it to the provided signature. Returns false if either the signature or secret is blank, or if the computed signature does not match the provided signature.
def self.valid_signature?(body, signature, secret)
return false if signature.blank? || secret.blank?
log "Validating signature"
digest = OpenSSL::Digest.new('sha256')
computed = Base64.strict_encode64(
OpenSSL::HMAC.digest(digest, secret, body)
)
ActiveSupport::SecurityUtils.secure_compare(computed, signature)
end
def self.log(msg)
Rails.logger.info "[QboWebhookProcessor] #{msg}"
end
end
+68
View File
@@ -0,0 +1,68 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class WebhookProcessJob < ActiveJob::Base
queue_as :default
retry_on StandardError, wait: 5.minutes, attempts: 5
ALLOWED_ENTITIES = %w[
Customer
Invoice
Estimate
Employee
].freeze
# Process incoming QBO webhook notifications and sync relevant data to Redmine
def perform(raw_body)
log "Received webhook: #{raw_body}"
data = JSON.parse(raw_body)
data.fetch('eventNotifications', []).each do |notification|
entities = notification.dig('dataChangeEvent', 'entities') || []
entities.each do |entity|
process_entity(entity)
end
end
Qbo.update_time_stamp
end
private
# Process a single entity from the webhook payload and sync it to Redmine if it's an allowed type
def process_entity(entity)
log "Processing entity: #{entity}"
name = entity['name']
id = entity['id']&.to_i
return unless ALLOWED_ENTITIES.include?(name)
model = name.safe_constantize
return unless model
if entity['deletedId']
model.delete(entity['deletedId'])
return
end
if entity['operation'] == "Delete"
model.delete(id)
else
model.sync_by_id(id)
end
rescue => e
log "#{e.message}"
end
def log(msg)
Rails.logger.info "[WebhookProcessJob] #{msg}"
end
end
+101
View File
@@ -0,0 +1,101 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
module QuickbooksOauth
extend ActiveSupport::Concern
#== Instance Methods
# This method will attempt to execute the block and if it encounters an OAuth2::Error or Quickbooks::AuthorizationFailure it will attempt to refresh the token and retry the block. It will try this up to 3 times before giving up and raising an exception.
def perform_authenticated_request(&block)
attempts = 0
begin
yield oauth_access_token
rescue OAuth2::Error, Quickbooks::AuthorizationFailure => ex
log "perform_authenticated_request: #{ex.message}"
# to prevent an infinite loop here keep a counter and bail out after N times...
attempts += 1
raise "QuickbooksOauth:ExceededAuthAttempts" if attempts >= 3
# check if its an invalid_grant first, but assume it is for now
refresh_token!
retry
end
end
# This method will attempt to refresh the access token and update the record with the new access token, refresh token and their respective expiration times. If the refresh token expires in more than 0 seconds then we will set the refresh token expiration time to that value, otherwise we will set it to 100 days from now.
def refresh_token!
log "refresh_token!"
t = oauth_access_token
refreshed = t.refresh!
if refreshed.params['x_refresh_token_expires_in'].to_i > 0
oauth2_refresh_token_expires_at = Time.now + refreshed.params['x_refresh_token_expires_in'].to_i.seconds
else
oauth2_refresh_token_expires_at = 100.days.from_now
end
log "refresh_token!: #{oauth2_refresh_token_expires_at}"
update!(
oauth2_access_token: refreshed.token,
oauth2_access_token_expires_at: Time.at(refreshed.expires_at),
oauth2_refresh_token: refreshed.refresh_token,
oauth2_refresh_token_expires_at: oauth2_refresh_token_expires_at
)
end
# This method will return an instance of the OAuth2::Client class that is configured with the consumer key, consumer secret and the appropriate URLs for the Intuit OAuth2 service. It will also set the sandbox mode based on the plugin settings.
def oauth_client
self.class.construct_oauth2_client
end
# This method will return an instance of the OAuth2::AccessToken class that is configured with the current access token, refresh token and the OAuth2 client. This access token can be used to make authenticated requests to the Intuit API.
def oauth_access_token
OAuth2::AccessToken.new(oauth_client, oauth2_access_token, refresh_token: oauth2_refresh_token)
end
# This method is an alias for the oauth_access_token method and is used to provide a more intuitive name for the access token when making authenticated requests.
def consumer
oauth_access_token
end
module ClassMethods
# This method will construct and return an instance of the OAuth2::Client class that is configured with the consumer key, consumer secret and the appropriate URLs for the Intuit OAuth2 service. It will also set the sandbox mode based on the plugin settings. This method is used by the instance method oauth_client to create a new OAuth2 client for each instance of the model that includes this concern.
def construct_oauth2_client
oauth_consumer_key = Setting.plugin_redmine_qbo['settingsOAuthConsumerKey']
oauth_consumer_secret = Setting.plugin_redmine_qbo['settingsOAuthConsumerSecret']
# Are we are playing in the sandbox?
Quickbooks.sandbox_mode = Setting.plugin_redmine_qbo[:sandbox] ? true : false
log "Sandbox mode: #{Quickbooks.sandbox_mode}"
options = {
site: "https://appcenter.intuit.com/connect/oauth2",
authorize_url: "https://appcenter.intuit.com/connect/oauth2",
token_url: "https://oauth.platform.intuit.com/oauth2/v1/tokens/bearer"
}
OAuth2::Client.new(oauth_consumer_key, oauth_consumer_secret, options)
end
end
private
def log(msg)
Rails.logger.info "[QuickbooksOauth] #{msg}"
end
end
+135 -135
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -9,182 +9,179 @@
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class Customer < ActiveRecord::Base class Customer < ActiveRecord::Base
unloadable
include Redmine::Acts::Searchable
include Redmine::Acts::Event
include Redmine::I18n
has_many :issues has_many :issues
has_many :qbo_purchases has_many :invoices
has_many :vehicles has_many :estimates
attr_accessible :name, :notes, :email, :primary_phone, :mobile_phone
validates_presence_of :id, :name validates_presence_of :id, :name
before_validation :normalize_phone_numbers
self.primary_key = :id self.primary_key = :id
# returns a human readable string acts_as_searchable columns: %w[name phone_number mobile_phone_number ],
def to_s scope: ->(_context) { left_joins(:project) },
return name date_column: :updated_at
acts_as_event :title => Proc.new {|o| "#{o}"},
:url => Proc.new {|o| { :controller => 'customers', :action => 'show', :id => o.id} },
:type => :to_s,
:description => Proc.new {|o| "#{I18n.t :label_primary_phone}: #{o.phone_number} #{I18n.t:label_mobile_phone}: #{o.mobile_phone_number}"},
:datetime => Proc.new {|o| o.updated_at || o.created_at}
# Returns the details of the customer. If the details have already been fetched, it returns the cached version. Otherwise, it fetches the details from QuickBooks Online and caches them for future use. This method is used to access the customer's information in a way that minimizes unnecessary API calls to QBO, improving performance and reducing latency.
def details
return Quickbooks::Model::Customer.new unless id.present?
@details ||= begin
xml = Rails.cache.fetch(details_cache_key, expires_in: 10.minutes) do
fetch_details.to_xml_ns
end end
# Convenience Method Quickbooks::Model::Customer.from_xml(xml)
# returns the customer's email end
end
# Generates a unique cache key for storing this customer's QBO details.
def details_cache_key
"customer:#{id}:qbo_details:#{updated_at.to_i}"
end
# Returns the customer's email address
def email def email
pull unless @details details
begin return @details&.email_address&.address
return @details.email_address.address
rescue
return nil
end
end end
# Convenience Method # Updates the customer's email address
# Sets the email
def email=(s) def email=(s)
pull unless @details details
@details.email_address = s @details.email_address = s
end end
# Convenience Method
# returns the customer's primary phone # Returns the last sync time formatted for display. If no sync has occurred, returns a default message.
def primary_phone def self.last_sync
pull unless @details return I18n.t(:label_qbo_never_synced) unless maximum(:updated_at)
begin format_time(maximum(:updated_at))
return @details.primary_phone.free_form_number end
rescue
return nil # Customers are not bound by a project
# but we need to implement this method for the Redmine::Acts::Searchable interface
def project
nil
end
# Magic Method
# Maps Get/Set methods to QBO customer object
def method_missing(method_name, *args, &block)
if Quickbooks::Model::Customer.method_defined?(method_name)
details
@details.public_send(method_name, *args, &block)
else
super
end end
end end
# Convenience Method
# Updates the customer's primary phone number
def primary_phone=(n)
pull unless @details
pn = Quickbooks::Model::TelephoneNumber.new
pn.free_form_number = n
@details.primary_phone = pn
end
# Convenience Method
# returns the customer's mobile phone # returns the customer's mobile phone
def mobile_phone def mobile_phone
pull unless @details details
begin return @details&.mobile_phone&.free_form_number
return @details.mobile_phone.free_form_number
rescue
return nil
end
end end
# Convenience Method
# Updates the custome's mobile phone number # Updates the custome's mobile phone number
def mobile_phone=(n) def mobile_phone=(n)
pull unless @details details
pn = Quickbooks::Model::TelephoneNumber.new pn = Quickbooks::Model::TelephoneNumber.new
pn.free_form_number = n pn.free_form_number = n
@details.mobile_phone = pn @details.mobile_phone = pn
end end
# Convenience Method
# Updates Both local DB name & QBO display_name # Updates Both local DB name & QBO display_name
def name=(s) def name=(s)
pull unless @details details
@details.display_name = s @details.display_name = s
super super
end end
# Magic Method # Normalizes phone numbers by removing non-digit characters. This method is called before validation to ensure that phone numbers are stored in a consistent format, which can help with searching and integration with external systems like QuickBooks Online.
# Maps Get/Set methods to QBO customer object def normalize_phone_numbers
def method_missing(sym, *arguments) self.phone_number = phone_number.to_s.gsub(/\D/, '') if phone_number.present?
# Check to see if the method exists self.mobile_phone_number = mobile_phone_number.to_s.gsub(/\D/, '') if mobile_phone_number.present?
if Quickbooks::Model::Customer.method_defined?(sym)
# download details if required
pull unless @details
method_name = sym.to_s
# Setter
if method_name[-1, 1] == "="
@details.method(method_name).call(arguments[0])
# Getter
else
return @details.method(method_name).call
end
end
end end
# proforms a bruteforce sync operation # Sets the notes for the customer
# This needs to be simplified def notes=(s)
def self.sync details
service = Qbo.get_base(:customer).service @details.notes = s
# Sync ALL customers if the database is empty
#if count == 0
customers = service.all
#else
# last = Qbo.first.last_sync
# query = "Select Id, DisplayName From Customer"
# query << " Where Metadata.LastUpdatedTime >= '#{last.iso8601}' " if last
# customers = service.query(query)
#end
customers.each do |customer|
qbo_customer = Customer.find_or_create_by(id: customer.id)
if customer.active?
if not qbo_customer.name.eql? customer.display_name
qbo_customer.name = customer.display_name
qbo_customer.id = customer.id
qbo_customer.save_without_push
end
else
if not qbo_customer.new_record?
qbo_customer.delete
end
end
end
end end
# Searchs the database for a customer by name # returns the customer's primary phone
def primary_phone
details
return @details&.primary_phone&.free_form_number
end
# Updates the customer's primary phone number
def primary_phone=(n)
details
pn = Quickbooks::Model::TelephoneNumber.new
pn.free_form_number = n
@details.primary_phone = pn
end
# Repsonds to missing methods by delegating to the QBO customer details object if the method is defined there. This allows for dynamic access to any attributes or methods of the QBO customer without having to explicitly define them in the Customer model, providing flexibility and reducing boilerplate code.
def respond_to_missing?(method_name, include_private = false)
Quickbooks::Model::Customer.method_defined?(method_name) || super
end
# Seach for customers by name or phone number
def self.search(search) def self.search(search)
customers = where("name LIKE ?", "%#{search}%") #return none if search.blank?
search = sanitize_sql_like(search)
#if customers.empty? where("name LIKE ? OR phone_number LIKE ? OR mobile_phone_number LIKE ?", "%#{search}%", "%#{search}%", "%#{search}%")
# service = Qbo.get_base(:customer).service
# results = service.query("Select Id From Customer Where PrimaryPhone LIKE '%#{search}%' AND Mobile LIKE '%#{search}%'")
# results.each do |customer|
# customers << Customer.find_by_id(customer.id)
# end
#end
return customers.order(:name)
end end
# proforms a bruteforce sync operation # Override the defult redmine seach method to rank results by id
# This needs to be simplified def self.search_result_ranks_and_ids(tokens, user, project = nil, options = {})
return {} if tokens.blank?
scope = self.all
tokens.each do |token|
scope = scope.search(token)
end
ids = scope.distinct.limit(options[:limit] || 100).pluck(:id)
ids.index_with { |id| id }
end
# performs a sync operation for all customers
def self.sync
CustomerSyncJob.perform_later(full_sync: false)
end
# performs a sync operation for a specific customer
def self.sync_by_id(id) def self.sync_by_id(id)
service = Qbo.get_base(:customer).service CustomerSyncJob.perform_later(id: id)
end
customer = service.fetch_by_id(id) # returns a human readable string
qbo_customer = Customer.find_or_create_by(id: customer.id) def to_s
if customer.active? last4 = phone_number&.last(4)
if not qbo_customer.name.eql? customer.display_name last4.present? ? "#{name} - #{last4}" : name.to_s
qbo_customer.name = customer.display_name
qbo_customer.id = customer.id
qbo_customer.save_without_push
end
else
if not qbo_customer.new_record?
qbo_customer.delete
end
end
end end
# Push the updates # Push the updates
def save_with_push def save_with_push
begin log "Starting push for customer ##{self.id}..."
@details = Qbo.get_base(:customer).service.update(@details) qbo = QboConnectionService.current!
#raise "QBO Fault" if @details.fault? CustomerService.new(qbo: qbo, customer: self).push()
self.id = @details.id Rails.cache.delete(details_cache_key)
rescue Exception => e
errors.add(e.message)
end
save_without_push save_without_push
end end
@@ -193,14 +190,17 @@ class Customer < ActiveRecord::Base
private private
# pull the details # Fetches the customer's details from QuickBooks Online. If the customer has an ID, it makes an authenticated request to QBO to retrieve the customer's information. If the customer does not have an ID or if there is an error during the fetch, it returns a new instance of Quickbooks::Model::Customer with default values. This method is used to ensure that the customer object has the most up-to-date information from QBO when needed.
def pull def fetch_details
begin return Quickbooks::Model::Customer.new unless id.present?
raise Exception unless self.id log "Fetching details for customer ##{id} from QBO..."
@details = Qbo.get_base(:customer).find_by_id(self.id) qbo = QboConnectionService.current!
rescue Exception => e CustomerService.new(qbo: qbo, customer: self).pull()
@details = Quickbooks::Model::Customer.new
end end
# Log messages with the entity type for better traceability
def log(msg)
Rails.logger.info "[Customer] #{msg}"
end end
end end
+42 -9
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,16 +8,49 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class CustomerToken < ActiveRecord::Base class CustomerToken < ApplicationRecord
unloadable belongs_to :issue
has_many :issues
attr_accessible :token, :expires_at, :issue_id
validates_presence_of :expires_at, :issue_id
before_create :generate_token
OAUTH_CONSUMER_SECRET = Setting.plugin_redmine_qbo['settingsOAuthConsumerSecret'] || 'CONFIGURE_QBO__' + SecureRandom.uuid validates :issue_id, presence: true
validates :token, presence: true, uniqueness: true
before_validation :generate_token, on: :create
before_validation :generate_expire_date, on: :create
scope :active, -> { where("expires_at > ?", Time.current) }
TOKEN_EXPIRATION = 1.month
# Check if the token has expired
def expired?
expires_at.present? && expires_at <= Time.current
end
# Remove expired tokens from the database
def self.remove_expired_tokens
where("expires_at <= ?", Time.current).delete_all
end
# Get or create a token for the given issue
def self.get_token(issue)
return unless issue
return unless User.current.allowed_to?(:view_issues, issue.project)
token = active.find_by(issue_id: issue.id)
return token if token
create!(issue: issue)
end
private
# Generate a unique token for the customer
def generate_token def generate_token
self.token = SecureRandom.base64(15).tr('+/=lIO0', OAUTH_CONSUMER_SECRET) self.token ||= SecureRandom.urlsafe_base64(32)
end
# Generate an expiration date for the token
def generate_expire_date
self.expires_at ||= Time.current + TOKEN_EXPIRATION
end end
end end
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,35 +8,29 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboEmployee < ActiveRecord::Base class Employee < ActiveRecord::Base
unloadable
include Redmine::I18n
has_many :users has_many :users
attr_accessible :name
validates_presence_of :id, :name validates_presence_of :id, :name
def self.get_base self.primary_key = :id
Qbo.get_base(:employee)
# Returns the last sync time formatted for display. If no sync has occurred, returns a default message.
def self.last_sync
return I18n.t(:label_qbo_never_synced) unless maximum(:updated_at)
format_time(maximum(:updated_at))
end end
# Sync all employees, typically triggered by a scheduled task or manual sync request
def self.sync def self.sync
employees = get_base.service.all EmployeeSyncJob.perform_later(full_sync: true)
transaction do
# Update the item table
employees.each { |employee|
qbo_employee = find_or_create_by(id: employee.id)
qbo_employee.name = employee.display_name
qbo_employee.id = employee.id
qbo_employee.save!
}
end
end end
# Sync a single employee by ID, typically triggered by a webhook notification or manual sync request
def self.sync_by_id(id) def self.sync_by_id(id)
employee = get_base.service.fetch_by_id(id) EmployeeSyncJob.perform_later(id: id)
qbo_employee = find_or_create_by(id: employee.id)
qbo_employee.name = employee.display_name
qbo_employee.id = employee.id
qbo_employee.save!
end end
end end
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,40 +8,45 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboItem < ActiveRecord::Base class Estimate < ActiveRecord::Base
unloadable
has_many :issues
attr_accessible :name
validates_presence_of :id, :name
include Redmine::I18n
has_and_belongs_to_many :issues
belongs_to :customer
validates_presence_of :doc_number, :id
self.primary_key = :id self.primary_key = :id
def self.get_base # Returns the last sync time formatted for display. If no sync has occurred, returns a default message.
Qbo.get_base(:item) def self.last_sync
return I18n.t(:label_qbo_never_synced) unless maximum(:updated_at)
format_time(maximum(:updated_at))
end end
# returns a human readable string
def to_s
return self[:doc_number]
end
# sync all estimates
def self.sync def self.sync
last = Qbo.first.last_sync EstimateSyncJob.perform_later(full_sync: false)
query = "SELECT Id, Name FROM Item WHERE Type = 'Service' "
query << " AND Metadata.LastUpdatedTime >= '#{last.iso8601}' " if last
if count == 0
items = get_base.service.all
else
items = get_base.service.query(query)
end end
unless items.count = 0 # sync only one estimate
items.find_by(:type, "Service").each { |i| def self.sync_by_id(id)
qbo_item = QboItem.find_or_create_by(id: i.id) EstimateSyncJob.perform_later(id: id)
qbo_item.name = i.name
qbo_item.id = i.id
qbo_item.save
}
end end
# QboItem.where.not(items.map(&:id)).destroy_all # sync only one estimate
def self.sync_by_doc_number(number)
EstimateSyncJob.perform_later(doc_number: number)
end
private
def log(msg)
Rails.logger.info "[Estimate] #{msg}"
end end
end end
+44
View File
@@ -0,0 +1,44 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class Invoice < ActiveRecord::Base
include Redmine::I18n
has_and_belongs_to_many :issues
belongs_to :customer
validates :id, presence: true, uniqueness: true
validates :doc_number, :txn_date, presence: true
self.primary_key = :id
# Returns the last sync time formatted for display. If no sync has occurred, returns a default message.
def self.last_sync
return I18n.t(:label_qbo_never_synced) unless maximum(:updated_at)
format_time(maximum(:updated_at))
end
# Return the invoice's document number as its string representation
def to_s
doc_number
end
# Sync all invoices from QuickBooks, typically triggered by a scheduled task or manual sync request
def self.sync
InvoiceSyncJob.perform_later(full_sync: true)
end
# Sync a single invoice by ID, typically triggered by a webhook notification or manual sync request
def self.sync_by_id(id)
InvoiceSyncJob.perform_later(id: id)
end
end
+7
View File
@@ -0,0 +1,7 @@
class LineItem < ApplicationRecord
belongs_to :issue
validates :description, presence: true
validates :quantity, numericality: { greater_than: 0 }
validates :unit_price, numericality: { greater_than_or_equal_to: 0 }
end
+25 -37
View File
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -9,50 +9,38 @@
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class Qbo < ActiveRecord::Base class Qbo < ActiveRecord::Base
unloadable
validates_presence_of :qb_token, :qb_secret, :company_id, :token_expires_at, :reconnect_token_at
OAUTH_CONSUMER_KEY = Setting.plugin_redmine_qbo['settingsOAuthConsumerKey'] include QuickbooksOauth
OAUTH_CONSUMER_SECRET = Setting.plugin_redmine_qbo['settingsOAuthConsumerSecret'] include Redmine::I18n
$qb_oauth_consumer = OAuth::Consumer.new(OAUTH_CONSUMER_KEY, OAUTH_CONSUMER_SECRET, { validate :single_record_only, on: :create
:site => "https://oauth.intuit.com",
:request_token_path => "/oauth/v1/get_request_token",
:authorize_url => "https://appcenter.intuit.com/Connect/Begin",
:access_token_path => "/oauth/v1/get_access_token"
})
# Configure quickbooks-ruby-base to access our database
Quickbooks::Base.configure do |c|
c.persistent_token = 'qb_token'
c.persistent_secret = 'qb_secret'
c.persistent_company_id = 'company_id'
end
def self.get_oauth_consumer
# Quickbooks Config Info
return $qb_oauth_consumer
end
# Get a quickbooks base object for type
# @params type of base
def self.get_base(type)
Quickbooks::Base.new(first, type)
end
# Get the QBO account
def self.get_account
first
end
# Updates last sync time stamp # Updates last sync time stamp
def self.update_time_stamp def self.update_time_stamp
qbo = Qbo.first date = DateTime.now
qbo.last_sync = DateTime.now log "Updating QBO timestamp to #{date}"
qbo = QboConnectionService.current!
qbo.last_sync = date
qbo.save qbo.save
end end
# Returns the last sync time formatted for display. If no sync has occurred, returns a default message.
def self.last_sync def self.last_sync
format_time(Qbo.first.last_sync) qbo = QboConnectionService.current!
return I18n.t(:label_qbo_never_synced) unless qbo&.last_sync
format_time(qbo.last_sync)
end end
private
# Logs a message with a QBO-specific prefix for easier identification in the logs.
def self.log(msg)
logger.info "[QBO] #{msg}"
end
# Validates that only one QBO connection record exists in the database. Adds an error if a record already exists.
def single_record_only
errors.add(:base, "Only one QBO connection allowed") if Qbo.exists?
end
end end
-53
View File
@@ -1,53 +0,0 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboEstimate < ActiveRecord::Base
unloadable
has_many :issues
attr_accessible :doc_number
validates_presence_of :id, :doc_number
def self.get_base
Qbo.get_base(:estimate)
end
def self.sync
estimates = get_base.service.all
# Update the item table
transaction do
estimates.each { |estimate|
qbo_estimate = QboEstimate.find_or_create_by(id: estimate.id)
qbo_estimate.doc_number = estimate.doc_number
qbo_estimate.id = estimate.id
qbo_estimate.save!
}
end
#remove deleted estimates
where.not(estimates.map(&:id)).destroy_all
end
def self.sync_by_id(id)
estimate = get_base.service.fetch_by_id(id)
qbo_estimate = QboEstimate.find_or_create_by(id: estimate.id)
qbo_estimate.doc_number = estimate.doc_number
qbo_estimate.id = estimate.id
qbo_estimate.save!
end
def self.update(id)
# Update the item table
estimate = get_base.service.fetch_by_id(id)
qbo_estimate = QboEstimate.find_or_create_by(id: id)
qbo_estimate.doc_number = estimate.doc_number
qbo_estimate.save!
end
end
-154
View File
@@ -1,154 +0,0 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboInvoice < ActiveRecord::Base
unloadable
has_and_belongs_to_many :issues
attr_accessible :doc_number, :id
validates_presence_of :doc_number, :id
self.primary_key = :id
def self.get_base
Qbo.get_base(:invoice).service
end
# sync ALL the invoices
def self.sync
last = Qbo.first.last_sync
query = "SELECT Id, DocNumber FROM Invoice"
query << " WHERE Metadata.LastUpdatedTime >= '#{last.iso8601}' " if last
if count == 0
invoices = get_base.all
else
invoices = get_base.query()
end
# Update the invoice table
invoices.each { | invoice |
process_invoice invoice
}
end
#sync by invoice ID
def self.sync_by_id(id)
#update the information in the database
invoice = get_base.fetch_by_id(id)
process_invoice invoice
end
private
# Attach the invoice to the issue
def self.attach_to_issue(issue, invoice)
return if issue.nil?
# skip this issue if the issue customer is not the same as the invoice customer
return if issue.customer_id != invoice.customer_ref.value.to_i
# Load the invoice into the database
qbo_invoice = QboInvoice.find_or_create_by(id: invoice.id)
qbo_invoice.doc_number = invoice.doc_number
qbo_invoice.id = invoice.id
qbo_invoice.save!
unless issue.qbo_invoices.include?(qbo_invoice)
issue.qbo_invoices << qbo_invoice
issue.save!
end
compare_custom_fields(issue, invoice)
end
# processes the invoice into the system
def self.process_invoice(invoice)
# Check the private notes
if not invoice.private_note.nil?
invoice.private_note.scan(/#(\w+)/).flatten.each { |issue|
attach_to_issue(Issue.find_by_id(issue.to_i), invoice)
}
end
# Scan the line items for hashtags and attach to the applicable issues
invoice.line_items.each { |line|
if line.description
line.description.scan(/#(\w+)/).flatten.each { |issue|
attach_to_issue(Issue.find_by_id(issue.to_i), invoice)
}
end
}
end
def self.compare_custom_fields(issue, invoice)
is_changed = false
# update the invoive custom fields with infomation from the work ticket if available
invoice.custom_fields.each { |cf|
# TODO Add some hooks here
# VIN from the attached vehicle
begin
if cf.name.eql? "VIN"
vin = Vehicle.find(issue.vehicles_id).vin
break if vin.nil?
if not cf.string_value.to_s.eql? vin
cf.string_value = vin.to_s
is_changed = true
end
end
rescue
#do nothing
end
# Custom Values
begin
value = issue.custom_values.find_by(custom_field_id: CustomField.find_by_name(cf.name).id)
# Check to see if the value is blank...
if not value.value.to_s.blank?
# Check to see if the value is diffrent
if not cf.string_value.to_s.eql? value.value.to_s
# Use the lowest Milage
if cf.name.eql? "Mileage In"
if cf.string_value.to_i > value.value.to_i or cf.string_value.blank?
cf.string_value = value.value.to_s
is_changed = true
end
end
# Use the max milage
if cf.name.eql? "Mileage Out"
if cf.string_value.to_i < value.value.to_i or cf.string_value.blank?
cf.string_value = value.value.to_s
is_changed = true
end
end
# Everything else
cf.string_value = value.value.to_s
is_changed = true
end
end
rescue
# Nothing to do here, there is no match
end
}
# TODO Add some hooks here
# Push updates
get_base.update(invoice) if is_changed
end
end
-130
View File
@@ -1,130 +0,0 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class Vehicle < ActiveRecord::Base
unloadable
API_KEY = Setting.plugin_redmine_qbo['settingsEdmundsAPIKey']
belongs_to :customer
has_many :issues, :foreign_key => 'vehicles_id'
attr_accessible :year, :make, :model, :customer_id, :notes, :vin
validates_presence_of :customer
validates :vin, uniqueness: true
#validates :year, numericality: { only_integer: true }
before_save :decode_vin
after_initialize :get_details
self.primary_key = :id
# returns a human readable string
def to_s
return "#{year} #{make} #{model}"
end
# returns the raw JSON details from EMUNDS
def details
return @details
end
# returns the style of the vehicle
def style
begin
return @details['years'][0]['styles'][0]['name'] if @details
rescue
return nil
end
end
# returns the drive of the vehicle i.e. 2 wheel, 4 wheel, ect.
def drive
return @details['drivenWheels'].to_s.upcase if @details
end
# returns the number of doors of the vehicle
def doors
return @details['numOfDoors'] if @details
end
# Force Upper Case for VIN numbers
def make=(val)
# The to_s is in case you get nil/non-string
write_attribute(:make, val.to_s.titleize)
end
# Force Upper Case for VIN numbers
def model=(val)
# The to_s is in case you get nil/non-string
write_attribute(:model, val.to_s.titleize)
end
# Force Upper Case for VIN numbers
def vin=(val)
# The to_s is in case you get nil/non-string
write_attribute(:vin, val.to_s.scan(/^[A-Za-z0-9]+$/).join.upcase)
end
# search for a vin
def self.search(search)
where("vin LIKE ?", "%#{search}%")
end
private
# init method to pull JSON details from Edmunds
def get_details
if self.vin?
begin
@details = JSON.parse get_decoder.full(self.vin)
raise @details['message'] if @details['status'].to_s.eql? "NOT_FOUND"
raise @details['message'] if @details['status'].to_s.eql? "BAD_REQUEST"
rescue Exception => e
errors.add(:vin, e.message)
end
end
end
# returns the Edmunds decoder service
def get_decoder
#TODO API Code via Settings
return decoder = Edmunds::Vin.new(API_KEY)
end
# decodes a vin and updates self
def decode_vin
get_details
if @details
begin
self.year = @details['years'][0]['year']
self.make = @details['make']['name']
self.model = @details['model']['name']
rescue Exception => e
errors.add(:vin, e.message)
end
end
self.name = to_s
end
# makes a squishvin
# https://api.edmunds.com/api/vehicle/v2/squishvins/#{vin}/?fmt=json&api_key=#{ENV['edmunds_key']}
def vin_squish
if not self.vin? or self.vin.size < 11
# this is to go ahead and query the API, letting them handle the error. :P
return '1000000000A'
end
v = self.vin[0,11]
return v.slice(0,8) + v.slice(9,11)
end
end
+62
View File
@@ -0,0 +1,62 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class CustomerService
# Initializes the service with a QBO client and an optional customer record. The QBO client is used to communicate with QuickBooks Online, while the customer record contains the data that needs to be pushed to QBO. If no customer is provided, the service will not perform any operations.
def initialize(qbo:, customer: nil)
raise "No QBO configuration found" unless qbo
raise "Customer record is required for push operation" unless customer
@qbo = qbo
@customer = customer
end
# Pulls the customer data from QuickBooks Online.
def pull
return Quickbooks::Model::Customer.new unless @customer.present?
log "Fetching details for customer ##{@customer.id} from QBO..."
qbo = QboConnectionService.current!
qbo.perform_authenticated_request do |access_token|
service = Quickbooks::Service::Customer.new(
company_id: qbo.realm_id,
access_token: access_token
)
service.fetch_by_id(@customer.id)
end
rescue => e
log "Fetch failed for #{@customer.id}: #{e.message}"
Quickbooks::Model::Customer.new
end
# Pushes the customer data to QuickBooks Online. This method handles the communication with QBO, including authentication and error handling. It uses the QBO client to send the customer data and logs the process for monitoring and debugging purposes. If the push is successful, it returns the customer record; otherwise, it logs the error and returns false.
def push
log "Pushing customer ##{@customer.id} to QBO..."
customer = @qbo.perform_authenticated_request do |access_token|
service = Quickbooks::Service::Customer.new(
company_id: @qbo.realm_id,
access_token: access_token
)
service.update(@customer.details)
end
@customer.id = customer.id unless @customer.persisted?
log "Push for customer ##{@customer.id} completed."
return @customer
end
private
# Log messages with the entity type for better traceability
def log(msg)
Rails.logger.info "[CustomerService] #{msg}"
end
end
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,18 +8,25 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class UsersShowHookListener < Redmine::Hook::ViewListener class CustomerSyncService < SyncServiceBase
# View User private
def view_users_form(context={})
# Update the users # Specify the local model this service syncs
#QboEmployee.update_all def self.model_class
Customer
# Check to see if there is a quickbooks user attached to the issue
@selected = context[:user].qbo_employee.id if context[:user].qbo_employee
# Generate the drop down list of quickbooks contacts
return "<p>#{context[:form].select :qbo_employee_id, QboEmployee.all.pluck(:name, :id), :selected => @selected, include_blank: true}</p>"
end end
# Determine if the remote entity should be deleted locally (e.g. if it's marked inactive in QBO)
def destroy_remote?(remote)
!remote.active?
end
# Map relevant attributes from the QBO Customer to the local Customer model
def process_attributes(local, remote)
local.name = remote.display_name
local.phone_number = remote.primary_phone&.free_form_number&.gsub(/\D/, '')
local.mobile_phone_number = remote.mobile_phone&.free_form_number&.gsub(/\D/, '')
end
end end
+30
View File
@@ -0,0 +1,30 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class EmployeeSyncService < SyncServiceBase
private
# Specify the local model this service syncs
def self.model_class
Employee
end
# Determine if the remote entity should be deleted locally (e.g. if it's marked inactive in QBO)
def destroy_remote?(remote)
!remote.active?
end
# Map relevant attributes from the QBO Employee to the local Employee model
def process_attributes(local, remote)
local.name = remote.display_name
end
end
@@ -1,15 +1,16 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. #The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class EstimatePdfService < PdfServiceBase
class UpdateVehicles < ActiveRecord::Migration def self.model_class
def change Estimate
add_column :vehicles, :name, :text
end end
end end
@@ -1,6 +1,6 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
@@ -8,17 +8,20 @@
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class CreateVehicles < ActiveRecord::Migration class EstimateSyncService < SyncServiceBase
def change private
create_table :vehicles do |t|
t.integer :year # Specify the local model this service syncs
t.string :make def self.model_class
t.string :model Estimate
t.string :vin
t.text :notes
end end
add_reference :vehicles, :qbo_customer, index: true # Map relevant attributes from the QBO Estimate to the local Estimate model
def process_attributes(local, remote)
local.doc_number = remote.doc_number
local.txn_date = remote.txn_date
local.customer = Customer.find_by(id: remote.customer_ref&.value)
end end
end end
@@ -0,0 +1,62 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class InvoiceAttachmentService
def initialize(invoice, remote)
@invoice = invoice
@remote = remote
end
# Attach invoice to issues based on issue IDs found in the invoice's private note and line descriptions
def attach
extract_issue_ids.each do |issue_id|
log "Processing issue ##{issue_id} for invoice ##{@invoice.doc_number}"
issue = Issue.find_by(id: issue_id)
next unless issue
next unless issue.customer&.id == @invoice.customer&.id
unless issue.invoices.exists?(@invoice.id)
issue.invoices << @invoice
issue.save! if issue.changed?
log "Attached invoice ##{@invoice.id} to issue ##{issue.id}"
end
InvoiceCustomFieldSyncService.new(issue, @invoice, @remote).sync
end
end
private
# Extract issue IDs from the invoice's private note and line descriptions
def extract_issue_ids
ids = []
if @remote.private_note.present?
ids += scan(@remote.private_note)
end
Array(@remote.line_items).each do |line|
ids += scan(line.description.to_s)
end
ids.uniq
end
# Scan text for issue IDs in the format #123
def scan(text)
text.scan(/#(\d+)/).flatten.map(&:to_i)
end
def log(msg)
Rails.logger.info "[InvoiceAttachmentService] #{msg}"
end
end
@@ -0,0 +1,69 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class InvoiceCustomFieldSyncService
def initialize(issue, invoice, remote)
@issue = issue
@invoice = invoice
@remote = remote
end
# Sync custom fields on the issue based on the invoice data, then push changes to QBO if any fields were updated
def sync
return if @invoice.qbo_sync_locked?
log "Syncing custom fields for issue ##{@issue.id} based on invoice ##{@invoice.doc_number}"
changed = false
# Process Invoice Custom Fields via Hooks
Redmine::Hook.call_hook(
:process_invoice_custom_fields,
issue: @issue,
invoice: @remote
).each do |context|
next unless context
changed ||= context[:is_changed]
log "Custom fields updated by hook, marking invoice for push to QBO" if context[:is_changed]
end
# Process Issue Custom Values from any issue custom fields that match the invoice custom fields
begin
value = @issue.custom_values.find_by(custom_field_id: CustomField.find_by_name(cf.name).id)
# Check to see if the value is blank...
if not value.value.to_s.blank?
# Check to see if the value is diffrent
if not cf.string_value.to_s.eql? value.value.to_s
# update the custom field on the invoice
cf.string_value = value.value.to_s
is_changed = true
end
end
rescue
# Nothing to do here, there is no match
end
push_if_changed if changed
end
private
# If any custom fields were changed during the sync process, this method will trigger a push of the invoice data to QuickBooks Online to ensure that the remote data stays in sync with the local changes. It uses the InvoicePushService to handle the actual communication with QBO.
def push_if_changed
InvoicePushService.new(@invoice).push
end
def log(msg)
Rails.logger.info "[InvoiceCustomFieldSyncService] #{msg}"
end
end
+16
View File
@@ -0,0 +1,16 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class InvoicePdfService < PdfServiceBase
def self.model_class
Invoice
end
end
@@ -1,51 +1,47 @@
#The MIT License (MIT) #The MIT License (MIT)
# #
#Copyright (c) 2016 rick barrette #Copyright (c) 2016 - 2026 rick barrette
# #
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: #Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
# #
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. #The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
# #
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. #THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class PaymentsController < ApplicationController
unloadable
include AuthHelper class InvoicePushService
before_filter :require_user def initialize(invoice)
@invoice = invoice
def new
@payment = Payment.new
@customers = Customer.all.sort_by &:name
@accounts = Qbo.get_base(:account).service.query("SELECT Id, Name FROM Account WHERE AccountType = 'Bank' Order By Name")
@payment_methods = Qbo.get_base(:payment_method).service.all
end end
def create # Push invoice changes to QBO if the invoice is linked to any issues with custom field changes that need to be synced
@payment = Payment.new(params[:payment]) def push
if @payment.save return if @invoice.qbo_sync_locked?
flash[:notice] = "Payment Saved"
redirect_to Customer.find_by_id(@payment.customer_id) log "Pushing invoice ##{@invoice.id} to QBO due to linked issue custom field changes"
else
flash[:error] = @payment.errors.full_messages.to_sentence @invoice.update_column(:qbo_sync_locked, true)
redirect_to new_customer_path
qbo = QboConnectionService.current!
qbo.perform_authenticated_request do |access_token|
service = Quickbooks::Service::Invoice.new( company_id: qbo.realm_id, access_token: access_token)
remote = service.fetch_by_id(@invoice.id)
# modify remote object here if needed
service.update(remote)
end end
rescue => e
Rails.logger.error "[InvoicePushService] #{e.message}"
ensure
@invoice.update_column(:qbo_sync_locked, false)
end end
private private
def only_one_non_zero?( array ) def log(msg)
found_non_zero = false Rails.logger.info "[InvoicePushService] #{msg}"
array.each do |val|
if val!=0
return false if found_non_zero
found_non_zero = true
end end
end end
found_non_zero
end
end
+35
View File
@@ -0,0 +1,35 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class InvoiceSyncService < SyncServiceBase
private
# Specify the local model this service syncs
def self.model_class
Invoice
end
# Map relevant attributes from the QBO Invoice to the local Invoice model
def process_attributes(local, remote)
local.doc_number = remote.doc_number
local.txn_date = remote.txn_date
local.due_date = remote.due_date
local.total_amount = remote.total
local.balance = remote.balance
local.qbo_updated_at = remote.meta_data&.last_updated_time
local.customer = Customer.find_by(id: remote.customer_ref&.value)
end
# Attach QBO Invoices to the local Issues
def attach_documents(local, remote)
InvoiceAttachmentService.new(local, remote).attach
end
end
+66
View File
@@ -0,0 +1,66 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class PdfServiceBase
require 'combine_pdf'
# Subclasses should initialize with a QBO client instance
def initialize(qbo:)
@qbo = qbo
@entity = self.class.model_class
end
# Subclasses must implement this to specify which document model to download pdf (e.g. Estimate, Invoice)
def self.model_class
raise NotImplementedError
end
# Fetches the PDF for the given entity IDs. If multiple IDs are provided, their PDFs are combined into a single document.
def fetch_pdf(doc_ids:)
log "Fetching PDFs for #{@entity} IDs: #{doc_ids.join(', ')}"
@qbo.perform_authenticated_request do |access_token|
service_class = "Quickbooks::Service::#{@entity.name}".constantize
service = service_class.new(company_id: @qbo.realm_id, access_token: access_token)
return single_pdf(service, doc_ids.first) if doc_ids.size == 1
combined_pdf(service, doc_ids)
end
end
private
# Fetches a single PDF for the given invoice ID.
def single_pdf(service, id)
log "Fetching PDF for #{@entity} ID: #{id}"
entity = service.fetch_by_id(id)
[service.pdf(entity), entity.doc_number]
end
# Combines PDFs for multiple entity IDs into a single PDF document and returns it along with a reference string.
def combined_pdf(service, ids)
log "Combining PDFs for #{@entity} IDs: #{ids.join(', ')}"
pdf = CombinePDF.new
ref = []
ids.each do |id|
entity = service.fetch_by_id(id)
ref << entity.doc_number
pdf << CombinePDF.parse(service.pdf(entity))
end
[pdf.to_pdf, ref.join(" ")]
end
# Logs messages with a consistent prefix for easier debugging.
def log(msg)
Rails.logger.info "[#{@entity}PdfService] #{msg}"
end
end
+32
View File
@@ -0,0 +1,32 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboConnectionService
# Replaces the existing QBO connection with new credentials. Deletes all existing records and creates a new one with the provided token, refresh token, and realm ID. Refreshes the token immediately after creation.
def self.replace!(token:, refresh_token:, realm_id:)
Qbo.transaction do
Qbo.destroy_all
qbo = Qbo.create!(
oauth2_access_token: token,
oauth2_refresh_token: refresh_token,
realm_id: realm_id
)
qbo.refresh_token!
qbo
end
end
# Returns the current QBO connection record. Raises an error if no connection exists.
def self.current!
Qbo.first || raise("QBO not connected")
end
end
+33
View File
@@ -0,0 +1,33 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class QboOauthService
# Generates the QuickBooks OAuth authorization URL with the specified callback URL. The URL includes necessary parameters such as response type, state, and scope.
def self.authorization_url(callback_url:)
client.auth_code.authorize_url(
redirect_uri: callback_url,
response_type: "code",
state: SecureRandom.hex(12),
scope: "com.intuit.quickbooks.accounting"
)
end
# Exchanges the authorization code for access and refresh tokens. Creates or replaces the QBO connection record with the new credentials and refreshes the token immediately after creation.
def self.exchange!(code:, callback_url:, realm_id:)
resp = client.auth_code.get_token(code, redirect_uri: callback_url)
QboConnectionService.replace!( token: resp.token, refresh_token: resp.refresh_token, realm_id: realm_id )
end
# Constructs and returns an OAuth2 client instance configured with the application's credentials and settings.
def self.client
Qbo.construct_oauth2_client
end
end
+127
View File
@@ -0,0 +1,127 @@
#The MIT License (MIT)
#
#Copyright (c) 2016 - 2026 rick barrette
#
#Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
#
#The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
#
#THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
class SyncServiceBase
PAGE_SIZE = 1000
# Subclasses should initialize with a QBO client instance
def initialize(qbo:)
raise "No QBO configuration found" unless qbo
@qbo = qbo
@entity = self.class.model_class
end
# Subclasses must implement this to specify which local model they sync (e.g. Customer, Invoice)
def self.model_class
raise NotImplementedError
end
# Sync all entities, or only those updated since the last sync
def sync(full_sync: false)
log "Starting #{full_sync ? 'full' : 'incremental'} #{@entity.name} sync"
@qbo.perform_authenticated_request do |access_token|
service_class = "Quickbooks::Service::#{@entity.name}".constantize
service = service_class.new(company_id: @qbo.realm_id, access_token: access_token)
page = 1
loop do
collection = fetch_page(service, page, full_sync)
entries = Array(collection&.entries)
break if entries.empty?
entries.each { |remote| persist(remote) }
break if entries.size < PAGE_SIZE
page += 1
end
end
log "#{@entity.name} sync complete"
end
# Sync a single entity by its QBO ID, used for webhook updates
def sync_by_id(id)
log "Syncing #{@entity.name} with ID #{id}"
@qbo.perform_authenticated_request do |access_token|
service_class = "Quickbooks::Service::#{@entity.name}".constantize
service = service_class.new(company_id: @qbo.realm_id, access_token: access_token)
remote = service.fetch_by_id(id)
persist(remote)
end
end
private
def attach_documents(local, remote)
# Override in subclasses if the entity has attachments (e.g. Invoice)
end
# Determine if a remote entity should be deleted locally (e.g. if it's marked inactive in QBO)
def destroy_remote?(remote)
false
end
# Log messages with the entity type for better traceability
def log(msg)
Rails.logger.info "[#{@entity.name}SyncService] #{msg}"
end
# Fetch a page of entities, either all or only those updated since the last sync
def fetch_page(service, page, full_sync)
log "Fetching page #{page} of #{@entity.name} from QBO (#{full_sync ? 'full' : 'incremental'} sync)"
start_position = (page - 1) * PAGE_SIZE + 1
if full_sync
service.query("SELECT * FROM #{@entity.name} STARTPOSITION #{start_position} MAXRESULTS #{PAGE_SIZE}")
else
last_update = @entity.maximum(:updated_at) || 1.year.ago
service.query(<<~SQL.squish)
SELECT * FROM #{@entity.name}
WHERE MetaData.LastUpdatedTime > '#{last_update.utc.iso8601}'
STARTPOSITION #{start_position}
MAXRESULTS #{PAGE_SIZE}
SQL
end
end
# Create or update a local entity record based on the QBO remote data
def persist(remote)
local = @entity.find_or_initialize_by(id: remote.id)
if destroy_remote?(remote)
if local.persisted?
local.destroy
log "Deleted #{@entity.name} #{remote.id}"
end
return
end
# Map remote attributes to local model fields, this should be implemented in subclasses
process_attributes(local, remote)
if local.changed?
local.save!
log "Updated #{@entity.name} #{remote.id}"
# Handle attaching documents if applicable to invoices
attach_documents(local, remote)
end
rescue => e
log "Failed to sync #{@entity.name} #{remote.id}: #{e.message}"
end
# This method should be implemented in subclasses to map remote attributes to local model
def process_attributes(local, remote)
raise NotImplementedError, "Subclasses must implement process_attributes"
end
end
+11
View File
@@ -0,0 +1,11 @@
<%= link_to t(:label_appointment), "https://calendar.google.com/calendar/render?action=TEMPLATE&text=#{@customer.name}+-&details=#{ link_to t(:customer_details), "https://#{Setting.host_name}#{customer_path @customer.id}"}%0A#{@customer.primary_phone}%3Cbr/%3E+&dates=#{Time.now.strftime("%Y%m%d")}T090000/#{Time.now.strftime("%Y%m%d")}T170000", target: :_blank, id: :appointment_link %>
<br/>
<br/>
<%= link_to t(:label_create_estimate), "https://qbo.intuit.com/app/estimate?nameId=#{@customer.id}", target: :_blank %>
<br/>
<br/>
<%= button_to t(:label_edit_customer), edit_customer_path(@customer), method: :get%>
+30 -24
View File
@@ -1,54 +1,60 @@
<table> <table>
<tbody> <tbody>
<tr> <tr>
<th>Email</th> <th><%=t(:label_name)%></th>
<td><%= customer.name %></td>
</tr>
<tr>
<th><%=t(:label_email)%></th>
<td><%= customer.email %></td> <td><%= customer.email %></td>
</tr> </tr>
<tr> <tr>
<th>Primary Phone</th> <th><%=t(:label_primary_phone)%></th>
<td><%= number_to_phone(customer.primary_phone.gsub(/[^\d]/, '').to_i, area_code: true) if customer.primary_phone %></td> <td><%= number_to_phone(customer&.primary_phone&.gsub(/[^\d]/, '').to_i, area_code: true) %></td>
</tr> </tr>
<tr> <tr>
<th>Mobile Phone</th> <th><%=t(:label_mobile_phone)%></th>
<td><%= number_to_phone(customer.mobile_phone.gsub(/[^\d]/, '').to_i, area_code: true) if customer.mobile_phone %></td> <td><%= number_to_phone(customer&.mobile_phone&.gsub(/[^\d]/, '').to_i, area_code: true) %></td>
</tr> </tr>
<tr> <tr>
<th>Bill Address</th> <th><%=t(:label_billing_address)%></th>
<td><%= customer.billing_address %></td> <td><pre><%= @billing_address %></pre></td>
</tr> </tr>
<tr> <tr>
<th>Shipping Address</th> <th><%=t(:label_shipping_address)%></th>
<td><%= customer.shipping_address %></td> <td><pre><%= @shipping_address %></pre></td>
</tr> </tr>
<tr> <tr>
<th>Issues</th> <th><%=t(:label_account_balance)%></th>
<td><%= customer.issues.count %></td>
</tr>
<tr>
<th>Account Balance</th>
<td>$<%= customer.balance %></td> <td>$<%= customer.balance %></td>
</tr> </tr>
<tr> <tr>
<th>Balance With Jobs</th> <th colspan="2"><h4><%=t(:field_notes)%></hr></th>
<td>$<%= customer.balance_with_jobs %></td>
</tr> </tr>
<tr> <tr>
<th>Notes</th> <td colspan="2">
<td><%= customer.notes %></td> <pre id="note-display" style="text-align: left; white-space: pre-wrap; font-family: inherit;">
</tr> <%= customer.notes %>
</pre>
<tr>
<td>
<%= button_to "Edit Customer", edit_customer_path(customer), method: :get%>
</td> </td>
</tr> </tr>
<script>
const preElement = document.getElementById('note-display');
// This takes the text, trims the edges, and puts it back
preElement.textContent = preElement.textContent.trim();
</script>
</tbody> </tbody>
</table> </table>
<br/>
<br/>
+16 -17
View File
@@ -5,48 +5,47 @@
<%= form_for @customer do |f| %> <%= form_for @customer do |f| %>
<div class="clearfix"> <div class="clearfix">
Display Name: <%=t(:label_display_name)%>
<div class="input"> <div class="input">
<%= f.text_field :name, :required => true %> <%= f.text_field :name, required: true, autocomplete: "off" %>
</div> </div>
</div> </div>
<div class="clearfix"> <div class="clearfix">
Phone Number: <%=t(:label_primary_phone)%>
<div class="input"> <div class="input">
<%= f.telephone_field :primary_phone %> <%= f.telephone_field :primary_phone, autocomplete: "off" %>
</div> </div>
</div> </div>
<div class="clearfix"> <div class="clearfix">
Mobile Phone Number: <%=t(:label_mobile_phone)%>:
<div class="input"> <div class="input">
<%= f.telephone_field :mobile_phone %> <%= f.telephone_field :mobile_phone, autocomplete: "off" %>
</div> </div>
</div> </div>
<div class="clearfix"> <div class="clearfix">
Email: <%=t(:label_email)%>:
<div class="input"> <div class="input">
<%= f.email_field :email %> <%= f.email_field :email, autocomplete: "off" %>
</div> </div>
</div> </div>
<div class="clearfix"> <div class="clearfix">
Notes: <%=t(:field_notes)%>:
<div class="input"> <div class="input">
<p> <p>
<%= link_to_function content_tag(:span, l(:button_edit), :class => 'icon icon-edit'), '$(this).hide(); $("#issue_description_and_toolbar").show()' unless @customer.new_record? %> <%= content_tag :span, id: "issue_description_and_toolbar" do %>
<%= content_tag 'span', :id => "issue_description_and_toolbar", :style => (@customer.new_record? ? nil : 'display:none') do %>
<%= f.text_area :notes, <%= f.text_area :notes,
:cols => 60, cols: 60,
:rows => 10, rows: 10,
:accesskey => accesskey(:edit), accesskey: accesskey(:edit),
:class => 'wiki-edit', class: 'wiki-edit',
:no_label => true %> no_label: true %>
<% end %> <% end %>
</p> </p>
<%= wikitoolbar_for 'issue_description' %> <%= wikitoolbar_for :issue_description %>
</div> </div>
</div> </div>
+5
View File
@@ -0,0 +1,5 @@
<%= form_tag(customers_path, method: "get", id: "customer-search-form") do %>
<%= text_field_tag :search, params[:search], placeholder: t(:label_search_customers), autocomplete: "off" %>
<%= submit_tag t(:label_search) %>
<% end %>
<%= button_to t(:label_new_customer), new_customer_path, method: :get%>
+2
View File
@@ -0,0 +1,2 @@
<h3><%=t(:label_customers)%></h3>
<%= render partial: 'customers/search' %>
+2 -2
View File
@@ -1,3 +1,3 @@
<h1>Edit Customer</h1> <h1><%=t(:label_edit_customer)%></h1>
<br/> <br/>
<%= render :partial => 'customers/form' %> <%= render partial: 'customers/form' %>
@@ -0,0 +1 @@
$('select#issue_estimate_id').html('<%= j content_tag(:option,'',value:"")+options_from_collection_for_select(@filtered_estimates, :id, :doc_number) %>');
@@ -1 +0,0 @@
$('select#issue_vehicles_id').html('<%= j options_from_collection_for_select(@filtered_vehicles, :id, :to_s) %>');
+5 -9
View File
@@ -1,10 +1,4 @@
<h1>Customers</h1> <h2><%=t(:field_customers)%> <span style="float:right"> <%= render partial: 'customers/search' %> </span> </h2>
<br/>
<%= form_tag(customers_path, :method => "get", id: "search-form") do %>
<%= text_field_tag :search, params[:search], placeholder: "Search Customers" %>
<%= submit_tag "Search" %>
<% end %>
<br/>
<% if @customers.present? %> <% if @customers.present? %>
<br/> <br/>
<% @customers.each do |c| %> <% @customers.each do |c| %>
@@ -15,14 +9,16 @@
</div> </div>
<% end %> <% end %>
<p><%=t(:label_matching)%> <%= @customers.count %> <%=t(:field_customers)%> </p>
<div class="actions"> <div class="actions">
<%= will_paginate @customers %> <%= will_paginate @customers %>
</div> </div>
<% else %> <% else %>
<p>There are no customers containing the term(s) <%= params[:search] %>.</p> <p><%=t(:label_no_customers)%> <%= params[:search] %>.</p>
<% end %> <% end %>
<div> <div>
<%= Customer.count %> Customers - <b>Last Sync: </b> <%= Qbo.last_sync if Qbo.exists? %> <%= render partial: 'qbo/stats' %>
</div> </div>
+2 -2
View File
@@ -1,3 +1,3 @@
<h1>New Customer</h1> <h2><%=t(:label_new_customer)%></h2>
<br/> <br/>
<%= render :partial => 'customers/form' %> <%= render partial: 'customers/form' %>
+42 -16
View File
@@ -1,27 +1,53 @@
<div id="content"> <h2><%=t(:field_customer)%> #<%= @customer.id %> - <%= link_to @customer.to_s, "https://#{Setting.plugin_redmine_qbo[:sandbox] ? "sandbox" : "app"}.qbo.intuit.com/app/customerdetail?nameId=#{@customer.id}", target: :_blank %> </h2>
<h2>Customer #<%= @customer.id %></h2> <div class="issue">
<br/>
<div class="subject"> <div class="splitcontent">
<div><h3><%= @customer.name %></h3></div>
</div>
<div class="attributes"> <div class="splitcontentleft">
<h4><%=t(:label_details)%>:</h4>
<!-- Customer Info -->
<div class="splitcontent"> <div class="splitcontent">
<div class="splitcontentleft"> <div class="splitcontentleft">
<h4>Details:</h4> <h4><%=t(:field_customer)%>:</h4>
<%= render :partial => 'customers/details', locals: {customer: @customer} %> <%= render partial: 'customers/details', locals: {customer: @customer} %>
</div> </div>
<div class="splitcontentleft"> <div class="splitcontentleft">
<h4>Vehicles:</h4> <h4><%=t(:label_actions)%>:</h4>
<%= render :partial => 'vehicles/list' %> <%= render partial: 'customers/actions', locals: {customer: @customer} %>
<%= button_to "New Vehicle", new_customer_vehicle_path(@customer), method: :get %>
</div> </div>
</div> </div>
<!-- QBO Info -->
<div class="splitcontent">
<div class="splitcontentleft">
<h4><%=t(:estimates)%>:</h4>
<%= render partial: 'estimates/list', locals: {estimates: @customer.estimates} %>
</div>
<div class="splitcontentleft">
<h4><%=t(:label_invoices)%>:</h4>
<%= render partial: 'invoices/list', locals: {invoices: @customer.invoices} %>
</div>
</div>
</div>
<div class="splitcontentleft">
<%= call_hook :show_customer_view_right, {customer: @customer} %>
</div>
</div>
</div>
<br/> <br/>
<h2>Issues:</h2> <h3><%=@open_issues.count%> <%=t(:label_open_issues)%> - <%=@hours.round(1)%> <%=t(:label_hours)%></h3>
<%= render :partial => 'issues/list_simple', locals: {issues: @issues} %> <%= render partial: 'issues/list_simple', locals: {issues: @open_issues.open} %>
</div>
</div> <h3><%=@closed_issues.count%> <%=t(:label_closed_issues)%> - <%= @closed_hours.round(1)%> <%=t(:label_hours)%></h3>
<%= render partial: 'issues/list_simple', locals: {issues: @closed_issues} %>
+31 -29
View File
@@ -1,12 +1,14 @@
<p style="float: right;"> <%= copy_object_url_link(request.url) %> </p>
<h2><%= issue_heading(@issue) %></h2> <h2><%= issue_heading(@issue) %></h2>
<div class="<%= @issue.css_classes %> details"> <div class="<%= @issue.css_classes %> details">
<%= avatar(@issue.author, :size => "50") %> <%= avatar(@issue.author, size: "50") %>
<div class="subject"> <div class="subject">
<%= render_issue_subject_with_tree(@issue) %> <%= render_issue_subject_with_tree(@issue) %>
This customer link expires in <%= distance_of_time_in_words(Time.now, @token.expires_at) %> <%=t(:label_customer_link_expires)%> <%= distance_of_time_in_words(Time.now, @token.expires_at) %>
</div> </div>
<p class="author"> <p class="author">
<%= authoring @issue.created_on, @issue.author %>. <%= authoring @issue.created_on, @issue.author %>.
@@ -17,39 +19,39 @@ This customer link expires in <%= distance_of_time_in_words(Time.now, @token.exp
<div class="attributes"> <div class="attributes">
<%= issue_fields_rows do |rows| <%= issue_fields_rows do |rows|
rows.left l(:field_status), @issue.status.name, :class => 'status' rows.left l(:field_status), @issue.status.name, class: :status
rows.left l(:field_priority), @issue.priority.name, :class => 'priority' rows.left l(:field_priority), @issue.priority.name, class: :priority
unless @issue.disabled_core_fields.include?('assigned_to_id') # unless @issue.disabled_core_fields.include?(:assigned_to_id)
rows.left l(:field_assigned_to), avatar(@issue.assigned_to, :size => "14").to_s.html_safe + (@issue.assigned_to ? link_to_user(@issue.assigned_to) : "-"), :class => 'assigned-to' # rows.left l(:field_assigned_to), avatar(@issue.assigned_to, size: "14").to_s.html_safe + (@issue.assigned_to ? @issue.assigned_to : "-"), class: 'assigned-to'
# end
unless @issue.disabled_core_fields.include?(:category_id) || (@issue.category.nil? && @issue.project.issue_categories.none?)
rows.left l(:field_category), (@issue.category ? @issue.category.name : "-"), class: :category
end end
unless @issue.disabled_core_fields.include?('category_id') || (@issue.category.nil? && @issue.project.issue_categories.none?) unless @issue.disabled_core_fields.include?(:fixed_version_id) || (@issue.fixed_version.nil? && @issue.assignable_versions.none?)
rows.left l(:field_category), (@issue.category ? @issue.category.name : "-"), :class => 'category' rows.left l(:field_fixed_version), (@issue.fixed_version ? @issue.fixed_version : "-"), class: 'fixed-version'
end end
unless @issue.disabled_core_fields.include?('fixed_version_id') || (@issue.fixed_version.nil? && @issue.assignable_versions.none?) unless @issue.disabled_core_fields.include?(:start_date)
rows.left l(:field_fixed_version), (@issue.fixed_version ? link_to_version(@issue.fixed_version) : "-"), :class => 'fixed-version' rows.right l(:field_start_date), format_date(@issue.start_date), class: 'start-date'
end end
unless @issue.disabled_core_fields.include?('start_date') unless @issue.disabled_core_fields.include?(:due_date)
rows.right l(:field_start_date), format_date(@issue.start_date), :class => 'start-date' rows.right l(:field_due_date), format_date(@issue.due_date), class: 'due-date'
end end
unless @issue.disabled_core_fields.include?('due_date') unless @issue.disabled_core_fields.include?(:done_ratio)
rows.right l(:field_due_date), format_date(@issue.due_date), :class => 'due-date' rows.right l(:field_done_ratio), progress_bar(@issue.done_ratio, legend: "#{@issue.done_ratio}%"), class: :progress
end end
unless @issue.disabled_core_fields.include?('done_ratio') unless @issue.disabled_core_fields.include?(:estimated_hours)
rows.right l(:field_done_ratio), progress_bar(@issue.done_ratio, :legend => "#{@issue.done_ratio}%"), :class => 'progress'
end
unless @issue.disabled_core_fields.include?('estimated_hours')
if @issue.estimated_hours.present? || @issue.total_estimated_hours.to_f > 0 if @issue.estimated_hours.present? || @issue.total_estimated_hours.to_f > 0
rows.right l(:field_estimated_hours), issue_estimated_hours_details(@issue), :class => 'estimated-hours' rows.right l(:field_estimated_hours), issue_estimated_hours_details(@issue), class: 'estimated-hours'
end end
end end
#if User.current.allowed_to_view_all_time_entries?(@project) #if User.current.allowed_to_view_all_time_entries?(@project)
if @issue.total_spent_hours > 0 if @issue.total_spent_hours > 0
rows.right l(:label_spent_time), issue_spent_hours_details(@issue), :class => 'spent-time' rows.right l(:label_spent_time), issue_spent_hours_details(@issue), class: 'spent-time'
end end
#end #end
end %> end %>
<%= render_custom_fields_rows(@issue) %> <%= render_full_width_custom_fields_rows(@issue) %>
<%= call_hook(:view_issues_show_details_bottom, :issue => @issue) %> <%= call_hook(:view_issues_show_details_bottom, issue: @issue) %>
</div> </div>
<% if @issue.description? || @issue.attachments.any? -%> <% if @issue.description? || @issue.attachments.any? -%>
@@ -57,19 +59,19 @@ end %>
<% if @issue.description? %> <% if @issue.description? %>
<div class="description"> <div class="description">
<div class="contextual"> <div class="contextual">
<%= link_to l(:button_quote), quoted_issue_path(@issue), :remote => true, :method => 'post', :class => 'icon icon-comment' if @issue.notes_addable? %> <%= link_to l(:button_quote), quoted_issue_path(@issue), remote: true, method: :post, class: 'icon icon-comment' if @issue.notes_addable? %>
</div> </div>
<p><strong><%=l(:field_description)%></strong></p> <p><strong><%=l(:field_description)%></strong></p>
<div class="wiki"> <div class="wiki">
<%= textilizable @issue, :description, :attachments => @issue.attachments %> <%= textilizable @issue, :description, attachments: @issue.attachments %>
</div> </div>
</div> </div>
<% end %> <% end %>
<%= link_to_attachments @issue, :thumbnails => true %> <%= link_to_attachments @issue, thumbnails: true %>
<% end -%> <% end -%>
<%= call_hook(:view_issues_show_description_bottom, :issue => @issue) %> <%= call_hook(:view_issues_show_description_bottom, issue: @issue) %>
<% if !@issue.leaf? || User.current.allowed_to?(:manage_subtasks, @project) %> <% if !@issue.leaf? || User.current.allowed_to?(:manage_subtasks, @project) %>
<hr /> <hr />
@@ -85,7 +87,7 @@ end %>
<% if @relations.present? || User.current.allowed_to?(:manage_issue_relations, @project) %> <% if @relations.present? || User.current.allowed_to?(:manage_issue_relations, @project) %>
<hr /> <hr />
<div id="relations"> <div id="relations">
<%= render :partial => 'issues/relations' %> <%= render partial: 'issues/relations' %>
</div> </div>
<% end %> <% end %>
@@ -94,14 +96,14 @@ end %>
<% if @changesets.present? %> <% if @changesets.present? %>
<div id="issue-changesets"> <div id="issue-changesets">
<h3><%=l(:label_associated_revisions)%></h3> <h3><%=l(:label_associated_revisions)%></h3>
<%= render :partial => 'issues/changesets', :locals => { :changesets => @changesets} %> <%= render partial: 'issues/changesets', locals: { changesets: @changesets} %>
</div> </div>
<% end %> <% end %>
<% if @journals.present? %> <% if @journals.present? %>
<div id="history"> <div id="history">
<h3><%=l(:label_history)%></h3> <h3><%=l(:label_history)%></h3>
<%= render :partial => 'issues/history', :locals => { :issue => @issue, :journals => @journals } %> <%= render partial: 'issues/history', locals: { issue: @issue, journals: @journals } %>
</div> </div>
<% end %> <% end %>
+12
View File
@@ -0,0 +1,12 @@
<% unless estimates.empty? %>
<% estimates.sort.reverse.each do |estimate| %>
<div class="row">
<%= check_box_tag "estimate_ids[]", estimate.id, false, onchange: "updateLink()", data: { url: estimate_path(estimate), text: "Estimate ##{estimate.to_s}" }, class: "estimate-checkbox appointment" %>
<b><%= link_to "##{estimate.doc_number}", estimate_path(estimate), target: :_blank %></b> <%= estimate.txn_date %>
</div>
<% end %>
<% else %>
<p><%=t(:label_no_estimates)%>.</p>
<% end %>
+4
View File
@@ -0,0 +1,4 @@
<%= form_tag(estimate_doc_path, method: "get", id: "estimate-search-form") do %>
<%= text_field_tag :search, params[:search], placeholder: t(:label_search_estimates), autocomplete: "off" %>
<%= submit_tag t(:label_search), formtarget: "_blank" %>
<% end %>
+2
View File
@@ -0,0 +1,2 @@
<h3><%=t(:label_estimates) %></h3>
<%= render partial: 'estimates/search' %>
+27
View File
@@ -0,0 +1,27 @@
<% unless invoices.empty? %>
<%= form_with(url: invoice_path, method: :get) do |form| %>
<% if invoices.count > 1 %>
<div class="form-check">
<%= check_box_tag "select-all-invoices", "1", false, id: "select-all-invoices" %>
<%= label_tag "select-all-invoices", t(:label_select_all) %>
</div>
<hr>
<% end %>
<% invoices.sort.reverse.each do |invoice| %>
<div class="row">
<%= check_box_tag "invoice_ids[]", invoice.id, false, onchange: "updateLink()", data: { url: invoice_path(invoice), text: "Invoice ##{invoice.to_s}" }, class: "invoice-checkbox appointment" if invoices.count > 1 %>
<b><%= link_to "##{invoice.doc_number}", invoice_path(invoice), target: :_blank %></b> <%= invoice.txn_date %>
</div>
<% end %>
<% if invoices.count > 1 %>
<%= form.submit t(:button_bulk_pdf) %>
<% end %>
<% end %>
<% else %>
<p><%=t(:label_no_invoices)%>.</p>
<% end %>
+12
View File
@@ -0,0 +1,12 @@
<p>
<label for="issue_customer"><%= t(:customer) %></label>
<%= search_customer %>
<%= customer_id %>
</p>
<p>
<%= select_estimate %>
</p>
<%= render "line_items/issue_form", f: f %>
+35
View File
@@ -0,0 +1,35 @@
<% reply_links = issue.notes_addable? -%>
<% for journal in journals %>
<div id="change-<%= journal.id %>" class="<%= journal.css_classes %>">
<div id="note-<%= journal.indice %>">
<div class="contextual">
<span class="journal-actions"><%= render_journal_actions(issue, journal, reply_links: reply_links) %></span>
<a href="#note-<%= journal.indice %>" class="journal-link">#<%= journal.indice %></a>
</div>
<h4>
<%= avatar(journal.user, size: "24") %>
<%= authoring journal.created_on, journal.user, label: :label_updated_time_by %>
<%= render_private_notes_indicator(journal) %>
</h4>
<% if journal.details.any? %>
<ul class="details">
<% details_to_strings(journal.visible_details).each do |string| %>
<li><%= string %></li>
<% end %>
</ul>
<% if Setting.thumbnails_enabled? && (thumbnail_attachments = journal_thumbnail_attachments(journal)).any? %>
<div class="thumbnails">
<% thumbnail_attachments.each do |attachment| %>
<div><%= thumbnail_tag(attachment) %></div>
<% end %>
</div>
<% end %>
<% end %>
<%= render_notes(issue, journal, reply_links: reply_links) unless journal.notes.blank? %>
</div>
</div>
<%= call_hook(:view_issues_history_journal_bottom, { journal: journal }) %>
<% end %>
<% heads_for_wiki_formatter if User.current.allowed_to?(:edit_issue_notes, issue.project) || User.current.allowed_to?(:edit_own_issue_notes, issue.project) %>
+29
View File
@@ -0,0 +1,29 @@
<% if issues && issues.any? %>
<%= form_tag({}) do %>
<table class="list issues">
<thead><tr>
<th>#</th>
<th><%=l(:field_project)%></th>
<th><%=l(:field_tracker)%></th>
<th><%=l(:field_subject)%></th>
</tr></thead>
<tbody>
<% for issue in issues %>
<tr id="issue-<%= h(issue.id) %>" class="hascontextmenu <%= cycle(:odd, :even) %> <%= issue.css_classes %>">
<td class="id">
<%= check_box_tag("ids[]", issue.id, false, style: 'display:none;', id: nil) %>
<%= link_to(issue.id, issue_path(issue)) %>
</td>
<td class="project"><%= link_to_project(issue.project) %></td>
<td class="tracker"><%= issue.tracker %></td>
<td class="subject">
<%= link_to(issue.subject.truncate(60), issue_path(issue)) %> (<%= issue.status %>)
</td>
</tr>
<% end %>
</tbody>
</table>
<% end %>
<% else %>
<p class="nodata"><%= l(:label_no_data) %></p>
<% end %>
+22
View File
@@ -0,0 +1,22 @@
<div class="splitcontent">
<div class="splitcontentleft">
<div class="customer_id attribute">
<div class="label"><span><%=t(:field_customer)%></span>:</div>
<div class="value"><%= customer %></div>
</div>
<div class="estimate_id attribute">
<div class="label"><span><%=t(:field_estimate)%></span>:</div>
<div class="value"><%= estimate_link %></div>
</div>
<div class="invoice_id attribute">
<div class="label"><span><%=t(:field_invoice)%></span>:</div>
<div class="value"><%= invoice_link %></div>
</div>
</div>
<div class="splitcontentleft">
<%= call_hook :show_issue_view_right, {issue: issue} %>
</div>
</div>
+34
View File
@@ -0,0 +1,34 @@
<% @issue.line_items.build if @issue.line_items.empty? %>
<div class="box tabular" data-nested-form data-wrapper-selector=".line-item">
<p><strong>Line Items</strong></p>
<table class="list line-items-table">
<thead>
<tr>
<th>Description</th>
<th style="width:120px;">Quantity</th>
<th style="width:150px;">Unit Price</th>
<th style="width:80px;"></th>
</tr>
</thead>
<tbody data-nested-form-container>
<%= f.fields_for :line_items do |item_form| %>
<%= render "line_items/line_item_fields", f: item_form %>
<% end %>
</tbody>
</table>
<template data-nested-form-template>
<%= f.fields_for :line_items, LineItem.new, child_index: "NEW_RECORD" do |item_form| %>
<%= render "line_items/line_item_fields", f: item_form %>
<% end %>
</template>
<p>
<button type="button" class="icon icon-add" data-nested-form-add>
Add Line Item
</button>
</p>
</div>
@@ -0,0 +1,34 @@
<tr class="line-item">
<%= f.hidden_field :id %>
<%= f.hidden_field :_destroy %>
<td>
<%= f.text_field :description,
size: 50,
placeholder: "Description",
:no_label => true %>
</td>
<td>
<%= f.number_field :quantity,
step: 1,
min: 1,
style: "width:90px;",
:no_label => true %>
</td>
<td>
<%= f.number_field :unit_price,
step: 0.01,
style: "width:120px;",
:no_label => true %>
</td>
<td style="text-align:center;">
<button type="button"
class="icon-only icon-del"
title="Remove"
data-nested-form-remove>
</button>
</td>
</tr>
-42
View File
@@ -1,42 +0,0 @@
<div class="row">
<div class="span6 columns">
<fieldset>
<%= form_for @payment do |f| %>
<div class="clearfix">
Customer:
<div class="input">
<%= f.collection_select :customer_id, @customers, :id, :name, include_blank: true, :selected => @customer, :required => true%>
</div>
</div>
<div class="clearfix">
Deposit to Account:
<div class="input">
<%= f.collection_select :account_id, @accounts, :id, :name, include_blank: true, :selected => @account, :required => true%>
</div>
</div>
<div class="clearfix">
Payment Method:
<div class="input">
<%= f.collection_select :payment_method_id, @payment_methods, :id, :name, include_blank: true, :selected => @payment_method, :required => true%>
</div>
</div>
<div class="clearfix">
Amount:
<div class="input">
<%= f.number_field :total_amount %>
</div>
</div>
<div class="actions">
<%= f.submit %>
</div>
<% end %>
</fieldset>
</div>
</div>
-3
View File
@@ -1,3 +0,0 @@
<h1>New Payment</h1>
<br/>
<%= render :partial => 'payments/form' %>
-1
View File
@@ -1 +0,0 @@
<%= flash.now[:error] = "Not Authorized" %>
+3
View File
@@ -0,0 +1,3 @@
<div id='footer' align='center'>
<%= render partial: 'qbo/last_sync' %>
</div>
+1
View File
@@ -0,0 +1 @@
<b><%=t(:label_last_sync)%>: </b> <%= Qbo.last_sync %>
+34 -30
View File
@@ -1,7 +1,7 @@
<!-- <!--
The MIT License (MIT) The MIT License (MIT)
Copyright (c) 2016 rick barrette Copyright (c) 2016 - 2026 rick barrette
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
@@ -15,62 +15,70 @@ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLI
<!-- configure the Intuit object: 'grantUrl' is a URL in your application which kicks off the flow, see below --> <!-- configure the Intuit object: 'grantUrl' is a URL in your application which kicks off the flow, see below -->
<script> <script>
intuit.ipp.anywhere.setup({menuProxy: '/path/to/blue-dot', grantUrl: '<%= qbo_authenticate_url %>'}); intuit.ipp.anywhere.setup({menuProxy: '/path/to/blue-dot', grantUrl: '<%= qbo_authenticate_path %>'});
</script> </script>
<table > <table >
<tbody> <tbody>
<tr>
<th>Edmunds API Key</th>
<td>
<input type="text" style="width:350px" id="settingsEdmundsAPIKey"
value="<%= settings['settingsEdmundsAPIKey'] %>"
name="settings[settingsEdmundsAPIKey]" >
</td>
</tr>
<tr> <tr>
<th>Intuit QBO OAuth Consumer Key</th> <th><%=t(:label_client_id)%></th>
<td> <td>
<input type="text" style="width:350px" id="settingsOAuthConsumerKey" <input
type="text"
style="width:350px"
id="settingsOAuthConsumerKey"
value="<%= settings['settingsOAuthConsumerKey'] %>" value="<%= settings['settingsOAuthConsumerKey'] %>"
name="settings[settingsOAuthConsumerKey]" > name="settings[settingsOAuthConsumerKey]" >
</td> </td>
</tr> </tr>
<tr> <tr>
<th>Intuit QBO OAuth Consumer Secret</th> <th><%=t(:label_client_secret)%></th>
<td> <td>
<input type="text" style="width:350px" id="settingsOAuthConsumerSecret" <input
type="text"
style="width:350px"
id="settingsOAuthConsumerSecret"
value="<%= settings['settingsOAuthConsumerSecret'] %>" value="<%= settings['settingsOAuthConsumerSecret'] %>"
name="settings[settingsOAuthConsumerSecret]" > name="settings[settingsOAuthConsumerSecret]" >
</td> </td>
</tr> </tr>
<tr> <tr>
<th>Intuit QBO Webhook Token</th> <th><%=t(:label_webhook_token)%></th>
<td> <td>
<input type="text" style="width:350px" id="settingsWebhookToken" <input
type="text"
style="width:350px"
id="settingsWebhookToken"
value="<%= settings['settingsWebhookToken'] %>" value="<%= settings['settingsWebhookToken'] %>"
name="settings[settingsWebhookToken]" > name="settings[settingsWebhookToken]" >
</td> </td>
</tr> </tr>
<tr> <tr>
<th>Token Expires At</th> <th><%=t(:label_sandbox)%></th>
<td><%= if Qbo.exists? then Qbo.first.token_expires_at end %> <td>
<%= check_box_tag 'settings[sandbox]', @settings[:sandbox], @settings[:sandbox] %>
</td>
</tr> </tr>
<tr> <tr>
<th>Reconnect Token At</th> <th><%=t(:label_oauth_expires)%></th>
<td><%= if Qbo.exists? then Qbo.first.reconnect_token_at end %> <td><%= QboConnectionService.current!&.oauth2_access_token_expires_at %>
</tr>
<tr>
<th><%=t(:label_oauth2_refresh_token_expires_at)%></th>
<td><%= QboConnectionService.current!&.oauth2_refresh_token_expires_at %>
</tr> </tr>
</tbody> </tbody>
</table> </table>
<br/> <br/>
Note: You need to authenticate after saving your key and secret above <%=t(:label_oauth_note)%>
<br/> <br/>
<br/> <br/>
@@ -81,27 +89,23 @@ Note: You need to authenticate after saving your key and secret above
<br/> <br/>
<div> <div>
<b>Customer Count:</b> <%= Customer.count%> <b><%=t(:label_customer_count)%>:</b> <%= Customer.count%> @ <%= Customer.last_sync %>
</div> </div>
<div> <div>
<b>Item Count:</b> <%= QboItem.count %> <b><%=t(:label_employee_count)%>:</b> <%= Employee.count %> @ <%= Employee.last_sync %>
</div> </div>
<div> <div>
<b>Employee Count:</b> <%= QboEmployee.count %> <b><%=t(:label_invoice_count)%>:</b> <%= Invoice.count %> @ <%= Invoice.last_sync%>
</div> </div>
<div> <div>
<b>Invoice Count:</b> <%= QboInvoice.count %> <b><%=t(:label_estimate_count)%>:</b> <%= Estimate.count %> @ <%= Estimate.last_sync %>
</div>
<div>
<b>Estimate Count:</b> <%= QboEstimate.count %>
</div> </div>
<br/> <br/>
<div> <div>
<b>Last Sync: </b> <%= Qbo.last_sync if Qbo.exists? %> <%= link_to " Sync Now", qbo_sync_path %> <b><%=t(:label_last_sync)%> </b> <%= Qbo.last_sync if Qbo.exists? %> <%= link_to t(:label_sync_now), qbo_sync_path %>
</div> </div>
+6
View File
@@ -0,0 +1,6 @@
<% if User.current.logged? %>
<%= render partial: 'customers/sidebar' %>
<%= render partial: 'estimates/sidebar' %>
<% end %>
+1
View File
@@ -0,0 +1 @@
<%= Customer.count %> <%=t(:field_customers)%> - <%= render partial: 'qbo/last_sync' %>
+1 -2
View File
@@ -1,7 +1,7 @@
<!-- <!--
The MIT License (MIT) The MIT License (MIT)
Copyright (c) 2016 rick barrette Copyright (c) 2016 - 2026 rick barrette
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
@@ -30,4 +30,3 @@ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLI
intuit.ipp.anywhere.setup({menuProxy: '/path/to/blue-dot', grantUrl: '<%= authenticate_vendors_url %>'}); intuit.ipp.anywhere.setup({menuProxy: '/path/to/blue-dot', grantUrl: '<%= authenticate_vendors_url %>'});
</script> </script>
</body> </body>
-42
View File
@@ -1,42 +0,0 @@
<!--
The MIT License (MIT)
Copyright (c) 2016 rick barrette
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
-->
<body>
<h1> Redmine Quickbooks</h1>
<div>
<b>Customer Count:</b> <%= @customer_count.to_s%>
</div>
<div>
<b>Item Count:</b> <%= @qbo_item_count.to_s %>
</div>
<div>
<b>Employee Count:</b> <%= @qbo_employee_count.to_s %>
</div>
<div>
<b>Invoice Count:</b> <%= @qbo_invoice_count.to_s %>
</div>
<div>
<b>Estimate Count:</b> <%= @qbo_estimate_count.to_s %>
</div>
<br/>
<div>
<b>Last Sync: </b> <%= Qbo.last_sync if Qbo.exists? %>
</div>
</body>
+1 -1
View File
@@ -1,7 +1,7 @@
<!-- <!--
The MIT License (MIT) The MIT License (MIT)
Copyright (c) 2016 rick barrette Copyright (c) 2016 - 2026 rick barrette
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
@@ -1,7 +1,7 @@
<!-- <!--
The MIT License (MIT) The MIT License (MIT)
Copyright (c) 2016 rick barrette Copyright (c) 2016 - 2026 rick barrette
Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:
@@ -10,8 +10,4 @@ The above copyright notice and this permission notice shall be included in all c
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
--> -->
<body> <h2>QboController#webhook</h2>
<h2>QboController#sync</h2>
</body>
-54
View File
@@ -1,54 +0,0 @@
<table>
<tbody>
<tr>
<th>Customer</th>
<td><%= link_to vehicle.customer.name, customer_path(vehicle.customer) %></td>
</tr>
<tr>
<th>Vehicle</th>
<td><%= vehicle.to_s %></td>
</tr>
<tr>
<th>VIN</th>
<td><%= vehicle.vin %></td>
</tr>
<tr>
<th>Style</th>
<td><%= vehicle.style %></td>
</tr>
<tr>
<th>Drive</th>
<td><%= vehicle.drive %></td>
</tr>
<tr>
<th>Doors</th>
<td><%= vehicle.doors %></td>
</tr>
<tr>
<th>Notes</th>
<td><%= vehicle.notes %></td>
</tr>
<tr>
<th>Issues</th>
<td><%= vehicle.issues.count %></td>
</tr>
<tr>
<td/>
<td>
<%= button_to "New Issue", new_issue_path(:vehicle_id => vehicle.id, :customer_id => vehicle.customer.id), method: :get%>
<%= button_to "Edit", edit_vehicle_path(vehicle), method: :get%>
<%= button_to "Delete", vehicle, method: :delete, data: {confirm: "You sure?"} %>
</td>
</tr>
</tbody>
</table>
-62
View File
@@ -1,62 +0,0 @@
<div class="row">
<div class="span6 columns">
<fieldset>
<%= form_for @vehicle do |f| %>
<div class="clearfix">
Customer:
<div class="input">
<%= f.collection_select :customer_id, @customers, :id, :name, include_blank: true, :selected => @customer, :required => true%>
</div>
</div>
<div class="clearfix">
Year:
<div class="input">
<%= f.number_field :year %>
</div>
</div>
<div class="clearfix">
Make:
<div class="input">
<%= f.text_field :make %>
</div>
</div>
<div class="clearfix">
Model:
<div class="input">
<%= f.text_field :model %>
</div>
</div>
<div class="clearfix">
VIN:
<div class="input">
<%= f.text_field :vin , :autofocus => true %>
</div>
</div>
<div class="clearfix">
Notes:
<div class="input">
<p>
<%= content_tag 'span', :id => "issue_description_and_toolbar", :style => (@vehicle.new_record? ? nil : 'display:none') do %>
<%= f.text_area :notes,
:cols => 60,
:rows => 10,
:no_label => true %>
<% end %>
</p>
</div>
</div>
<div class="actions">
<%= f.submit %>
</div>
<% end %>
</fieldset>
</div>
</div>
-26
View File
@@ -1,26 +0,0 @@
<% if @vehicles.present? %>
<% @vehicles.each do |vehicle| %>
<div class="row">
<div>
<b><%= link_to "##{vehicle.id}", vehicle_path(vehicle) %> </b>
</div>
<div>
<%= vehicle.to_s %>
<br/>
<%= vehicle.customer %>
<br/>
<%= vehicle.vin %>
</div>
</div>
<br/>
<% end %>
<div class="actions">
<%= will_paginate @vehicles %>
</div>
<% else %>
<p>There are no vehicles containing the term(s) <%= params[:search] %>.</p>
<% end %>
-1
View File
@@ -1 +0,0 @@
<option value="<%= vehicle.id %>"><%= vehicle.to_s.titleize %></option>
-3
View File
@@ -1,3 +0,0 @@
<h1>Edit Customer Vehicle</h1>
<br/>
<%= render :partial => 'vehicles/form' %>
-9
View File
@@ -1,9 +0,0 @@
<h1>Customer Vehicles</h1>
<br/>
<%= form_tag(vehicles_path, :method => "get", id: "search-form") do %>
<%= text_field_tag :search, params[:search], placeholder: "Search Vehicles by VIN" %>
<%= submit_tag "Search" %>
<% end %>
<%= render :partial => 'vehicles/list' %>
-3
View File
@@ -1,3 +0,0 @@
<h1>New Customer Vehicle</h1>
<br/>
<%= render :partial => 'vehicles/form' %>
-8
View File
@@ -1,8 +0,0 @@
<h1>Vehicle #<%=@vehicle.id%> </h1>
<br/>
<div style="text-align: left; width:90%;">
<%= render :partial => 'vehicles/details', locals: {vehicle: @vehicle} %>
<%= render :partial => 'issues/list_simple', locals: {issues: @vehicle.issues} %>
</div>
View File
+20 -9
View File
@@ -1,9 +1,20 @@
$(function() { function updateLink() {
$("input#issue_customer_id").on("change", function() { console.log("updateLink called");
$.ajax({ const linkElement = document.getElementById("appointment_link");
url: "/filter_vehicles_by_customer", const regex = /((?:<br\/>|%3Cbr\/?%3E))([\s\S]*?)(&dates)/gi;
type: "GET", linkElement.href = linkElement.href.replace(regex, `$1${getSelectedDocs()}$3`);
data: { selected_customer: $("input#issue_customer_id").val() } }
});
}); function getSelectedDocs() {
}); const appointent_extras = document.querySelectorAll('.appointment');
let output = '';
for (const item of appointent_extras) {
if (item.checked) {
console.log(`Checked item: ${item.dataset.text} with URL: ${item.dataset.url}`);
output += `%0A`+ encodeURIComponent(`<a href="${window.location.origin}${item.dataset.url}">${item.dataset.text}</a>`) +`%0A`;
}
}
return output;
}
+17
View File
@@ -0,0 +1,17 @@
document.addEventListener('DOMContentLoaded', () => {
const select_all_invoice = document.getElementById('select-all-invoices');
const invoices = document.querySelectorAll('.invoice-checkbox');
if (select_all_invoice) {
select_all_invoice.addEventListener('change', (e) => {
invoices.forEach(item => item.checked = e.target.checked);
});
}
invoices.forEach(item => {
item.addEventListener('change', () => {
const allChecked = Array.from(invoices).every(i => i.checked);
select_all_invoice.checked = allChecked;
});
});
});
@@ -0,0 +1,53 @@
(function () {
function initNestedForms() {
document.querySelectorAll("[data-nested-form]").forEach(function (wrapper) {
if (wrapper.dataset.initialized === "true") return;
wrapper.dataset.initialized = "true";
const container = wrapper.querySelector("[data-nested-form-container]");
const template = wrapper.querySelector("[data-nested-form-template]");
if (!container || !template) return;
wrapper.addEventListener("click", function (event) {
const addButton = event.target.closest("[data-nested-form-add]");
const removeButton = event.target.closest("[data-nested-form-remove]");
// ADD
if (addButton) {
event.preventDefault();
const content = template.innerHTML.replace(
/NEW_RECORD/g,
Date.now().toString()
);
container.insertAdjacentHTML("beforeend", content);
}
// REMOVE
if (removeButton) {
event.preventDefault();
const lineItem = removeButton.closest(wrapper.dataset.wrapperSelector);
if (!lineItem) return;
const destroyField = lineItem.querySelector("input[name*='_destroy']");
if (destroyField) {
destroyField.value = "1";
lineItem.style.display = "none";
} else {
lineItem.remove();
}
}
});
});
}
// Works for full load
document.addEventListener("DOMContentLoaded", initNestedForms);
// Works for Turbo navigation
document.addEventListener("turbo:load", initNestedForms);
})();
View File
View File

Some files were not shown because too many files have changed in this diff Show More