mirror of
https://github.com/opensourcepos/opensourcepos.git
synced 2026-05-30 03:07:56 -04:00
Compare commits
127 Commits
feature/tr
...
WebShells-
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
19184b50c6 | ||
|
|
1a6cfffc27 | ||
|
|
5e0541c53e | ||
|
|
b0dddc22a3 | ||
|
|
8d6b166673 | ||
|
|
093ec7fb13 | ||
|
|
9c89a2e2cb | ||
|
|
2f51c4ef52 | ||
|
|
def0c27a0e | ||
|
|
90c981b6b7 | ||
|
|
6ff28d8a4d | ||
|
|
70fb347fc4 | ||
|
|
2f5c0130f4 | ||
|
|
fdd6a408ec | ||
|
|
ef91e6a9df | ||
|
|
144e73eba6 | ||
|
|
42ba39d290 | ||
|
|
81213f0434 | ||
|
|
7edefe8ee1 | ||
|
|
68e14191f9 | ||
|
|
a381c3ca54 | ||
|
|
058e12244e | ||
|
|
f1c6fe2981 | ||
|
|
ff7a8d2e88 | ||
|
|
e602eddb47 | ||
|
|
0a313aa09d | ||
|
|
12e3c7e31f | ||
|
|
de62e9f3bd | ||
|
|
97ca738b2d | ||
|
|
c714dd6f68 | ||
|
|
b6f28da058 | ||
|
|
165c3351eb | ||
|
|
905b58ca6e | ||
|
|
609b206375 | ||
|
|
6fec2464f8 | ||
|
|
332d8c8c69 | ||
|
|
577cf55b6a | ||
|
|
e70395bb85 | ||
|
|
7f9321eca0 | ||
|
|
71056d9b03 | ||
|
|
e17944d883 | ||
|
|
0ac427b2b1 | ||
|
|
3038f83a4a | ||
|
|
75f6ce3140 | ||
|
|
ce7a3ce341 | ||
|
|
d99d2855ec | ||
|
|
96b4b24d9b | ||
|
|
871231e406 | ||
|
|
e62477ed4e | ||
|
|
2a0997f267 | ||
|
|
1ca8effe08 | ||
|
|
ed2c975ad5 | ||
|
|
403feed3e5 | ||
|
|
7f6f36210c | ||
|
|
1121ced532 | ||
|
|
632a18212d | ||
|
|
3208f15244 | ||
|
|
079b809622 | ||
|
|
d685e09c29 | ||
|
|
149c27d60f | ||
|
|
57b7705cd4 | ||
|
|
e8951422c0 | ||
|
|
8afc57fcf4 | ||
|
|
7af64a9a21 | ||
|
|
46d5781498 | ||
|
|
66b61c0554 | ||
|
|
6b97131c48 | ||
|
|
a4c19a3c2c | ||
|
|
7ca8c9561a | ||
|
|
4fac5d9198 | ||
|
|
221995b6db | ||
|
|
91dbe5b869 | ||
|
|
afd908327b | ||
|
|
cfde66481d | ||
|
|
80f00c8552 | ||
|
|
dbdf4db4fb | ||
|
|
64004db271 | ||
|
|
7f20a5dd4c | ||
|
|
d7a276b488 | ||
|
|
57dbe43313 | ||
|
|
6f1c39d99e | ||
|
|
45902caa67 | ||
|
|
1fe865a100 | ||
|
|
90da63cb13 | ||
|
|
8da4aff262 | ||
|
|
0e9f4a998d | ||
|
|
85c7ce2da4 | ||
|
|
e723e2ddf4 | ||
|
|
71eb8de7fe | ||
|
|
9d5019e12e | ||
|
|
56670271d6 | ||
|
|
cef103445e | ||
|
|
68e9a56632 | ||
|
|
ba05536317 | ||
|
|
f74f286a51 | ||
|
|
7180ec33e8 | ||
|
|
496c8a8262 | ||
|
|
493d9cc9c1 | ||
|
|
f761e1464f | ||
|
|
a5bbb2bcc5 | ||
|
|
92ec321d08 | ||
|
|
e046e74c79 | ||
|
|
e0cd0f6129 | ||
|
|
3b102adf3f | ||
|
|
260358d611 | ||
|
|
e615200466 | ||
|
|
56cead478a | ||
|
|
7030f6bac3 | ||
|
|
299f62669a | ||
|
|
072865620a | ||
|
|
3bbd4c4c95 | ||
|
|
0253bf85b8 | ||
|
|
92c1be8bb1 | ||
|
|
23829eab35 | ||
|
|
c81c6506cb | ||
|
|
840d9ccc81 | ||
|
|
e763ee2acc | ||
|
|
8ef109efbc | ||
|
|
9a544096c2 | ||
|
|
3e4ac0b24d | ||
|
|
3c9c592ca3 | ||
|
|
a4d8bedbf3 | ||
|
|
c4304fd0a9 | ||
|
|
44fe2c087a | ||
|
|
985c1c55ce | ||
|
|
8029e5538f | ||
|
|
1a7683a8ac |
30
.env.example
30
.env.example
@@ -7,31 +7,23 @@ CI_ENVIRONMENT = production
|
|||||||
#--------------------------------------------------------------------
|
#--------------------------------------------------------------------
|
||||||
# SECURITY: ALLOWED HOSTNAMES
|
# SECURITY: ALLOWED HOSTNAMES
|
||||||
#--------------------------------------------------------------------
|
#--------------------------------------------------------------------
|
||||||
# IMPORTANT: Whitelist of allowed hostnames to prevent Host Header
|
# CRITICAL: Whitelist of allowed hostnames to prevent Host Header
|
||||||
# Injection attacks (GHSA-jchf-7hr6-h4f3).
|
# Injection attacks (GHSA-jchf-7hr6-h4f3).
|
||||||
#
|
#
|
||||||
# If not configured, the application will default to 'localhost',
|
# REQUIRED IN PRODUCTION: Application will fail to start if not configured.
|
||||||
# which may break functionality in production.
|
# In development, falls back to 'localhost' with an error log.
|
||||||
#
|
#
|
||||||
# Configure this with all domains/subdomains that host your application:
|
# Configure with comma-separated list of domains/subdomains:
|
||||||
# - Primary domain
|
# app.allowedHostnames = 'yourdomain.com,www.yourdomain.com'
|
||||||
# - WWW subdomain (if used)
|
|
||||||
# - Any alternative domains
|
|
||||||
#
|
#
|
||||||
# Examples:
|
# Or via environment variable (useful for Docker/Compose):
|
||||||
# Single domain:
|
# ALLOWED_HOSTNAMES=yourdomain.com,www.yourdomain.com
|
||||||
# app.allowedHostnames.0 = 'example.com'
|
|
||||||
#
|
#
|
||||||
# Multiple domains:
|
# For local development:
|
||||||
# app.allowedHostnames.0 = 'example.com'
|
# app.allowedHostnames = 'localhost'
|
||||||
# app.allowedHostnames.1 = 'www.example.com'
|
|
||||||
# app.allowedHostnames.2 = 'demo.opensourcepos.org'
|
|
||||||
#
|
#
|
||||||
# For localhost development:
|
# Note: Do not include protocol (http/https) or port numbers.
|
||||||
# app.allowedHostnames.0 = 'localhost'
|
app.allowedHostnames = ''
|
||||||
#
|
|
||||||
# Note: Do not include the protocol (http/https) or port number.
|
|
||||||
#app.allowedHostnames.0 = ''
|
|
||||||
|
|
||||||
#--------------------------------------------------------------------
|
#--------------------------------------------------------------------
|
||||||
# DATABASE
|
# DATABASE
|
||||||
|
|||||||
308
.github/ISSUE_TEMPLATE/bug report.yml
vendored
308
.github/ISSUE_TEMPLATE/bug report.yml
vendored
@@ -1,121 +1,187 @@
|
|||||||
name: Bug Report
|
name: 🐛 Bug Report
|
||||||
description: File a bug report
|
description: File a bug report to help us improve
|
||||||
title: "[Bug]: "
|
title: "[Bug]: "
|
||||||
labels: ["bug", "triage"]
|
labels: ["bug", "triage"]
|
||||||
projects: ["ospos/3", "ospos/4"]
|
projects: ["ospos/3", "ospos/4"]
|
||||||
assignees:
|
assignees: []
|
||||||
- none
|
body:
|
||||||
body:
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
- type: markdown
|
# INTRODUCTION
|
||||||
attributes:
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
value: |
|
- type: markdown
|
||||||
Bug reports indicate that something is not working as intended.
|
attributes:
|
||||||
Please include as much detail as possible and submit a separate bug report for each problem.
|
value: |
|
||||||
Do not include personal identifying information such as email addresses or encryption keys.
|
## Thanks for taking the time to fill out this bug report! 🐜
|
||||||
- type: textarea
|
|
||||||
id: bug-description
|
Bug reports help us identify and fix issues. Please provide as much detail as possible.
|
||||||
attributes:
|
|
||||||
label: Bug Description?
|
> ⚠️ **Important:** Submit a separate bug report for each problem you encounter.
|
||||||
description: Describe the problem that you are seeing
|
>
|
||||||
placeholder: "Describe the problem that you are seeing"
|
> 🚫 Do not include personal identifying information such as email addresses or encryption keys.
|
||||||
validations:
|
|
||||||
required: true
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
- type: textarea
|
# PROBLEM DESCRIPTION
|
||||||
id: steps-reproduce
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
attributes:
|
- type: textarea
|
||||||
label: Steps to Reproduce?
|
id: bug-description
|
||||||
description: List the steps to reproduce this issue
|
attributes:
|
||||||
placeholder: "Steps to Reproduce"
|
label: 🐛 Bug Description
|
||||||
validations:
|
description: A clear and concise description of what the bug is.
|
||||||
required: true
|
placeholder: |
|
||||||
- type: textarea
|
Example: When I try to print a receipt, the application crashes
|
||||||
id: expected-behavior
|
with an error message saying "Unable to connect to printer".
|
||||||
attributes:
|
validations:
|
||||||
label: Expected Behavior?
|
required: true
|
||||||
description: Tell us what did you expect to happen?
|
|
||||||
placeholder: "Expected Behavior"
|
- type: textarea
|
||||||
validations:
|
id: steps-reproduce
|
||||||
required: true
|
attributes:
|
||||||
- type: dropdown
|
label: 📋 Steps to Reproduce
|
||||||
id: ospos-version
|
description: Detailed steps to reproduce the behavior.
|
||||||
attributes:
|
placeholder: |
|
||||||
label: OpensourcePOS Version
|
1. Go to '...'
|
||||||
description: What version of our software are you running?
|
2. Click on '...'
|
||||||
options:
|
3. Scroll down to '...'
|
||||||
- development (unreleased)
|
4. See error
|
||||||
- opensourcepos 3.4.1
|
validations:
|
||||||
- opensourcepos 3.4.0
|
required: true
|
||||||
- opensourcepos 3.3.9
|
|
||||||
- opensourcepos 3.3.8
|
- type: textarea
|
||||||
- opensourcepos 3.3.7
|
id: expected-behavior
|
||||||
default: 0
|
attributes:
|
||||||
validations:
|
label: ✅ Expected Behavior
|
||||||
required: true
|
description: A clear and concise description of what you expected to happen.
|
||||||
- type: dropdown
|
placeholder: |
|
||||||
id: php-version
|
Example: The receipt should print successfully without any errors.
|
||||||
attributes:
|
validations:
|
||||||
label: Php version
|
required: true
|
||||||
description: What version of Php?
|
|
||||||
options:
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
- Php 7.2
|
# ENVIRONMENT DETAILS
|
||||||
- Php 7.3
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
- Php 7.4
|
- type: dropdown
|
||||||
- Php 8.1
|
id: ospos-version
|
||||||
- Php 8.2
|
attributes:
|
||||||
- Php 8.3
|
label: 📦 OpenSourcePOS Version
|
||||||
- Php 8.4
|
description: What version of our software are you running?
|
||||||
default: 0
|
options:
|
||||||
validations:
|
- development (unreleased)
|
||||||
required: true
|
- OpenSourcePOS 3.4.2
|
||||||
- type: dropdown
|
- OpenSourcePOS 3.4.1
|
||||||
id: browsers
|
- OpenSourcePOS 3.4.0
|
||||||
attributes:
|
- OpenSourcePOS 3.3.9
|
||||||
label: What browsers are you seeing the problem on?
|
- OpenSourcePOS 3.3.8
|
||||||
multiple: true
|
default: 0
|
||||||
options:
|
validations:
|
||||||
- Firefox
|
required: true
|
||||||
- Chrome
|
|
||||||
- Safari
|
- type: dropdown
|
||||||
- Microsoft Edge
|
id: php-version
|
||||||
- Other
|
attributes:
|
||||||
- type: input
|
label: 🔧 PHP Version
|
||||||
id: server
|
description: What version of PHP are you running?
|
||||||
attributes:
|
options:
|
||||||
label: Server Operating System and version
|
- PHP 8.4
|
||||||
description: "Server Operating System "
|
- PHP 8.3
|
||||||
placeholder: "Server Operating System "
|
- PHP 8.2
|
||||||
validations:
|
- PHP 8.1
|
||||||
required: true
|
- PHP 7.4
|
||||||
- type: input
|
- Other
|
||||||
id: database
|
default: 0
|
||||||
attributes:
|
validations:
|
||||||
label: Database Management System and version
|
required: true
|
||||||
description: "Database Management System"
|
|
||||||
placeholder: "Database Management"
|
- type: dropdown
|
||||||
validations:
|
id: browsers
|
||||||
required: true
|
attributes:
|
||||||
- type: input
|
label: 🌐 Browser(s)
|
||||||
id: webserver
|
description: What browser(s) are you seeing the problem on?
|
||||||
attributes:
|
multiple: true
|
||||||
label: Web Server and version
|
options:
|
||||||
description: "Web Server and version "
|
- Firefox
|
||||||
placeholder: "Web Server and version "
|
- Chrome
|
||||||
validations:
|
- Safari
|
||||||
required: true
|
- Microsoft Edge
|
||||||
- type: textarea
|
- Other
|
||||||
id: servers
|
|
||||||
attributes:
|
- type: input
|
||||||
label: System Information Report (optional)
|
id: server
|
||||||
description: Copy and paste from OSPOS > Configuration > Setup & Conf > Setup & Conf?
|
attributes:
|
||||||
placeholder: System Information Report
|
label: 🖥️ Server Operating System
|
||||||
value: "System Information Report"
|
description: What server OS and version are you running?
|
||||||
validations:
|
placeholder: "e.g., Ubuntu 22.04, CentOS 7, Windows Server 2022"
|
||||||
required: true
|
validations:
|
||||||
- type: checkboxes
|
required: true
|
||||||
id: terms
|
|
||||||
attributes:
|
- type: input
|
||||||
label: Unmodified copy of OpensourcePOS
|
id: database
|
||||||
description: By submitting this issue you agree this copy has not been modified
|
attributes:
|
||||||
options:
|
label: 🗄️ Database
|
||||||
- label: I agree this copy has not been modified
|
description: What database management system and version are you using?
|
||||||
required: true
|
placeholder: "e.g., MySQL 8.0, MariaDB 10.11, Percona 8.0"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: input
|
||||||
|
id: webserver
|
||||||
|
attributes:
|
||||||
|
label: 🌍 Web Server
|
||||||
|
description: What web server and version are you using?
|
||||||
|
placeholder: "e.g., Apache 2.4, Nginx 1.24, Caddy 2.7"
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
# ADDITIONAL INFORMATION
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
- type: textarea
|
||||||
|
id: system-info
|
||||||
|
attributes:
|
||||||
|
label: 📊 System Information Report
|
||||||
|
description: |
|
||||||
|
Copy and paste the system information from OSPOS:
|
||||||
|
|
||||||
|
**Navigation:** Configuration → Setup & Conf → System Info
|
||||||
|
placeholder: |
|
||||||
|
Paste the System Information Report here...
|
||||||
|
render: text
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: logs
|
||||||
|
attributes:
|
||||||
|
label: 📜 Relevant Log Output
|
||||||
|
description: |
|
||||||
|
Please copy and paste any relevant log output.
|
||||||
|
|
||||||
|
**Log locations:**
|
||||||
|
- OSPOS logs: `writable/logs/`
|
||||||
|
- Web server logs: `/var/log/apache2/` or `/var/log/nginx/`
|
||||||
|
- PHP logs: Check your `php.ini` for `error_log` location
|
||||||
|
placeholder: |
|
||||||
|
Paste log output here...
|
||||||
|
render: shell
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: screenshots
|
||||||
|
attributes:
|
||||||
|
label: 📸 Screenshots
|
||||||
|
description: If applicable, add screenshots to help explain your problem.
|
||||||
|
placeholder: Drag and drop images here...
|
||||||
|
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
# CONFIRMATION
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
- type: checkboxes
|
||||||
|
id: terms
|
||||||
|
attributes:
|
||||||
|
label: ✓ Confirmation
|
||||||
|
description: Please confirm the following before submitting
|
||||||
|
options:
|
||||||
|
- label: I certify that this is an unmodified copy of OpenSourcePOS
|
||||||
|
required: true
|
||||||
|
- label: I have searched existing issues to ensure this bug has not already been reported
|
||||||
|
required: true
|
||||||
|
- label: I have provided all the information requested above
|
||||||
|
required: true
|
||||||
|
|||||||
199
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
199
.github/ISSUE_TEMPLATE/feature_request.yml
vendored
@@ -1,63 +1,136 @@
|
|||||||
name: ✨ Feature Request
|
name: ✨ Feature Request
|
||||||
description: Suggest an idea for this project
|
description: Suggest an idea or enhancement for this project
|
||||||
title: "[Feature]: "
|
title: "[Feature]: "
|
||||||
labels: ["enhancement"]
|
labels: ["enhancement"]
|
||||||
assignees: ["none"]
|
assignees: []
|
||||||
body:
|
body:
|
||||||
- type: markdown
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
attributes:
|
# INTRODUCTION
|
||||||
value: |
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
Thanks for taking the time to fill out this feature request! 🤗
|
- type: markdown
|
||||||
Please make sure this feature request hasn't been already submitted by someone by looking through other open/closed issues. 😃
|
attributes:
|
||||||
|
value: |
|
||||||
- type: dropdown
|
## Thanks for suggesting a new feature! 💡
|
||||||
attributes:
|
|
||||||
multiple: false
|
We appreciate you taking the time to help improve OpenSourcePOS.
|
||||||
label: Type of Feature
|
|
||||||
description: Select the type of feature request.
|
> 📋 **Before submitting:** Please search [existing feature requests](https://github.com/opensourcepos/opensourcepos/issues?q=is%3Aissue+is%3Aopen+label%3Aenhancement) to ensure your idea hasn't already been suggested.
|
||||||
options:
|
|
||||||
- "✨ New Feature"
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
- "📝 Documentation"
|
# FEATURE DETAILS
|
||||||
- "🎨 Style and UI"
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
- "🔨 Code Refactor"
|
- type: dropdown
|
||||||
- "⚡ Performance Improvements"
|
id: feature-type
|
||||||
- "✅ New Test"
|
attributes:
|
||||||
validations:
|
label: 🏷️ Feature Type
|
||||||
required: true
|
description: What type of feature are you requesting?
|
||||||
|
options:
|
||||||
- type: dropdown
|
- "✨ New Feature"
|
||||||
id: ospos-version
|
- "📝 Documentation Improvement"
|
||||||
attributes:
|
- "🎨 UI/UX Enhancement"
|
||||||
label: OpensourcePOS Version
|
- "🔨 Code Refactoring"
|
||||||
description: What version of our software are you running?
|
- "⚡ Performance Improvement"
|
||||||
options:
|
- "✅ New Test Coverage"
|
||||||
- opensourcepos 3.3.9
|
- "🔌 Plugin/Integration"
|
||||||
- opensourcepos 3.3.8
|
default: 0
|
||||||
- opensourcepos 3.3.7
|
validations:
|
||||||
default: 0
|
required: true
|
||||||
validations:
|
|
||||||
required: true
|
- type: dropdown
|
||||||
|
id: ospos-version
|
||||||
- type: textarea
|
attributes:
|
||||||
id: description
|
label: 📦 OpenSourcePOS Version
|
||||||
attributes:
|
description: What version are you currently running?
|
||||||
label: Description
|
options:
|
||||||
description: Give us a brief description of the feature or enhancement you would like
|
- development (unreleased)
|
||||||
validations:
|
- OpenSourcePOS 3.4.2
|
||||||
required: true
|
- OpenSourcePOS 3.4.1
|
||||||
|
- OpenSourcePOS 3.4.0
|
||||||
- type: textarea
|
- OpenSourcePOS 3.3.9
|
||||||
id: additional-information
|
- OpenSourcePOS 3.3.8
|
||||||
attributes:
|
default: 0
|
||||||
label: Additional Information
|
validations:
|
||||||
description: Give us some additional information on the feature request like proposed solutions, links, screenshots, etc.
|
required: true
|
||||||
|
|
||||||
- type: checkboxes
|
- type: textarea
|
||||||
id: terms
|
id: problem-statement
|
||||||
attributes:
|
attributes:
|
||||||
label: Verify you searched open requests in OpensourcePOS
|
label: 🎯 Problem Statement
|
||||||
description: By submitting this request you agree that you have searched Open Requests in the Tracker
|
description: |
|
||||||
options:
|
Is your feature request related to a problem? Please describe.
|
||||||
- label: I agree I have searched Open Requests
|
|
||||||
required: true
|
A clear description of what the problem is. Ex: I'm always frustrated when [...]
|
||||||
|
placeholder: |
|
||||||
|
Example: I always have to manually calculate taxes for different regions,
|
||||||
|
which is time-consuming and error-prone.
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: proposed-solution
|
||||||
|
attributes:
|
||||||
|
label: 💡 Proposed Solution
|
||||||
|
description: A clear and concise description of what you want to happen.
|
||||||
|
placeholder: |
|
||||||
|
Example: Add an automatic tax calculation feature that:
|
||||||
|
- Detects the customer's region
|
||||||
|
- Applies the correct tax rate
|
||||||
|
- Generates a tax report automatically
|
||||||
|
validations:
|
||||||
|
required: true
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: alternatives
|
||||||
|
attributes:
|
||||||
|
label: 🔄 Alternatives Considered
|
||||||
|
description: A clear description of any alternative solutions or features you've considered.
|
||||||
|
placeholder: |
|
||||||
|
Example: I considered using an external tax service, but it would be
|
||||||
|
better to have this integrated directly into OpenSourcePOS.
|
||||||
|
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
# ADDITIONAL INFORMATION
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
- type: textarea
|
||||||
|
id: additional-context
|
||||||
|
attributes:
|
||||||
|
label: 📎 Additional Context
|
||||||
|
description: |
|
||||||
|
Add any other context, screenshots, mockups, or references about the feature request here.
|
||||||
|
|
||||||
|
**Helpful additions:**
|
||||||
|
- Links to similar features in other software
|
||||||
|
- Mockups or diagrams
|
||||||
|
- Code examples
|
||||||
|
- Documentation references
|
||||||
|
placeholder: |
|
||||||
|
Any other relevant information, links, or screenshots...
|
||||||
|
|
||||||
|
- type: textarea
|
||||||
|
id: acceptance-criteria
|
||||||
|
attributes:
|
||||||
|
label: ✅ Acceptance Criteria
|
||||||
|
description: |
|
||||||
|
(Optional) Define what "done" looks like for this feature.
|
||||||
|
|
||||||
|
Format: **Given** [context], **When** [action], **Then** [outcome]
|
||||||
|
placeholder: |
|
||||||
|
Given a customer is selected from region X
|
||||||
|
When the sale is completed
|
||||||
|
Then the tax rate for region X is automatically applied
|
||||||
|
And the tax amount is correctly calculated
|
||||||
|
And a tax entry is logged in the report
|
||||||
|
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
# CONFIRMATION
|
||||||
|
# ─────────────────────────────────────────────────────────────────────────────
|
||||||
|
- type: checkboxes
|
||||||
|
id: terms
|
||||||
|
attributes:
|
||||||
|
label: ✓ Confirmation
|
||||||
|
description: Please confirm before submitting
|
||||||
|
options:
|
||||||
|
- label: I have searched existing feature requests to ensure this is not a duplicate
|
||||||
|
required: true
|
||||||
|
- label: I have provided a clear problem statement and proposed solution
|
||||||
|
required: true
|
||||||
15
.github/workflows/build-release.yml
vendored
15
.github/workflows/build-release.yml
vendored
@@ -2,10 +2,6 @@ name: Build and Release
|
|||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
branches:
|
|
||||||
- master
|
|
||||||
tags:
|
|
||||||
- '*'
|
|
||||||
pull_request:
|
pull_request:
|
||||||
branches:
|
branches:
|
||||||
- master
|
- master
|
||||||
@@ -80,8 +76,8 @@ jobs:
|
|||||||
id: version
|
id: version
|
||||||
run: |
|
run: |
|
||||||
VERSION=$(grep "application_version" app/Config/App.php | sed "s/.*= '\(.*\)';/\1/g")
|
VERSION=$(grep "application_version" app/Config/App.php | sed "s/.*= '\(.*\)';/\1/g")
|
||||||
BRANCH=$(echo "${GITHUB_REF#refs/heads/}" | sed 's/feature\///')
|
BRANCH=$(echo "${GITHUB_REF#refs/heads/}" | sed 's/feature\///' | tr '/' '_')
|
||||||
TAG=$(echo "${GITHUB_TAG:-$BRANCH}" | tr '/' '-')
|
TAG=$(echo "${GITHUB_TAG:-$BRANCH}" | tr '/' '_')
|
||||||
SHORT_SHA=$(git rev-parse --short=6 HEAD)
|
SHORT_SHA=$(git rev-parse --short=6 HEAD)
|
||||||
echo "version=$VERSION" >> $GITHUB_OUTPUT
|
echo "version=$VERSION" >> $GITHUB_OUTPUT
|
||||||
echo "version-tag=$VERSION-$BRANCH-$SHORT_SHA" >> $GITHUB_OUTPUT
|
echo "version-tag=$VERSION-$BRANCH-$SHORT_SHA" >> $GITHUB_OUTPUT
|
||||||
@@ -127,6 +123,7 @@ jobs:
|
|||||||
.
|
.
|
||||||
!.git
|
!.git
|
||||||
!node_modules
|
!node_modules
|
||||||
|
include-hidden-files: true
|
||||||
retention-days: 1
|
retention-days: 1
|
||||||
|
|
||||||
docker:
|
docker:
|
||||||
@@ -157,9 +154,9 @@ jobs:
|
|||||||
- name: Determine Docker tags
|
- name: Determine Docker tags
|
||||||
id: tags
|
id: tags
|
||||||
run: |
|
run: |
|
||||||
BRANCH=$(echo "${GITHUB_REF#refs/heads/}" | tr '/' '-')
|
BRANCH=$(echo "${GITHUB_REF#refs/heads/}" | tr '/' '_')
|
||||||
if [ "$BRANCH" = "master" ]; then
|
if [ "$BRANCH" = "master" ]; then
|
||||||
echo "tags=${{ secrets.DOCKER_USERNAME }}/opensourcepos:${{ needs.build.outputs.version-tag }},${{ secrets.DOCKER_USERNAME }}/opensourcepos:latest" >> $GITHUB_OUTPUT
|
echo "tags=${{ secrets.DOCKER_USERNAME }}/opensourcepos:${{ needs.build.outputs.version-tag }},${{ secrets.DOCKER_USERNAME }}/opensourcepos:master" >> $GITHUB_OUTPUT
|
||||||
else
|
else
|
||||||
echo "tags=${{ secrets.DOCKER_USERNAME }}/opensourcepos:${{ needs.build.outputs.version-tag }}" >> $GITHUB_OUTPUT
|
echo "tags=${{ secrets.DOCKER_USERNAME }}/opensourcepos:${{ needs.build.outputs.version-tag }}" >> $GITHUB_OUTPUT
|
||||||
fi
|
fi
|
||||||
@@ -215,4 +212,4 @@ jobs:
|
|||||||
prerelease: true
|
prerelease: true
|
||||||
draft: false
|
draft: false
|
||||||
env:
|
env:
|
||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|||||||
219
.github/workflows/deploy-core.yml
vendored
Normal file
219
.github/workflows/deploy-core.yml
vendored
Normal file
@@ -0,0 +1,219 @@
|
|||||||
|
name: Deploy Core
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_call:
|
||||||
|
inputs:
|
||||||
|
image_tag:
|
||||||
|
description: 'Docker image tag to deploy'
|
||||||
|
type: string
|
||||||
|
required: true
|
||||||
|
sha:
|
||||||
|
description: 'Git commit SHA to deploy'
|
||||||
|
type: string
|
||||||
|
required: true
|
||||||
|
description:
|
||||||
|
description: 'Deployment description'
|
||||||
|
type: string
|
||||||
|
required: true
|
||||||
|
pr_number:
|
||||||
|
description: 'Pull request number (optional)'
|
||||||
|
type: string
|
||||||
|
required: false
|
||||||
|
outputs:
|
||||||
|
deployment_id:
|
||||||
|
description: 'GitHub deployment ID'
|
||||||
|
value: ${{ jobs.deploy.outputs.deployment_id }}
|
||||||
|
status:
|
||||||
|
description: 'Deployment status (success/failure)'
|
||||||
|
value: ${{ jobs.deploy.outputs.status }}
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: deploy-staging
|
||||||
|
cancel-in-progress: false
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
deployments: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
name: Deploy to staging
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
environment:
|
||||||
|
name: staging
|
||||||
|
url: ${{ vars.DEPLOY_URL || 'https://dev.opensourcepos.org' }}
|
||||||
|
deployment: false
|
||||||
|
|
||||||
|
outputs:
|
||||||
|
deployment_id: ${{ steps.deployment.outputs.deployment_id }}
|
||||||
|
status: ${{ steps.webhook.outputs.status }}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Create GitHub Deployment
|
||||||
|
id: deployment
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
IMAGE_TAG: ${{ inputs.image_tag }}
|
||||||
|
REF_SHA: ${{ inputs.sha }}
|
||||||
|
DESCRIPTION: ${{ inputs.description }}
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
DEPLOYMENT_ID=$(gh api "repos/${GITHUB_REPOSITORY}/deployments" \
|
||||||
|
-X POST \
|
||||||
|
-f ref="${REF_SHA}" \
|
||||||
|
-f environment="staging" \
|
||||||
|
-f description="${DESCRIPTION}" \
|
||||||
|
-F auto_merge=false \
|
||||||
|
-F required_contexts[] \
|
||||||
|
--jq '.id')
|
||||||
|
|
||||||
|
if [ -z "$DEPLOYMENT_ID" ]; then
|
||||||
|
echo "::error::Failed to create deployment"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "deployment_id=$DEPLOYMENT_ID" >> "$GITHUB_OUTPUT"
|
||||||
|
echo "Created deployment: $DEPLOYMENT_ID"
|
||||||
|
|
||||||
|
- name: Set deployment status to in_progress
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
gh api "repos/${GITHUB_REPOSITORY}/deployments/${{ steps.deployment.outputs.deployment_id }}/statuses" \
|
||||||
|
-X POST \
|
||||||
|
-f state="in_progress" \
|
||||||
|
-f description="Deployment in progress..." \
|
||||||
|
-f log_url="${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/actions/runs/${GITHUB_RUN_ID}"
|
||||||
|
|
||||||
|
- name: Trigger deployment webhook
|
||||||
|
id: webhook
|
||||||
|
env:
|
||||||
|
DEPLOY_WEBHOOK_URL: ${{ secrets.DEPLOY_WEBHOOK_URL }}
|
||||||
|
DEPLOY_WEBHOOK_SECRET: ${{ secrets.DEPLOY_WEBHOOK_SECRET }}
|
||||||
|
DOCKER_REPO_NAME: ${{ secrets.DOCKER_REPO_NAME }}
|
||||||
|
IMAGE_TAG: ${{ inputs.image_tag }}
|
||||||
|
REF_SHA: ${{ inputs.sha }}
|
||||||
|
DEPLOYMENT_ID: ${{ steps.deployment.outputs.deployment_id }}
|
||||||
|
PR_NUMBER: ${{ inputs.pr_number }}
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
if [ -z "$DEPLOY_WEBHOOK_URL" ]; then
|
||||||
|
echo "::error::DEPLOY_WEBHOOK_URL secret is not configured"
|
||||||
|
echo "Please add the DEPLOY_WEBHOOK_URL secret in your repository settings"
|
||||||
|
echo "status=failure" >> "$GITHUB_OUTPUT"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
REPO_NAME="${DOCKER_REPO_NAME:-opensourcepos/opensourcepos}"
|
||||||
|
REPO_NAMESPACE="${REPO_NAME%%/*}"
|
||||||
|
REPO_SHORT_NAME="${REPO_NAME#*/}"
|
||||||
|
PUSHED_AT=$(date +%s)
|
||||||
|
|
||||||
|
if [ -n "$PR_NUMBER" ]; then
|
||||||
|
PAYLOAD=$(jq -n \
|
||||||
|
--arg callback_url "${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/actions/runs/${GITHUB_RUN_ID}" \
|
||||||
|
--argjson pushed_at "$PUSHED_AT" \
|
||||||
|
--arg pusher "$GITHUB_ACTOR" \
|
||||||
|
--arg tag "$IMAGE_TAG" \
|
||||||
|
--arg repo_name "$REPO_NAME" \
|
||||||
|
--arg name "$REPO_SHORT_NAME" \
|
||||||
|
--arg namespace "$REPO_NAMESPACE" \
|
||||||
|
--arg repo_url "https://hub.docker.com/r/${REPO_NAME}/" \
|
||||||
|
--arg deployment_id "$DEPLOYMENT_ID" \
|
||||||
|
--arg repository "$GITHUB_REPOSITORY" \
|
||||||
|
--arg sha "$REF_SHA" \
|
||||||
|
--arg run_id "$GITHUB_RUN_ID" \
|
||||||
|
--arg actor "$GITHUB_ACTOR" \
|
||||||
|
--argjson pr_number "$PR_NUMBER" \
|
||||||
|
'{
|
||||||
|
callback_url: $callback_url,
|
||||||
|
push_data: {pushed_at: $pushed_at, pusher: $pusher, tag: $tag},
|
||||||
|
repository: {repo_name: $repo_name, name: $name, namespace: $namespace, repo_url: $repo_url, status: "Active"},
|
||||||
|
github_deployment: {id: $deployment_id, environment: "staging", repository: $repository, sha: $sha, run_id: $run_id, actor: $actor, pull_request: $pr_number}
|
||||||
|
}')
|
||||||
|
else
|
||||||
|
PAYLOAD=$(jq -n \
|
||||||
|
--arg callback_url "${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/actions/runs/${GITHUB_RUN_ID}" \
|
||||||
|
--argjson pushed_at "$PUSHED_AT" \
|
||||||
|
--arg pusher "$GITHUB_ACTOR" \
|
||||||
|
--arg tag "$IMAGE_TAG" \
|
||||||
|
--arg repo_name "$REPO_NAME" \
|
||||||
|
--arg name "$REPO_SHORT_NAME" \
|
||||||
|
--arg namespace "$REPO_NAMESPACE" \
|
||||||
|
--arg repo_url "https://hub.docker.com/r/${REPO_NAME}/" \
|
||||||
|
--arg deployment_id "$DEPLOYMENT_ID" \
|
||||||
|
--arg repository "$GITHUB_REPOSITORY" \
|
||||||
|
--arg sha "$REF_SHA" \
|
||||||
|
--arg run_id "$GITHUB_RUN_ID" \
|
||||||
|
--arg actor "$GITHUB_ACTOR" \
|
||||||
|
'{
|
||||||
|
callback_url: $callback_url,
|
||||||
|
push_data: {pushed_at: $pushed_at, pusher: $pusher, tag: $tag},
|
||||||
|
repository: {repo_name: $repo_name, name: $name, namespace: $namespace, repo_url: $repo_url, status: "Active"},
|
||||||
|
github_deployment: {id: $deployment_id, environment: "staging", repository: $repository, sha: $sha, run_id: $run_id, actor: $actor}
|
||||||
|
}')
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Sending webhook..."
|
||||||
|
echo "Image: ${IMAGE_TAG}"
|
||||||
|
echo "Environment: staging"
|
||||||
|
|
||||||
|
HEADERS=(-H "Content-Type: application/json")
|
||||||
|
|
||||||
|
if [ -n "$DEPLOY_WEBHOOK_SECRET" ]; then
|
||||||
|
SIGNATURE=$(printf '%s' "$PAYLOAD" | openssl dgst -sha256 -hmac "$DEPLOY_WEBHOOK_SECRET" | sed 's/.*= //')
|
||||||
|
HEADERS+=(-H "X-Hub-Signature-256: sha256=$SIGNATURE")
|
||||||
|
echo "Using HMAC-SHA256 signature verification"
|
||||||
|
else
|
||||||
|
echo "::warning::DEPLOY_WEBHOOK_SECRET not set - webhook calls will not be signed"
|
||||||
|
echo "For security, configure DEPLOY_WEBHOOK_SECRET in your repository settings"
|
||||||
|
fi
|
||||||
|
|
||||||
|
HTTP_CODE=$(curl -sS --connect-timeout 10 --max-time 120 \
|
||||||
|
-o response.txt -w "%{http_code}" \
|
||||||
|
-X POST \
|
||||||
|
"${HEADERS[@]}" \
|
||||||
|
-d "$PAYLOAD" \
|
||||||
|
"$DEPLOY_WEBHOOK_URL") || HTTP_CODE="000"
|
||||||
|
|
||||||
|
echo "Response code: $HTTP_CODE"
|
||||||
|
if [ -s response.txt ]; then
|
||||||
|
cat response.txt
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "$HTTP_CODE" -ge 200 ] && [ "$HTTP_CODE" -lt 300 ]; then
|
||||||
|
echo "status=success" >> "$GITHUB_OUTPUT"
|
||||||
|
else
|
||||||
|
echo "status=failure" >> "$GITHUB_OUTPUT"
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Set deployment status
|
||||||
|
if: always()
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
IMAGE_TAG: ${{ inputs.image_tag }}
|
||||||
|
run: |
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
STATE="${{ steps.webhook.outputs.status }}"
|
||||||
|
|
||||||
|
if [ "$STATE" = "success" ]; then
|
||||||
|
DESCRIPTION=$(jq -nr --arg tag "$IMAGE_TAG" \
|
||||||
|
'"Deployed image \($tag) to staging"')
|
||||||
|
|
||||||
|
gh api "repos/${GITHUB_REPOSITORY}/deployments/${{ steps.deployment.outputs.deployment_id }}/statuses" \
|
||||||
|
-X POST \
|
||||||
|
-f state="success" \
|
||||||
|
-f description="$DESCRIPTION"
|
||||||
|
else
|
||||||
|
gh api "repos/${GITHUB_REPOSITORY}/deployments/${{ steps.deployment.outputs.deployment_id }}/statuses" \
|
||||||
|
-X POST \
|
||||||
|
-f state="failure" \
|
||||||
|
-f description="Deployment failed"
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
79
.github/workflows/deploy-pr.yml
vendored
Normal file
79
.github/workflows/deploy-pr.yml
vendored
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
name: PR Deploy
|
||||||
|
|
||||||
|
on:
|
||||||
|
pull_request_review:
|
||||||
|
types: [submitted]
|
||||||
|
|
||||||
|
concurrency:
|
||||||
|
group: staging-deploy
|
||||||
|
cancel-in-progress: false
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
deployments: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
prepare:
|
||||||
|
name: Prepare deployment
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
if: >
|
||||||
|
github.event.review.state == 'approved' &&
|
||||||
|
github.event.pull_request.head.repo.full_name == github.repository
|
||||||
|
outputs:
|
||||||
|
image_tag: ${{ steps.image.outputs.tag }}
|
||||||
|
sha: ${{ github.event.pull_request.head.sha }}
|
||||||
|
pr_number: ${{ github.event.pull_request.number }}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout PR
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
ref: ${{ github.event.pull_request.head.sha }}
|
||||||
|
|
||||||
|
- name: Get image tag
|
||||||
|
id: image
|
||||||
|
env:
|
||||||
|
PR_NUMBER: ${{ github.event.pull_request.number }}
|
||||||
|
PR_SHA: ${{ github.event.pull_request.head.sha }}
|
||||||
|
run: |
|
||||||
|
IMAGE_TAG="pr-${PR_NUMBER}-${PR_SHA:0:7}"
|
||||||
|
echo "tag=$IMAGE_TAG" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
name: Deploy to staging
|
||||||
|
needs: prepare
|
||||||
|
uses: ./.github/workflows/deploy-core.yml
|
||||||
|
with:
|
||||||
|
image_tag: ${{ needs.prepare.outputs.image_tag }}
|
||||||
|
sha: ${{ needs.prepare.outputs.sha }}
|
||||||
|
description: Deploy PR #${{ needs.prepare.outputs.pr_number }} to staging
|
||||||
|
pr_number: ${{ needs.prepare.outputs.pr_number }}
|
||||||
|
secrets: inherit
|
||||||
|
|
||||||
|
comment:
|
||||||
|
name: Comment deployment status
|
||||||
|
needs: [prepare, deploy]
|
||||||
|
if: always()
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
env:
|
||||||
|
GH_TOKEN: ${{ github.token }}
|
||||||
|
IMAGE_TAG: ${{ needs.prepare.outputs.image_tag }}
|
||||||
|
PR_NUMBER: ${{ needs.prepare.outputs.pr_number }}
|
||||||
|
REF_SHA: ${{ needs.prepare.outputs.sha }}
|
||||||
|
STATUS: ${{ needs.deploy.outputs.status }}
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Comment on PR
|
||||||
|
run: |
|
||||||
|
if [ "$STATUS" = "success" ]; then
|
||||||
|
BODY=$(jq -nr --arg tag "$IMAGE_TAG" --arg sha "$REF_SHA" --arg url "${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/actions/runs/${GITHUB_RUN_ID}" \
|
||||||
|
'"✅ **Staging deployment completed**\n\n🔗 **URL**: https://dev.opensourcepos.org\n📦 **Image Tag**: `\($tag)`\n🔨 **Commit**: \($sha)\n\nView logs: \($url)"')
|
||||||
|
else
|
||||||
|
BODY=$(jq -nr --arg url "${GITHUB_SERVER_URL}/${GITHUB_REPOSITORY}/actions/runs/${GITHUB_RUN_ID}" \
|
||||||
|
'"❌ **Staging deployment failed**\n\nCheck the [workflow logs](\($url)) for details."')
|
||||||
|
fi
|
||||||
|
|
||||||
|
gh api "repos/${GITHUB_REPOSITORY}/issues/${PR_NUMBER}/comments" \
|
||||||
|
-X POST \
|
||||||
|
-f body="$BODY"
|
||||||
23
.github/workflows/deploy.yml
vendored
Normal file
23
.github/workflows/deploy.yml
vendored
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
name: Deploy
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
image_tag:
|
||||||
|
description: 'Docker image tag to deploy (e.g., v3.4.0, latest)'
|
||||||
|
required: true
|
||||||
|
default: 'latest'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
deployments: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
name: Deploy to staging
|
||||||
|
uses: ./.github/workflows/deploy-core.yml
|
||||||
|
with:
|
||||||
|
image_tag: ${{ inputs.image_tag }}
|
||||||
|
sha: ${{ github.sha }}
|
||||||
|
description: Deploy image ${{ inputs.image_tag }}
|
||||||
|
secrets: inherit
|
||||||
1
.github/workflows/main.yml
vendored
1
.github/workflows/main.yml
vendored
@@ -28,7 +28,6 @@ jobs:
|
|||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
php-version:
|
php-version:
|
||||||
- '8.1'
|
|
||||||
- '8.2'
|
- '8.2'
|
||||||
- '8.3'
|
- '8.3'
|
||||||
- '8.4'
|
- '8.4'
|
||||||
|
|||||||
33
.github/workflows/opencode.yml
vendored
33
.github/workflows/opencode.yml
vendored
@@ -1,33 +0,0 @@
|
|||||||
name: opencode
|
|
||||||
|
|
||||||
on:
|
|
||||||
issue_comment:
|
|
||||||
types: [created]
|
|
||||||
pull_request_review_comment:
|
|
||||||
types: [created]
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
opencode:
|
|
||||||
if: |
|
|
||||||
contains(github.event.comment.body, ' /oc') ||
|
|
||||||
startsWith(github.event.comment.body, '/oc') ||
|
|
||||||
contains(github.event.comment.body, ' /opencode') ||
|
|
||||||
startsWith(github.event.comment.body, '/opencode')
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
id-token: write
|
|
||||||
contents: read
|
|
||||||
pull-requests: read
|
|
||||||
issues: read
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v6
|
|
||||||
with:
|
|
||||||
persist-credentials: false
|
|
||||||
|
|
||||||
- name: Run opencode
|
|
||||||
uses: anomalyco/opencode/github@latest
|
|
||||||
env:
|
|
||||||
ANTHROPIC_API_KEY: ${{ secrets.ANTHROPIC_API_KEY }}
|
|
||||||
with:
|
|
||||||
model: anthropic/claude-3-haiku-20240307
|
|
||||||
8
.github/workflows/php-linter.yml
vendored
8
.github/workflows/php-linter.yml
vendored
@@ -12,14 +12,6 @@ jobs:
|
|||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
- name: PHP Lint 8.0
|
|
||||||
uses: dbfx/github-phplint/8.0@master
|
|
||||||
with:
|
|
||||||
folder-to-exclude: "! -path \"./vendor/*\" ! -path \"./folder/excluded/*\""
|
|
||||||
- name: PHP Lint 8.1
|
|
||||||
uses: dbfx/github-phplint/8.1@master
|
|
||||||
with:
|
|
||||||
folder-to-exclude: "! -path \"./vendor/*\" ! -path \"./folder/excluded/*\""
|
|
||||||
- name: PHP Lint 8.2
|
- name: PHP Lint 8.2
|
||||||
uses: dbfx/github-phplint/8.2@master
|
uses: dbfx/github-phplint/8.2@master
|
||||||
with:
|
with:
|
||||||
|
|||||||
3
.github/workflows/phpunit.yml
vendored
3
.github/workflows/phpunit.yml
vendored
@@ -34,7 +34,6 @@ jobs:
|
|||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
php-version:
|
php-version:
|
||||||
- '8.1'
|
|
||||||
- '8.2'
|
- '8.2'
|
||||||
- '8.3'
|
- '8.3'
|
||||||
- '8.4'
|
- '8.4'
|
||||||
@@ -119,4 +118,4 @@ jobs:
|
|||||||
|
|
||||||
- name: Stop MariaDB
|
- name: Stop MariaDB
|
||||||
if: always()
|
if: always()
|
||||||
run: docker stop mysql && docker rm mysql
|
run: docker stop mysql && docker rm mysql
|
||||||
|
|||||||
172
.github/workflows/release.yml
vendored
Normal file
172
.github/workflows/release.yml
vendored
Normal file
@@ -0,0 +1,172 @@
|
|||||||
|
name: Release Version Bump
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
version_type:
|
||||||
|
description: 'Version bump type'
|
||||||
|
required: true
|
||||||
|
type: choice
|
||||||
|
options:
|
||||||
|
- minor
|
||||||
|
- major
|
||||||
|
- patch
|
||||||
|
default: 'minor'
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
prepare-release:
|
||||||
|
name: Prepare Release
|
||||||
|
runs-on: ubuntu-22.04
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
|
|
||||||
|
- name: Get current version
|
||||||
|
id: current_version
|
||||||
|
run: |
|
||||||
|
CURRENT_VERSION=$(grep "application_version" app/Config/App.php | sed "s/.*= '\(.*\)';/\1/g")
|
||||||
|
echo "current_version=$CURRENT_VERSION" >> $GITHUB_OUTPUT
|
||||||
|
echo "Current version: $CURRENT_VERSION"
|
||||||
|
|
||||||
|
- name: Calculate new version
|
||||||
|
id: version
|
||||||
|
run: |
|
||||||
|
CURRENT_VERSION="${{ steps.current_version.outputs.current_version }}"
|
||||||
|
VERSION_TYPE="${{ github.event.inputs.version_type }}"
|
||||||
|
|
||||||
|
# Parse current version
|
||||||
|
MAJOR=$(echo $CURRENT_VERSION | cut -d. -f1)
|
||||||
|
MINOR=$(echo $CURRENT_VERSION | cut -d. -f2)
|
||||||
|
PATCH=$(echo $CURRENT_VERSION | cut -d. -f3)
|
||||||
|
|
||||||
|
# Bump version based on type
|
||||||
|
case $VERSION_TYPE in
|
||||||
|
major)
|
||||||
|
MAJOR=$((MAJOR + 1))
|
||||||
|
MINOR=0
|
||||||
|
PATCH=0
|
||||||
|
;;
|
||||||
|
minor)
|
||||||
|
MINOR=$((MINOR + 1))
|
||||||
|
PATCH=0
|
||||||
|
;;
|
||||||
|
patch)
|
||||||
|
PATCH=$((PATCH + 1))
|
||||||
|
;;
|
||||||
|
esac
|
||||||
|
|
||||||
|
NEW_VERSION="${MAJOR}.${MINOR}.${PATCH}"
|
||||||
|
echo "new_version=$NEW_VERSION" >> $GITHUB_OUTPUT
|
||||||
|
echo "previous_version=$CURRENT_VERSION" >> $GITHUB_OUTPUT
|
||||||
|
echo "New version: $NEW_VERSION (was: $CURRENT_VERSION, type: $VERSION_TYPE)"
|
||||||
|
|
||||||
|
- name: Update version in App.php
|
||||||
|
run: |
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
sed -i "s/public string \\\$application_version = '[^']*';/public string \\\$application_version = '$NEW_VERSION';/" app/Config/App.php
|
||||||
|
echo "Updated app/Config/App.php"
|
||||||
|
|
||||||
|
- name: Update version in package.json
|
||||||
|
run: |
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
sed -i "s/\"version\": \"[^\"]*\",/\"version\": \"$NEW_VERSION\",/" package.json
|
||||||
|
echo "Updated package.json"
|
||||||
|
|
||||||
|
- name: Update version in docker-compose.nginx.yml
|
||||||
|
run: |
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
sed -i "s/jekkos\/opensourcepos:[^ ]*/jekkos\/opensourcepos:$NEW_VERSION/" docker-compose.nginx.yml
|
||||||
|
echo "Updated docker-compose.nginx.yml"
|
||||||
|
|
||||||
|
- name: Update version in README.md
|
||||||
|
run: |
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
# Extract major.minor for the "latest X.Y version" text
|
||||||
|
MAJOR_MINOR=$(echo "$NEW_VERSION" | cut -d. -f1,2)
|
||||||
|
sed -i "s/The latest \`[0-9]*\.[0-9]*\` version/The latest \`${MAJOR_MINOR}\` version/" README.md
|
||||||
|
echo "Updated README.md with version ${MAJOR_MINOR}"
|
||||||
|
|
||||||
|
- name: Generate changelog
|
||||||
|
id: changelog
|
||||||
|
run: |
|
||||||
|
PREVIOUS_VERSION="${{ steps.version.outputs.previous_version }}"
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
|
||||||
|
# Get commits since last version
|
||||||
|
if git rev-parse "$PREVIOUS_VERSION" >/dev/null 2>&1; then
|
||||||
|
COMMITS=$(git log "$PREVIOUS_VERSION"..HEAD --pretty=format:"- %s" --no-merges)
|
||||||
|
else
|
||||||
|
COMMITS=$(git log --pretty=format:"- %s" --no-merges -50)
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Create changelog entry
|
||||||
|
CHANGELOG_FILE="CHANGELOG.md"
|
||||||
|
|
||||||
|
# Create the new version comparison link
|
||||||
|
NEW_LINK="[${NEW_VERSION}]: https://github.com/opensourcepos/opensourcepos/compare/${PREVIOUS_VERSION}...${NEW_VERSION}"
|
||||||
|
|
||||||
|
# Insert new link after [unreleased] line
|
||||||
|
sed -i "/^\[unreleased\]/a $NEW_LINK" "$CHANGELOG_FILE"
|
||||||
|
|
||||||
|
# Update [unreleased] link to start from new version
|
||||||
|
sed -i "s|^\[unreleased\]: .*|\[unreleased\]: https://github.com/opensourcepos/opensourcepos/compare/${NEW_VERSION}...HEAD|" "$CHANGELOG_FILE"
|
||||||
|
|
||||||
|
# Create version header and content using temp file to avoid sed issues with special characters
|
||||||
|
VERSION_DATE=$(date +%Y-%m-%d)
|
||||||
|
VERSION_HEADER="## [$NEW_VERSION] - $VERSION_DATE"
|
||||||
|
|
||||||
|
# Create temp file with changelog entry
|
||||||
|
TMP_FILE=$(mktemp)
|
||||||
|
{
|
||||||
|
echo ""
|
||||||
|
echo "$VERSION_HEADER"
|
||||||
|
echo ""
|
||||||
|
echo "$COMMITS"
|
||||||
|
} > "$TMP_FILE"
|
||||||
|
|
||||||
|
# Insert after Unreleased header
|
||||||
|
sed -i "/^## \[Unreleased\]/r $TMP_FILE" "$CHANGELOG_FILE"
|
||||||
|
rm "$TMP_FILE"
|
||||||
|
|
||||||
|
echo "Updated CHANGELOG.md"
|
||||||
|
echo "Changelog entries:"
|
||||||
|
echo "$COMMITS"
|
||||||
|
|
||||||
|
- name: Update version in issue templates
|
||||||
|
run: |
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
|
||||||
|
# Calculate version to remove (keep 5 versions)
|
||||||
|
PREVIOUS_VERSION="${{ steps.version.outputs.previous_version }}"
|
||||||
|
|
||||||
|
# Bug report template - insert new version after development (unreleased)
|
||||||
|
BUG_TEMPLATE=".github/ISSUE_TEMPLATE/bug report.yml"
|
||||||
|
sed -i "/- development (unreleased)/a\\ - OpenSourcePOS ${NEW_VERSION}" "$BUG_TEMPLATE"
|
||||||
|
# Remove the oldest version (5th version from the end)
|
||||||
|
sed -i "/OpenSourcePOS 3\\.3\\.7/d" "$BUG_TEMPLATE"
|
||||||
|
echo "Updated $BUG_TEMPLATE"
|
||||||
|
|
||||||
|
# Feature request template - insert new version after development (unreleased)
|
||||||
|
FEATURE_TEMPLATE=".github/ISSUE_TEMPLATE/feature_request.yml"
|
||||||
|
sed -i "/- development (unreleased)/a\\ - OpenSourcePOS ${NEW_VERSION}" "$FEATURE_TEMPLATE"
|
||||||
|
# Remove the oldest version (5th version from the end)
|
||||||
|
sed -i "/OpenSourcePOS 3\\.3\\.7/d" "$FEATURE_TEMPLATE"
|
||||||
|
echo "Updated $FEATURE_TEMPLATE"
|
||||||
|
|
||||||
|
- name: Commit version bump
|
||||||
|
run: |
|
||||||
|
git config user.name "github-actions[bot]"
|
||||||
|
git config user.email "github-actions[bot]@users.noreply.github.com"
|
||||||
|
|
||||||
|
NEW_VERSION="${{ steps.version.outputs.new_version }}"
|
||||||
|
|
||||||
|
git add app/Config/App.php package.json docker-compose.nginx.yml CHANGELOG.md README.md .github/ISSUE_TEMPLATE/
|
||||||
|
git commit -m "chore: release version $NEW_VERSION"
|
||||||
|
git push origin HEAD
|
||||||
72
.github/workflows/update-issue-templates.yml
vendored
72
.github/workflows/update-issue-templates.yml
vendored
@@ -1,72 +0,0 @@
|
|||||||
name: Update Issue Templates
|
|
||||||
|
|
||||||
on:
|
|
||||||
release:
|
|
||||||
types: [published]
|
|
||||||
workflow_dispatch:
|
|
||||||
schedule:
|
|
||||||
- cron: '0 0 * * 0'
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-templates:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Fetch releases and update templates
|
|
||||||
env:
|
|
||||||
GH_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
|
||||||
run: |
|
|
||||||
# Fetch releases from GitHub API
|
|
||||||
RELEASES=$(gh api repos/${{ github.repository }}/releases --jq '.[].tag_name' | head -n 10)
|
|
||||||
|
|
||||||
# Create temporary file with options
|
|
||||||
OPTIONS_FILE=$(mktemp)
|
|
||||||
echo " - development (unreleased)" >> "$OPTIONS_FILE"
|
|
||||||
while IFS= read -r release; do
|
|
||||||
echo " - opensourcepos $release" >> "$OPTIONS_FILE"
|
|
||||||
done <<< "$RELEASES"
|
|
||||||
|
|
||||||
update_template() {
|
|
||||||
local template="$1"
|
|
||||||
local template_path=".github/ISSUE_TEMPLATE/$template"
|
|
||||||
|
|
||||||
# Find the line numbers for the OpensourcePOS Version dropdown
|
|
||||||
start_line=$(grep -n "label: OpensourcePOS Version" "$template_path" | cut -d: -f1)
|
|
||||||
|
|
||||||
if [ -z "$start_line" ]; then
|
|
||||||
echo "Could not find OpensourcePOS Version in $template"
|
|
||||||
return 1
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Find the options section and default line
|
|
||||||
options_start=$((start_line + 3))
|
|
||||||
default_line=$(grep -n "default:" "$template_path" | awk -F: -v opts="$options_start" '$1 > opts {print $1; exit}')
|
|
||||||
|
|
||||||
# Create new template file
|
|
||||||
head -n $((options_start - 1)) "$template_path" > "${template_path}.new"
|
|
||||||
cat "$OPTIONS_FILE" >> "${template_path}.new"
|
|
||||||
tail -n +$default_line "$template_path" >> "${template_path}.new"
|
|
||||||
mv "${template_path}.new" "$template_path"
|
|
||||||
|
|
||||||
echo "Updated $template"
|
|
||||||
}
|
|
||||||
|
|
||||||
update_template "bug report.yml"
|
|
||||||
update_template "feature_request.yml"
|
|
||||||
|
|
||||||
- name: Commit and push changes
|
|
||||||
run: |
|
|
||||||
git config user.name "github-actions[bot]"
|
|
||||||
git config user.email "github-actions[bot]@users.noreply.github.com"
|
|
||||||
git add .github/ISSUE_TEMPLATE/*.yml
|
|
||||||
if git diff --staged --quiet; then
|
|
||||||
echo "No changes to commit"
|
|
||||||
else
|
|
||||||
git commit -m "Update issue templates with latest releases [skip ci]"
|
|
||||||
git push
|
|
||||||
fi
|
|
||||||
33
CHANGELOG.md
33
CHANGELOG.md
@@ -1,5 +1,4 @@
|
|||||||
[unreleased]: https://github.com/opensourcepos/opensourcepos/compare/3.4.0...HEAD
|
[unreleased]: https://github.com/opensourcepos/opensourcepos/compare/3.4.1...HEAD
|
||||||
[3.4.2]: https://github.com/opensourcepos/opensourcepos/compare/3.4.1...3.4.2
|
|
||||||
[3.4.1]: https://github.com/opensourcepos/opensourcepos/compare/3.4.0...3.4.1
|
[3.4.1]: https://github.com/opensourcepos/opensourcepos/compare/3.4.0...3.4.1
|
||||||
[3.4.0]: https://github.com/opensourcepos/opensourcepos/compare/3.3.9...3.4.0
|
[3.4.0]: https://github.com/opensourcepos/opensourcepos/compare/3.3.9...3.4.0
|
||||||
[3.3.9]: https://github.com/opensourcepos/opensourcepos/compare/3.3.8...3.3.9
|
[3.3.9]: https://github.com/opensourcepos/opensourcepos/compare/3.3.8...3.3.9
|
||||||
@@ -34,10 +33,36 @@ All notable changes to this project will be documented in this file.
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
## [3.4.0] - 2025-02-06
|
## [3.4.1] - 2025-06-05
|
||||||
|
- Feature: PSR-12 Compliant Indentation by @objecttothis in ([#4196](https://github.com/opensourcepos/opensourcepos/pull/4196))
|
||||||
|
- Add .env to dist zip by @jekkos in ([#4199](https://github.com/opensourcepos/opensourcepos/pull/4199))
|
||||||
|
- Add CI4 coding standards linter ([#3708](https://github.com/opensourcepos/opensourcepos/issues/3708)) by @jekkos in ([#4198](https://github.com/opensourcepos/opensourcepos/pull/4198))
|
||||||
|
- Bump canvg from 3.0.10 to 3.0.11 by @dependabot in ([#4189](https://github.com/opensourcepos/opensourcepos/pull/4189))
|
||||||
|
- Bump jspdf and jspdf-autotable by @dependabot in ([#4190](https://github.com/opensourcepos/opensourcepos/pull/4190))
|
||||||
|
- Feature bump ci to 4.6.0 by @objecttothis in ([#4197](https://github.com/opensourcepos/opensourcepos/pull/4197))
|
||||||
|
- Add Kurdish language option to UI by @BudsieBuds in ([#4210](https://github.com/opensourcepos/opensourcepos/pull/4210))
|
||||||
|
- Convert language ku to ckb by @BudsieBuds in ([#4211](https://github.com/opensourcepos/opensourcepos/pull/4211))
|
||||||
|
- Fix PHP 8.4 errors by @BudsieBuds in ([#4215](https://github.com/opensourcepos/opensourcepos/pull/4215))
|
||||||
|
- Add default bootstrap to themes by @BudsieBuds in ([#4219](https://github.com/opensourcepos/opensourcepos/pull/4219))
|
||||||
|
- Update language names by @BudsieBuds in ([#4218](https://github.com/opensourcepos/opensourcepos/pull/4218))
|
||||||
|
- Update install docs by @BudsieBuds in ([#4217](https://github.com/opensourcepos/opensourcepos/pull/4217))
|
||||||
|
- Convert menu icons to SVG by @BudsieBuds in ([#4220](https://github.com/opensourcepos/opensourcepos/pull/4220))
|
||||||
|
- Enhance license handling by @BudsieBuds in ([#4223](https://github.com/opensourcepos/opensourcepos/pull/4223))
|
||||||
|
- Fix datetime rendering ([#4226](https://github.com/opensourcepos/opensourcepos/issues/4226)) by @jekkos in ([#4227](https://github.com/opensourcepos/opensourcepos/pull/4227))
|
||||||
|
- Fix datetime rendering by @jekkos in ([#4228](https://github.com/opensourcepos/opensourcepos/pull/4228))
|
||||||
|
- Fix null error when sending by email a receipt of a sale that has no invoice by @diego-ramos in ([#4229](https://github.com/opensourcepos/opensourcepos/pull/4229))
|
||||||
|
- Update Receivings.php to save form. by @odiea in ([#4231](https://github.com/opensourcepos/opensourcepos/pull/4231))
|
||||||
|
- Update Cashups.php for ajax cashup total to work. by @odiea in ([#4238](https://github.com/opensourcepos/opensourcepos/pull/4238))
|
||||||
|
- Coding style updates for PSR-12 compliance & improved readability by @BudsieBuds in ([#4204](https://github.com/opensourcepos/opensourcepos/pull/4204))
|
||||||
|
- Fix Codeigniter disallowed characters error with payment types that have accents by @diego-ramos in ([#4232](https://github.com/opensourcepos/opensourcepos/pull/4232))
|
||||||
|
- Fixed broken escape string for success & warning messages by @Franchovy in ([#4253](https://github.com/opensourcepos/opensourcepos/pull/4253))
|
||||||
|
- Bugfix constraint migration fix by @objecttothis in ([#4230](https://github.com/opensourcepos/opensourcepos/pull/4230))
|
||||||
|
- Fix item number lookup in sales/receivings ([#4212](https://github.com/opensourcepos/opensourcepos/issues/4212)) by @jekkos in ([#4250](https://github.com/opensourcepos/opensourcepos/pull/4250))
|
||||||
|
|
||||||
|
## [3.4.0] - 2025-03-23
|
||||||
|
|
||||||
- Translation updates (Spanish, Indonesian, Swedish, Urdu, Chinese, Thai, French, Dutch)
|
- Translation updates (Spanish, Indonesian, Swedish, Urdu, Chinese, Thai, French, Dutch)
|
||||||
- PHP 8.x support
|
- PHP `8.x` support
|
||||||
- Security fixes (XSS, SQLi)
|
- Security fixes (XSS, SQLi)
|
||||||
- Migration to Gulp as buildsystem
|
- Migration to Gulp as buildsystem
|
||||||
- Decimal validation fix
|
- Decimal validation fix
|
||||||
|
|||||||
@@ -1,98 +1,85 @@
|
|||||||
Contributor Covenant Code of Conduct
|
[comment]: # (Contributor Covenant 2.1 - from https://www.contributor-covenant.org/version/2/1/code_of_conduct/code_of_conduct.md)
|
||||||
Our Pledge
|
|
||||||
We as members, contributors, and leaders pledge to make participation in our
|
# Contributor Covenant Code of Conduct
|
||||||
community a harassment-free experience for everyone, regardless of age, body
|
|
||||||
size, visible or invisible disability, ethnicity, sex characteristics, gender
|
## Our Pledge
|
||||||
identity and expression, level of experience, education, socio-economic status,
|
|
||||||
nationality, personal appearance, race, caste, color, religion, or sexual
|
We as members, contributors, and leaders pledge to make participation in our community a harassment-free experience for everyone, regardless of age, body size, visible or invisible disability, ethnicity, sex characteristics, gender identity and expression, level of experience, education, socio-economic status, nationality, personal appearance, race, caste, color, religion, or sexual identity and orientation.
|
||||||
identity and orientation.
|
|
||||||
We pledge to act and interact in ways that contribute to an open, welcoming,
|
We pledge to act and interact in ways that contribute to an open, welcoming, diverse, inclusive, and healthy community.
|
||||||
diverse, inclusive, and healthy community.
|
|
||||||
Our Standards
|
## Our Standards
|
||||||
Examples of behavior that contributes to a positive environment for our
|
|
||||||
community include:
|
Examples of behavior that contributes to a positive environment for our community include:
|
||||||
|
|
||||||
* Demonstrating empathy and kindness toward other people
|
* Demonstrating empathy and kindness toward other people
|
||||||
* Being respectful of differing opinions, viewpoints, and experiences
|
* Being respectful of differing opinions, viewpoints, and experiences
|
||||||
* Giving and gracefully accepting constructive feedback
|
* Giving and gracefully accepting constructive feedback
|
||||||
* Accepting responsibility and apologizing to those affected by our mistakes,
|
* Accepting responsibility and apologizing to those affected by our mistakes, and learning from the experience
|
||||||
and learning from the experience
|
* Focusing on what is best not just for us as individuals, but for the overall community
|
||||||
* Focusing on what is best not just for us as individuals, but for the overall
|
|
||||||
community
|
|
||||||
|
|
||||||
Examples of unacceptable behavior include:
|
Examples of unacceptable behavior include:
|
||||||
|
|
||||||
* The use of sexualized language or imagery, and sexual attention or advances of
|
* The use of sexualized language or imagery, and sexual attention or advances of any kind
|
||||||
any kind
|
|
||||||
* Trolling, insulting or derogatory comments, and personal or political attacks
|
* Trolling, insulting or derogatory comments, and personal or political attacks
|
||||||
* Public or private harassment
|
* Public or private harassment
|
||||||
* Publishing others’ private information, such as a physical or email address,
|
* Publishing others' private information, such as a physical or email address, without their explicit permission
|
||||||
without their explicit permission
|
* Other conduct which could reasonably be considered inappropriate in a professional setting
|
||||||
* Other conduct which could reasonably be considered inappropriate in a
|
|
||||||
professional setting
|
|
||||||
|
|
||||||
Enforcement Responsibilities
|
## Enforcement Responsibilities
|
||||||
Community leaders are responsible for clarifying and enforcing our standards of
|
|
||||||
acceptable behavior and will take appropriate and fair corrective action in
|
|
||||||
response to any behavior that they deem inappropriate, threatening, offensive,
|
|
||||||
or harmful.
|
|
||||||
Community leaders have the right and responsibility to remove, edit, or reject
|
|
||||||
comments, commits, code, wiki edits, issues, and other contributions that are
|
|
||||||
not aligned to this Code of Conduct, and will communicate reasons for moderation
|
|
||||||
decisions when appropriate.
|
|
||||||
Scope
|
|
||||||
This Code of Conduct applies within all community spaces, and also applies when
|
|
||||||
an individual is officially representing the community in public spaces.
|
|
||||||
Examples of representing our community include using an official email address,
|
|
||||||
posting via an official social media account, or acting as an appointed
|
|
||||||
representative at an online or offline event.
|
|
||||||
Enforcement
|
|
||||||
Instances of abusive, harassing, or otherwise unacceptable behavior may be
|
|
||||||
reported to the community leaders responsible for enforcement at
|
|
||||||
[INSERT CONTACT METHOD].
|
|
||||||
All complaints will be reviewed and investigated promptly and fairly.
|
|
||||||
All community leaders are obligated to respect the privacy and security of the
|
|
||||||
reporter of any incident.
|
|
||||||
Enforcement Guidelines
|
|
||||||
Community leaders will follow these Community Impact Guidelines in determining
|
|
||||||
the consequences for any action they deem in violation of this Code of Conduct:
|
|
||||||
1. Correction
|
|
||||||
Community Impact: Use of inappropriate language or other behavior deemed
|
|
||||||
unprofessional or unwelcome in the community.
|
|
||||||
Consequence: A private, written warning from community leaders, providing
|
|
||||||
clarity around the nature of the violation and an explanation of why the
|
|
||||||
behavior was inappropriate. A public apology may be requested.
|
|
||||||
2. Warning
|
|
||||||
Community Impact: A violation through a single incident or series of
|
|
||||||
actions.
|
|
||||||
Consequence: A warning with consequences for continued behavior. No
|
|
||||||
interaction with the people involved, including unsolicited interaction with
|
|
||||||
those enforcing the Code of Conduct, for a specified period of time. This
|
|
||||||
includes avoiding interactions in community spaces as well as external channels
|
|
||||||
like social media. Violating these terms may lead to a temporary or permanent
|
|
||||||
ban.
|
|
||||||
3. Temporary Ban
|
|
||||||
Community Impact: A serious violation of community standards, including
|
|
||||||
sustained inappropriate behavior.
|
|
||||||
Consequence: A temporary ban from any sort of interaction or public
|
|
||||||
communication with the community for a specified period of time. No public or
|
|
||||||
private interaction with the people involved, including unsolicited interaction
|
|
||||||
with those enforcing the Code of Conduct, is allowed during this period.
|
|
||||||
Violating these terms may lead to a permanent ban.
|
|
||||||
4. Permanent Ban
|
|
||||||
Community Impact: Demonstrating a pattern of violation of community
|
|
||||||
standards, including sustained inappropriate behavior, harassment of an
|
|
||||||
individual, or aggression toward or disparagement of classes of individuals.
|
|
||||||
Consequence: A permanent ban from any sort of public interaction within the
|
|
||||||
community.
|
|
||||||
Attribution
|
|
||||||
This Code of Conduct is adapted from the Contributor Covenant,
|
|
||||||
version 2.1, available at
|
|
||||||
https://www.contributor-covenant.org/version/2/1/code_of_conduct.html.
|
|
||||||
Community Impact Guidelines were inspired by
|
|
||||||
Mozilla’s code of conduct enforcement ladder.
|
|
||||||
For answers to common questions about this code of conduct, see the FAQ at
|
|
||||||
https://www.contributor-covenant.org/faq. Translations are available at
|
|
||||||
https://www.contributor-covenant.org/translations.
|
|
||||||
|
|
||||||
|
Community leaders are responsible for clarifying and enforcing our standards of acceptable behavior and will take appropriate and fair corrective action in response to any behavior that they deem inappropriate, threatening, offensive, or harmful.
|
||||||
|
|
||||||
|
Community leaders have the right and responsibility to remove, edit, or reject comments, commits, code, wiki edits, issues, and other contributions that are not aligned to this Code of Conduct, and will communicate reasons for moderation decisions when appropriate.
|
||||||
|
|
||||||
|
## Scope
|
||||||
|
|
||||||
|
This Code of Conduct applies within all community spaces, and also applies when an individual is officially representing the community in public spaces. Examples of representing our community include using an official e-mail address, posting via an official social media account, or acting as an appointed representative at an online or offline event.
|
||||||
|
|
||||||
|
## Enforcement
|
||||||
|
|
||||||
|
Instances of abusive, harassing, or otherwise unacceptable behavior may be reported to the community leaders responsible for enforcement at [INSERT CONTACT METHOD]. All complaints will be reviewed and investigated promptly and fairly.
|
||||||
|
|
||||||
|
All community leaders are obligated to respect the privacy and security of the reporter of any incident.
|
||||||
|
|
||||||
|
## Enforcement Guidelines
|
||||||
|
|
||||||
|
Community leaders will follow these Community Impact Guidelines in determining the consequences for any action they deem in violation of this Code of Conduct:
|
||||||
|
|
||||||
|
### 1. Correction
|
||||||
|
|
||||||
|
**Community Impact**: Use of inappropriate language or other behavior deemed unprofessional or unwelcome in the community.
|
||||||
|
|
||||||
|
**Consequence**: A private, written warning from community leaders, providing clarity around the nature of the violation and an explanation of why the behavior was inappropriate. A public apology may be requested.
|
||||||
|
|
||||||
|
### 2. Warning
|
||||||
|
|
||||||
|
**Community Impact**: A violation through a single incident or series of actions.
|
||||||
|
|
||||||
|
**Consequence**: A warning with consequences for continued behavior. No interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, for a specified period of time. This includes avoiding interactions in community spaces as well as external channels like social media. Violating these terms may lead to a temporary or permanent ban.
|
||||||
|
|
||||||
|
### 3. Temporary Ban
|
||||||
|
|
||||||
|
**Community Impact**: A serious violation of community standards, including sustained inappropriate behavior.
|
||||||
|
|
||||||
|
**Consequence**: A temporary ban from any sort of interaction or public communication with the community for a specified period of time. No public or private interaction with the people involved, including unsolicited interaction with those enforcing the Code of Conduct, is allowed during this period. Violating these terms may lead to a permanent ban.
|
||||||
|
|
||||||
|
### 4. Permanent Ban
|
||||||
|
|
||||||
|
**Community Impact**: Demonstrating a pattern of violation of community standards, including sustained inappropriate behavior, harassment of an individual, or aggression toward or disparagement of classes of individuals.
|
||||||
|
|
||||||
|
**Consequence**: A permanent ban from any sort of public interaction within the community.
|
||||||
|
|
||||||
|
## Attribution
|
||||||
|
|
||||||
|
This Code of Conduct is adapted from the [Contributor Covenant][homepage], version 2.1, available at [https://www.contributor-covenant.org/version/2/1/code_of_conduct.html][v2.1].
|
||||||
|
|
||||||
|
Community Impact Guidelines were inspired by [Mozilla's code of conduct enforcement ladder][Mozilla CoC].
|
||||||
|
|
||||||
|
For answers to common questions about this code of conduct, see the FAQ at [https://www.contributor-covenant.org/faq][FAQ]. Translations are available at [https://www.contributor-covenant.org/translations][translations].
|
||||||
|
|
||||||
|
[homepage]: https://www.contributor-covenant.org
|
||||||
|
[v2.1]: https://www.contributor-covenant.org/version/2/1/code_of_conduct.html
|
||||||
|
[Mozilla CoC]: https://github.com/mozilla/diversity
|
||||||
|
[FAQ]: https://www.contributor-covenant.org/faq
|
||||||
|
[translations]: https://www.contributor-covenant.org/translations
|
||||||
|
|||||||
@@ -13,7 +13,8 @@ RUN echo "date.timezone = \"\${PHP_TIMEZONE}\"" > /usr/local/etc/php/conf.d/time
|
|||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY --chown=www-data:www-data . /app
|
COPY --chown=www-data:www-data . /app
|
||||||
RUN chmod 770 /app/writable/uploads /app/writable/logs /app/writable/cache \
|
RUN chmod 750 /app/writable/logs /app/writable/uploads /app/writable/cache /app/public/uploads /app/public/uploads/item_pics \
|
||||||
|
&& chmod 640 /app/writable/uploads/importCustomers.csv \
|
||||||
&& ln -s /app/*[^public] /var/www \
|
&& ln -s /app/*[^public] /var/www \
|
||||||
&& rm -rf /var/www/html \
|
&& rm -rf /var/www/html \
|
||||||
&& ln -nsf /app/public /var/www/html
|
&& ln -nsf /app/public /var/www/html
|
||||||
|
|||||||
@@ -1,3 +0,0 @@
|
|||||||
FROM php:8.4-cli
|
|
||||||
RUN apt-get update && apt-get install -y libicu-dev && docker-php-ext-install intl
|
|
||||||
WORKDIR /app
|
|
||||||
38
INSTALL.md
38
INSTALL.md
@@ -1,6 +1,6 @@
|
|||||||
## Server Requirements
|
## Server Requirements
|
||||||
|
|
||||||
- PHP version `8.1` to `8.4` are supported, PHP version `≤7.4` is NOT supported. Please note that PHP needs to have the extensions `php-json`, `php-gd`, `php-bcmath`, `php-intl`, `php-openssl`, `php-mbstring`, `php-curl` and `php-xml` installed and enabled. An unstable master build can be downloaded in the releases section.
|
- PHP version `8.2` to `8.4` are supported, PHP version `≤ 8.1` is NOT supported. Please note that PHP needs to have the extensions `php-json`, `php-gd`, `php-bcmath`, `php-intl`, `php-openssl`, `php-mbstring`, `php-curl` and `php-xml` installed and enabled. An unstable master build can be downloaded in the releases section.
|
||||||
- MySQL `5.7` is supported, also MariaDB replacement `10.x` is supported and might offer better performance.
|
- MySQL `5.7` is supported, also MariaDB replacement `10.x` is supported and might offer better performance.
|
||||||
- Apache `2.4` is supported. Nginx should work fine too, see [wiki page here](https://github.com/opensourcepos/opensourcepos/wiki/Local-Deployment-using-LEMP).
|
- Apache `2.4` is supported. Nginx should work fine too, see [wiki page here](https://github.com/opensourcepos/opensourcepos/wiki/Local-Deployment-using-LEMP).
|
||||||
- Raspberry PI based installations proved to work, see [wiki page here](<https://github.com/opensourcepos/opensourcepos/wiki/Installing-on-Raspberry-PI---Orange-PI-(Headless-OSPOS)>).
|
- Raspberry PI based installations proved to work, see [wiki page here](<https://github.com/opensourcepos/opensourcepos/wiki/Installing-on-Raspberry-PI---Orange-PI-(Headless-OSPOS)>).
|
||||||
@@ -8,26 +8,36 @@
|
|||||||
|
|
||||||
## Security Configuration
|
## Security Configuration
|
||||||
|
|
||||||
### Allowed Hostnames (Required for Production)
|
### Allowed Hostnames (REQUIRED for Production)
|
||||||
|
|
||||||
OpenSourcePOS validates the Host header against a whitelist to prevent Host Header Injection attacks (GHSA-jchf-7hr6-h4f3). **You must configure this for production deployments.**
|
⚠️ **CRITICAL**: OpenSourcePOS validates the Host header to prevent Host Header Injection attacks (GHSA-jchf-7hr6-h4f3). **You MUST configure `app.allowedHostnames` for production deployments. If not configured, the application will fail to start.**
|
||||||
|
|
||||||
Add the following to your `.env` file:
|
**Add to your `.env` file:**
|
||||||
|
|
||||||
```
|
```bash
|
||||||
app.allowedHostnames.0 = 'yourdomain.com'
|
# Comma-separated list of allowed hostnames (no protocols or ports)
|
||||||
app.allowedHostnames.1 = 'www.yourdomain.com'
|
app.allowedHostnames = 'yourdomain.com,www.yourdomain.com'
|
||||||
```
|
```
|
||||||
|
|
||||||
**For local development**, use:
|
**For local development:**
|
||||||
```
|
|
||||||
app.allowedHostnames.0 = 'localhost'
|
```bash
|
||||||
|
app.allowedHostnames = 'localhost'
|
||||||
```
|
```
|
||||||
|
|
||||||
If `allowedHostnames` is not configured:
|
**If you see this error at startup:**
|
||||||
1. A security warning will be logged
|
|
||||||
2. The application will fall back to 'localhost' as the hostname
|
```text
|
||||||
3. This means URLs generated by the application (links, redirects, etc.) will point to 'localhost'
|
RuntimeException: Security: allowedHostnames is not configured.
|
||||||
|
```
|
||||||
|
|
||||||
|
**Solution**: Add `app.allowedHostnames` to your `.env` file with your domain(s).
|
||||||
|
|
||||||
|
**Why this matters:**
|
||||||
|
- Prevents Host Header Injection attacks (GHSA-jchf-7hr6-h4f3)
|
||||||
|
- Ensures URLs are generated with the correct domain
|
||||||
|
- Security advisory: https://github.com/opensourcepos/opensourcepos/security/advisories/GHSA-jchf-7hr6-h4f3
|
||||||
|
- Fixes issue #4480: .env configuration now works via comma-separated values
|
||||||
|
|
||||||
### HTTPS Behind Proxy
|
### HTTPS Behind Proxy
|
||||||
|
|
||||||
|
|||||||
@@ -102,11 +102,11 @@ NOTE: If you're running non-release code, please make sure you always run the la
|
|||||||
|
|
||||||
- If you have suhosin installed and face an issue with CSRF, please make sure you read [issue #1492](https://github.com/opensourcepos/opensourcepos/issues/1492).
|
- If you have suhosin installed and face an issue with CSRF, please make sure you read [issue #1492](https://github.com/opensourcepos/opensourcepos/issues/1492).
|
||||||
|
|
||||||
- PHP `≥ 8.1` is required to run this app.
|
- PHP `≥ 8.2` is required to run this app.
|
||||||
|
|
||||||
## 🏃 Keep the Machine Running
|
## 🏃 Keep the Machine Running
|
||||||
|
|
||||||
If you like our project, please consider buying us a coffee through the button below so we can keep adding features.
|
If you like our project, please consider buying us a coffee through the button below so we can keep adding features. Please star the project if you like it!
|
||||||
|
|
||||||
[](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=MUN6AEG7NY6H8)\
|
[](https://www.paypal.com/cgi-bin/webscr?cmd=_s-xclick&hosted_button_id=MUN6AEG7NY6H8)\
|
||||||
Or refer to the [FUNDING.yml](.github/FUNDING.yml) file.
|
Or refer to the [FUNDING.yml](.github/FUNDING.yml) file.
|
||||||
|
|||||||
131
SECURITY.md
131
SECURITY.md
@@ -5,8 +5,9 @@
|
|||||||
- [Supported Versions](#supported-versions)
|
- [Supported Versions](#supported-versions)
|
||||||
- [Security Advisories](#security-advisories)
|
- [Security Advisories](#security-advisories)
|
||||||
- [Reporting a Vulnerability](#reporting-a-vulnerability)
|
- [Reporting a Vulnerability](#reporting-a-vulnerability)
|
||||||
|
- [Disclosure Process](#disclosure-process)
|
||||||
|
|
||||||
<!-- END doctoc generated TOC please keep comment here to allow auto update -->
|
<!-- END doctoc generated TOC please keep comment here to allow update -->
|
||||||
|
|
||||||
# Security Policy
|
# Security Policy
|
||||||
|
|
||||||
@@ -21,26 +22,116 @@ We release patches for security vulnerabilities.
|
|||||||
|
|
||||||
## Security Advisories
|
## Security Advisories
|
||||||
|
|
||||||
The following security vulnerabilities have been published:
|
For a complete list of published and draft security advisories with CVE details, see our [GitHub Security Advisories page](https://github.com/opensourcepos/opensourcepos/security/advisories).
|
||||||
|
|
||||||
### High Severity
|
|
||||||
|
|
||||||
| CVE | Vulnerability | CVSS | Published | Fixed In | Credit |
|
|
||||||
|-----|--------------|------|-----------|----------|--------|
|
|
||||||
| [CVE-2025-68434](https://github.com/opensourcepos/opensourcepos/security/advisories/GHSA-wjm4-hfwg-5w5r) | CSRF leading to Admin Creation | 8.8 | 2025-12-17 | 3.4.2 | @Nixon-H, @jekkos |
|
|
||||||
| [CVE-2025-68147](https://github.com/opensourcepos/opensourcepos/security/advisories/GHSA-xgr7-7pvw-fpmh) | Stored XSS in Return Policy | 8.1 | 2025-12-17 | 3.4.2 | @Nixon-H, @jekkos |
|
|
||||||
| [CVE-2025-66924](https://github.com/opensourcepos/opensourcepos/security/advisories/GHSA-gv8j-f6gq-g59m) | Stored XSS in Item Kits | 7.2 | 2026-03-04 | 3.4.2 | @hungnqdz, @omkaryepre |
|
|
||||||
|
|
||||||
### Medium Severity
|
|
||||||
|
|
||||||
| CVE | Vulnerability | CVSS | Published | Fixed In | Credit |
|
|
||||||
|-----|--------------|------|-----------|----------|--------|
|
|
||||||
| [CVE-2025-68658](https://github.com/opensourcepos/opensourcepos/security/advisories/GHSA-32r8-8r9r-9chw) | Stored XSS in Company Name | 4.3 | 2026-01-13 | 3.4.2 | @hungnqdz |
|
|
||||||
|
|
||||||
For a complete list including draft advisories, see our [GitHub Security Advisories page](https://github.com/opensourcepos/opensourcepos/security/advisories).
|
|
||||||
|
|
||||||
## Reporting a Vulnerability
|
## Reporting a Vulnerability
|
||||||
|
|
||||||
Please report (suspected) security vulnerabilities to **[jeroen@steganos.dev](mailto:jeroen@steganos.dev)**.
|
**Option 1: GitHub Security Advisory (Preferred)**
|
||||||
|
|
||||||
You will receive a response from us within 48 hours. If the issue is confirmed, we will release a patch as soon as possible depending on complexity but historically within a few days.
|
1. Create a draft security advisory directly on GitHub:
|
||||||
|
- Go to https://github.com/opensourcepos/opensourcepos/security/advisories
|
||||||
|
- Click "New draft security advisory"
|
||||||
|
- Fill in the vulnerability details using our [template below](#vulnerability-template)
|
||||||
|
- Submit as **draft** (not published)
|
||||||
|
|
||||||
|
2. Notify us for triage:
|
||||||
|
- Send an email to **[jeroen@steganos.dev](mailto:jeroen@steganos.dev)** with:
|
||||||
|
- Subject: `[GHSA] Brief description of vulnerability`
|
||||||
|
- Link to the draft advisory
|
||||||
|
- Brief summary
|
||||||
|
|
||||||
|
**Option 2: Email Report**
|
||||||
|
|
||||||
|
Send vulnerability details to **[jeroen@steganos.dev](mailto:jeroen@steganos.dev)**.
|
||||||
|
|
||||||
|
You will receive a response within 48 hours. Confirmed vulnerabilities will be patched within a few days depending on complexity.
|
||||||
|
|
||||||
|
## Disclosure Process
|
||||||
|
|
||||||
|
### Timeline
|
||||||
|
|
||||||
|
| Step | Timeline | Action |
|
||||||
|
|------|----------|--------|
|
||||||
|
| 1. Report received | Day 0 | We acknowledge within 48 hours |
|
||||||
|
| 2. Triage & confirmation | Day 1-3 | We validate the vulnerability |
|
||||||
|
| 3. Fix development | Day 3-7 | We develop and test the fix |
|
||||||
|
| 4. Patch release | Day 7-10 | We release a security patch |
|
||||||
|
| 5. CVE request | Day 7-14 | We request CVE from GitHub (if applicable) |
|
||||||
|
| 6. Advisory published | Day 14 | We publish the advisory with credit |
|
||||||
|
| 7. Public disclosure | Day 14+ | Full disclosure after patch release |
|
||||||
|
|
||||||
|
### CVE Process
|
||||||
|
|
||||||
|
**We request CVE identifiers through GitHub's security advisory system.** This is the preferred and easiest method:
|
||||||
|
|
||||||
|
1. After we confirm and fix the vulnerability, we'll request a CVE through GitHub
|
||||||
|
2. GitHub coordinates with MITRE on our behalf
|
||||||
|
3. The CVE is automatically linked to the advisory
|
||||||
|
4. You'll be credited as the reporter in the published advisory
|
||||||
|
|
||||||
|
**Already have a CVE?** If you've already obtained a CVE from another source (e.g., VulDB, CVE.MITRE.ORG), please include it in your report or advisory. We'll update our advisory to reference the existing CVE.
|
||||||
|
|
||||||
|
### No Bug Bounty Program
|
||||||
|
|
||||||
|
**Important:** Open Source Point of Sale does not offer a bug bounty program.
|
||||||
|
|
||||||
|
- All security research and vulnerability triage is done on a **voluntary basis** in our free time
|
||||||
|
- We do not offer monetary rewards for vulnerability reports
|
||||||
|
- We do credit reporters in published advisories (unless anonymity is requested)
|
||||||
|
- We greatly appreciate the security research community's efforts to help improve project security
|
||||||
|
|
||||||
|
### Security Best Practices for Researchers
|
||||||
|
|
||||||
|
- **Do not** access, modify, or delete data that doesn't belong to you
|
||||||
|
- **Do not** perform denial of service attacks
|
||||||
|
- **Do not** publicly disclose vulnerabilities before we've had time to fix them
|
||||||
|
- **Do** provide sufficient information to reproduce the vulnerability
|
||||||
|
- **Do** allow us reasonable time to fix before public disclosure
|
||||||
|
- **Do** report through official channels (GitHub advisories or email)
|
||||||
|
|
||||||
|
### Vulnerability Template
|
||||||
|
|
||||||
|
When creating a draft advisory, please include:
|
||||||
|
|
||||||
|
```
|
||||||
|
## Summary
|
||||||
|
[Brief description of the vulnerability]
|
||||||
|
|
||||||
|
## Impact
|
||||||
|
- **Confidentiality:** [High/Medium/Low - what data can be exposed]
|
||||||
|
- **Integrity:** [High/Medium/Low - what can be modified]
|
||||||
|
- **Availability:** [High/Medium/Low - service disruption potential]
|
||||||
|
- **Privilege Required:** [None/Low/High - authentication level needed]
|
||||||
|
- **CVSS v3.1:** [Score] ([Vector string])
|
||||||
|
|
||||||
|
## Details
|
||||||
|
[Technical details about the vulnerability]
|
||||||
|
|
||||||
|
**Affected Code:**
|
||||||
|
```php
|
||||||
|
// Path to affected file and vulnerable code
|
||||||
|
```
|
||||||
|
|
||||||
|
**Attack Vector:**
|
||||||
|
[How an attacker can exploit this]
|
||||||
|
|
||||||
|
## Proof of Concept
|
||||||
|
```bash
|
||||||
|
# Steps to reproduce
|
||||||
|
```
|
||||||
|
|
||||||
|
## Patch
|
||||||
|
[Suggested fix or approach]
|
||||||
|
|
||||||
|
## Affected Versions
|
||||||
|
- OpenSourcePOS X.Y.Z and earlier
|
||||||
|
|
||||||
|
## Credit
|
||||||
|
[Your GitHub username or preferred name]
|
||||||
|
```
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Thank you to all security researchers who have contributed to making Open Source Point of Sale more secure.** Your voluntary efforts help protect thousands of users worldwide and contribute to a safer, more trustworthy free and open-source software ecosystem. We deeply appreciate your responsible disclosure and the time you invest in improving our project.
|
||||||
|
|
||||||
|
If you've reported a vulnerability and would like to discuss CVE coordination or have questions about the process, please reach out to us at [jeroen@steganos.dev](mailto:jeroen@steganos.dev).
|
||||||
@@ -55,21 +55,13 @@ class App extends BaseConfig
|
|||||||
public string $baseURL; // Defined in the constructor
|
public string $baseURL; // Defined in the constructor
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Allowed Hostnames for the Site URL.
|
* Allowed Hostnames in the Site URL other than the hostname in the baseURL.
|
||||||
|
* If you want to accept multiple Hostnames, set this.
|
||||||
|
*
|
||||||
|
* Or via environment variable (useful for Docker/Compose):
|
||||||
|
* ALLOWED_HOSTNAMES=example.com,www.example.com
|
||||||
*
|
*
|
||||||
* Security: This is used to validate the HTTP Host header to prevent
|
* ['media.example.com', 'accounts.example.com']
|
||||||
* Host Header Injection attacks. If the Host header doesn't match
|
|
||||||
* an entry in this list, the request will use the first allowed hostname.
|
|
||||||
*
|
|
||||||
* IMPORTANT: This MUST be configured for production deployments.
|
|
||||||
* If empty, the application will fall back to 'localhost'.
|
|
||||||
*
|
|
||||||
* Configure via .env file:
|
|
||||||
* app.allowedHostnames.0 = 'example.com'
|
|
||||||
* app.allowedHostnames.1 = 'www.example.com'
|
|
||||||
*
|
|
||||||
* For local development:
|
|
||||||
* app.allowedHostnames.0 = 'localhost'
|
|
||||||
*
|
*
|
||||||
* @var list<string>
|
* @var list<string>
|
||||||
*/
|
*/
|
||||||
@@ -125,7 +117,7 @@ class App extends BaseConfig
|
|||||||
| DO NOT CHANGE THIS UNLESS YOU FULLY UNDERSTAND THE REPERCUSSIONS!!
|
| DO NOT CHANGE THIS UNLESS YOU FULLY UNDERSTAND THE REPERCUSSIONS!!
|
||||||
|
|
|
|
||||||
*/
|
*/
|
||||||
public string $permittedURIChars = 'a-z 0-9~%.:_\-=';
|
public string $permittedURIChars = 'a-z 0-9~%.:_\-';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* --------------------------------------------------------------------------
|
* --------------------------------------------------------------------------
|
||||||
@@ -286,13 +278,28 @@ class App extends BaseConfig
|
|||||||
* @see http://www.html5rocks.com/en/tutorials/security/content-security-policy/
|
* @see http://www.html5rocks.com/en/tutorials/security/content-security-policy/
|
||||||
* @see http://www.w3.org/TR/CSP/
|
* @see http://www.w3.org/TR/CSP/
|
||||||
*/
|
*/
|
||||||
public bool $CSPEnabled = false; // TODO: Currently CSP3 tags are not supported so enabling this causes problems with script-src-elem, style-src-attr and style-src-elem
|
public bool $CSPEnabled = false;
|
||||||
|
|
||||||
public function __construct()
|
public function __construct()
|
||||||
{
|
{
|
||||||
parent::__construct();
|
parent::__construct();
|
||||||
|
|
||||||
|
// Solution for CodeIgniter 4 limitation: arrays cannot be set from .env
|
||||||
|
// See: https://github.com/codeigniter4/CodeIgniter4/issues/7311
|
||||||
|
// Support both: app.allowedHostnames (from .env) and ALLOWED_HOSTNAMES (from environment/Docker)
|
||||||
|
$envAllowedHostnames = getenv('ALLOWED_HOSTNAMES');
|
||||||
|
if ($envAllowedHostnames === false || trim($envAllowedHostnames) === '') {
|
||||||
|
$envAllowedHostnames = getenv('app.allowedHostnames');
|
||||||
|
}
|
||||||
|
if ($envAllowedHostnames !== false && trim($envAllowedHostnames) !== '') {
|
||||||
|
$this->allowedHostnames = array_values(array_filter(
|
||||||
|
array_map('trim', explode(',', $envAllowedHostnames)),
|
||||||
|
static fn (string $hostname): bool => $hostname !== ''
|
||||||
|
));
|
||||||
|
}
|
||||||
|
|
||||||
$this->https_on = (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on') || (isset($_ENV['FORCE_HTTPS']) && $_ENV['FORCE_HTTPS'] == 'true');
|
$this->https_on = (isset($_SERVER['HTTPS']) && $_SERVER['HTTPS'] == 'on') || (isset($_ENV['FORCE_HTTPS']) && $_ENV['FORCE_HTTPS'] == 'true');
|
||||||
|
|
||||||
$host = $this->getValidHost();
|
$host = $this->getValidHost();
|
||||||
$this->baseURL = $this->https_on ? 'https' : 'http';
|
$this->baseURL = $this->https_on ? 'https' : 'http';
|
||||||
$this->baseURL .= '://' . $host . '/';
|
$this->baseURL .= '://' . $host . '/';
|
||||||
@@ -301,23 +308,40 @@ class App extends BaseConfig
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates and returns a trusted hostname.
|
* Validates and returns a trusted hostname.
|
||||||
*
|
*
|
||||||
* Security: Prevents Host Header Injection attacks (GHSA-jchf-7hr6-h4f3)
|
* Security: Prevents Host Header Injection attacks (GHSA-jchf-7hr6-h4f3)
|
||||||
* by validating the HTTP_HOST against a whitelist of allowed hostnames.
|
* by validating the HTTP_HOST against a whitelist of allowed hostnames.
|
||||||
*
|
*
|
||||||
|
* In production: Fails fast if allowedHostnames is not configured.
|
||||||
|
* In development: Allows localhost fallback with an error log.
|
||||||
|
*
|
||||||
* @return string A validated hostname
|
* @return string A validated hostname
|
||||||
|
* @throws \RuntimeException If allowedHostnames is not configured in production
|
||||||
*/
|
*/
|
||||||
private function getValidHost(): string
|
private function getValidHost(): string
|
||||||
{
|
{
|
||||||
$httpHost = $_SERVER['HTTP_HOST'] ?? 'localhost';
|
$httpHost = $_SERVER['HTTP_HOST'] ?? 'localhost';
|
||||||
|
|
||||||
|
// Determine environment
|
||||||
|
// CodeIgniter's test bootstrap sets $_SERVER['CI_ENVIRONMENT'] = 'testing'
|
||||||
|
// Check $_SERVER first, then $_ENV, then fall back to 'production'
|
||||||
|
$environment = $_SERVER['CI_ENVIRONMENT'] ?? $_ENV['CI_ENVIRONMENT'] ?? getenv('CI_ENVIRONMENT') ?: 'production';
|
||||||
|
|
||||||
if (empty($this->allowedHostnames)) {
|
if (empty($this->allowedHostnames)) {
|
||||||
log_message('warning',
|
$errorMessage =
|
||||||
'Security: allowedHostnames is not configured. ' .
|
'Security: allowedHostnames is not configured. ' .
|
||||||
'Host header injection protection is disabled. ' .
|
'Host header injection protection is disabled. ' .
|
||||||
'Please set app.allowedHostnames in your .env file. ' .
|
'Set app.allowedHostnames in your .env file or ALLOWED_HOSTNAMES environment variable. ' .
|
||||||
'Received Host: ' . $httpHost
|
'Example: app.allowedHostnames = "example.com,www.example.com" ' .
|
||||||
);
|
'Received Host: ' . $httpHost;
|
||||||
|
|
||||||
|
// Production: Fail explicitly to prevent silent security vulnerabilities
|
||||||
|
// Testing and development: Allow localhost fallback
|
||||||
|
if ($environment === 'production') {
|
||||||
|
throw new \RuntimeException($errorMessage);
|
||||||
|
}
|
||||||
|
|
||||||
|
log_message('error', $errorMessage . ' Using localhost fallback (development only).');
|
||||||
return 'localhost';
|
return 'localhost';
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -325,7 +349,8 @@ class App extends BaseConfig
|
|||||||
return $httpHost;
|
return $httpHost;
|
||||||
}
|
}
|
||||||
|
|
||||||
log_message('warning',
|
// Host not in whitelist - use first configured hostname as fallback
|
||||||
|
log_message('warning',
|
||||||
'Security: Rejected HTTP_HOST "' . $httpHost . '" - not in allowedHostnames whitelist. ' .
|
'Security: Rejected HTTP_HOST "' . $httpHost . '" - not in allowedHostnames whitelist. ' .
|
||||||
'Using fallback: ' . $this->allowedHostnames[0]
|
'Using fallback: ' . $this->allowedHostnames[0]
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -17,8 +17,6 @@ use CodeIgniter\Config\AutoloadConfig;
|
|||||||
*
|
*
|
||||||
* NOTE: This class is required prior to Autoloader instantiation,
|
* NOTE: This class is required prior to Autoloader instantiation,
|
||||||
* and does not extend BaseConfig.
|
* and does not extend BaseConfig.
|
||||||
*
|
|
||||||
* @immutable
|
|
||||||
*/
|
*/
|
||||||
class Autoload extends AutoloadConfig
|
class Autoload extends AutoloadConfig
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1,23 +1,38 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
|
/*
|
||||||
|
* The environment testing is reserved for PHPUnit testing. It has special
|
||||||
|
* conditions built into the framework at various places to assist with that.
|
||||||
|
* You can’t use it for your development.
|
||||||
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| ERROR DISPLAY
|
| ERROR DISPLAY
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
*/
|
| In development, we want to show as many errors as possible to help
|
||||||
|
| make sure they don't make it to production. And save us hours of
|
||||||
|
| painful debugging.
|
||||||
|
*/
|
||||||
error_reporting(E_ALL);
|
error_reporting(E_ALL);
|
||||||
ini_set('display_errors', '1');
|
ini_set('display_errors', '1');
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| DEBUG BACKTRACES
|
| DEBUG BACKTRACES
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
*/
|
| If true, this constant will tell the error screens to display debug
|
||||||
|
| backtraces along with the other error information. If you would
|
||||||
|
| prefer to not see this, set this value to false.
|
||||||
|
*/
|
||||||
defined('SHOW_DEBUG_BACKTRACE') || define('SHOW_DEBUG_BACKTRACE', true);
|
defined('SHOW_DEBUG_BACKTRACE') || define('SHOW_DEBUG_BACKTRACE', true);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
| DEBUG MODE
|
| DEBUG MODE
|
||||||
|--------------------------------------------------------------------------
|
|--------------------------------------------------------------------------
|
||||||
*/
|
| Debug mode is an experimental flag that can allow changes throughout
|
||||||
defined('CI_DEBUG') || define('CI_DEBUG', true);
|
| the system. It's not widely used currently, and may not survive
|
||||||
|
| release of the framework.
|
||||||
|
*/
|
||||||
|
defined('CI_DEBUG') || define('CI_DEBUG', true);
|
||||||
|
|||||||
@@ -6,6 +6,22 @@ use CodeIgniter\Config\BaseConfig;
|
|||||||
|
|
||||||
class CURLRequest extends BaseConfig
|
class CURLRequest extends BaseConfig
|
||||||
{
|
{
|
||||||
|
/**
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
* CURLRequest Share Connection Options
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* Share connection options between requests.
|
||||||
|
*
|
||||||
|
* @var list<int>
|
||||||
|
*
|
||||||
|
* @see https://www.php.net/manual/en/curl.constants.php#constant.curl-lock-data-connect
|
||||||
|
*/
|
||||||
|
public array $shareConnectionOptions = [
|
||||||
|
CURL_LOCK_DATA_CONNECT,
|
||||||
|
CURL_LOCK_DATA_DNS,
|
||||||
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* --------------------------------------------------------------------------
|
* --------------------------------------------------------------------------
|
||||||
* CURLRequest Share Options
|
* CURLRequest Share Options
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
namespace Config;
|
namespace Config;
|
||||||
|
|
||||||
use CodeIgniter\Cache\CacheInterface;
|
use CodeIgniter\Cache\CacheInterface;
|
||||||
|
use CodeIgniter\Cache\Handlers\ApcuHandler;
|
||||||
use CodeIgniter\Cache\Handlers\DummyHandler;
|
use CodeIgniter\Cache\Handlers\DummyHandler;
|
||||||
use CodeIgniter\Cache\Handlers\FileHandler;
|
use CodeIgniter\Cache\Handlers\FileHandler;
|
||||||
use CodeIgniter\Cache\Handlers\MemcachedHandler;
|
use CodeIgniter\Cache\Handlers\MemcachedHandler;
|
||||||
@@ -78,7 +79,7 @@ class Cache extends BaseConfig
|
|||||||
* Your file storage preferences can be specified below, if you are using
|
* Your file storage preferences can be specified below, if you are using
|
||||||
* the File driver.
|
* the File driver.
|
||||||
*
|
*
|
||||||
* @var array<string, int|string|null>
|
* @var array{storePath?: string, mode?: int}
|
||||||
*/
|
*/
|
||||||
public array $file = [
|
public array $file = [
|
||||||
'storePath' => WRITEPATH . 'cache/',
|
'storePath' => WRITEPATH . 'cache/',
|
||||||
@@ -95,7 +96,7 @@ class Cache extends BaseConfig
|
|||||||
*
|
*
|
||||||
* @see https://codeigniter.com/user_guide/libraries/caching.html#memcached
|
* @see https://codeigniter.com/user_guide/libraries/caching.html#memcached
|
||||||
*
|
*
|
||||||
* @var array<string, bool|int|string>
|
* @var array{host?: string, port?: int, weight?: int, raw?: bool}
|
||||||
*/
|
*/
|
||||||
public array $memcached = [
|
public array $memcached = [
|
||||||
'host' => '127.0.0.1',
|
'host' => '127.0.0.1',
|
||||||
@@ -108,17 +109,28 @@ class Cache extends BaseConfig
|
|||||||
* -------------------------------------------------------------------------
|
* -------------------------------------------------------------------------
|
||||||
* Redis settings
|
* Redis settings
|
||||||
* -------------------------------------------------------------------------
|
* -------------------------------------------------------------------------
|
||||||
|
*
|
||||||
* Your Redis server can be specified below, if you are using
|
* Your Redis server can be specified below, if you are using
|
||||||
* the Redis or Predis drivers.
|
* the Redis or Predis drivers.
|
||||||
*
|
*
|
||||||
* @var array<string, int|string|null>
|
* @var array{
|
||||||
|
* host?: string,
|
||||||
|
* password?: string|null,
|
||||||
|
* port?: int,
|
||||||
|
* timeout?: int,
|
||||||
|
* async?: bool,
|
||||||
|
* persistent?: bool,
|
||||||
|
* database?: int
|
||||||
|
* }
|
||||||
*/
|
*/
|
||||||
public array $redis = [
|
public array $redis = [
|
||||||
'host' => '127.0.0.1',
|
'host' => '127.0.0.1',
|
||||||
'password' => null,
|
'password' => null,
|
||||||
'port' => 6379,
|
'port' => 6379,
|
||||||
'timeout' => 0,
|
'timeout' => 0,
|
||||||
'database' => 0,
|
'async' => false, // specific to Predis and ignored by the native Redis extension
|
||||||
|
'persistent' => false,
|
||||||
|
'database' => 0,
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -132,6 +144,7 @@ class Cache extends BaseConfig
|
|||||||
* @var array<string, class-string<CacheInterface>>
|
* @var array<string, class-string<CacheInterface>>
|
||||||
*/
|
*/
|
||||||
public array $validHandlers = [
|
public array $validHandlers = [
|
||||||
|
'apcu' => ApcuHandler::class,
|
||||||
'dummy' => DummyHandler::class,
|
'dummy' => DummyHandler::class,
|
||||||
'file' => FileHandler::class,
|
'file' => FileHandler::class,
|
||||||
'memcached' => MemcachedHandler::class,
|
'memcached' => MemcachedHandler::class,
|
||||||
@@ -158,4 +171,28 @@ class Cache extends BaseConfig
|
|||||||
* @var bool|list<string>
|
* @var bool|list<string>
|
||||||
*/
|
*/
|
||||||
public $cacheQueryString = false;
|
public $cacheQueryString = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
* Web Page Caching: Cache Status Codes
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* HTTP status codes that are allowed to be cached. Only responses with
|
||||||
|
* these status codes will be cached by the PageCache filter.
|
||||||
|
*
|
||||||
|
* Default: [] - Cache all status codes (backward compatible)
|
||||||
|
*
|
||||||
|
* Recommended: [200] - Only cache successful responses
|
||||||
|
*
|
||||||
|
* You can also use status codes like:
|
||||||
|
* [200, 404, 410] - Cache successful responses and specific error codes
|
||||||
|
* [200, 201, 202, 203, 204] - All 2xx successful responses
|
||||||
|
*
|
||||||
|
* WARNING: Using [] may cache temporary error pages (404, 500, etc).
|
||||||
|
* Consider restricting to [200] for production applications to avoid
|
||||||
|
* caching errors that should be temporary.
|
||||||
|
*
|
||||||
|
* @var list<int>
|
||||||
|
*/
|
||||||
|
public array $cacheStatusCodes = [];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,6 +30,11 @@ class ContentSecurityPolicy extends BaseConfig
|
|||||||
*/
|
*/
|
||||||
public ?string $reportURI = null;
|
public ?string $reportURI = null;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies a reporting endpoint to which violation reports ought to be sent.
|
||||||
|
*/
|
||||||
|
public ?string $reportTo = null;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Instructs user agents to rewrite URL schemes, changing
|
* Instructs user agents to rewrite URL schemes, changing
|
||||||
* HTTP to HTTPS. This directive is for websites with
|
* HTTP to HTTPS. This directive is for websites with
|
||||||
@@ -38,12 +43,12 @@ class ContentSecurityPolicy extends BaseConfig
|
|||||||
public bool $upgradeInsecureRequests = false;
|
public bool $upgradeInsecureRequests = false;
|
||||||
|
|
||||||
// -------------------------------------------------------------------------
|
// -------------------------------------------------------------------------
|
||||||
// Sources allowed
|
// CSP DIRECTIVES SETTINGS
|
||||||
// NOTE: once you set a policy to 'none', it cannot be further restricted
|
// NOTE: once you set a policy to 'none', it cannot be further restricted
|
||||||
// -------------------------------------------------------------------------
|
// -------------------------------------------------------------------------
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Will default to self if not overridden
|
* Will default to `'self'` if not overridden
|
||||||
*
|
*
|
||||||
* @var list<string>|string|null
|
* @var list<string>|string|null
|
||||||
*/
|
*/
|
||||||
@@ -64,6 +69,21 @@ class ContentSecurityPolicy extends BaseConfig
|
|||||||
'www.google.com www.gstatic.com'
|
'www.google.com www.gstatic.com'
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies valid sources for JavaScript <script> elements.
|
||||||
|
*
|
||||||
|
* @var list<string>|string
|
||||||
|
*/
|
||||||
|
public array|string $scriptSrcElem = 'self';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies valid sources for JavaScript inline event
|
||||||
|
* handlers and JavaScript URLs.
|
||||||
|
*
|
||||||
|
* @var list<string>|string
|
||||||
|
*/
|
||||||
|
public array|string $scriptSrcAttr = 'self';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lists allowed stylesheets' URLs.
|
* Lists allowed stylesheets' URLs.
|
||||||
*
|
*
|
||||||
@@ -76,6 +96,21 @@ class ContentSecurityPolicy extends BaseConfig
|
|||||||
'https://fonts.googleapis.com',
|
'https://fonts.googleapis.com',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies valid sources for stylesheets <link> elements.
|
||||||
|
*
|
||||||
|
* @var list<string>|string
|
||||||
|
*/
|
||||||
|
public array|string $styleSrcElem = 'self';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Specifies valid sources for stylesheets inline
|
||||||
|
* style attributes and `<style>` elements.
|
||||||
|
*
|
||||||
|
* @var list<string>|string
|
||||||
|
*/
|
||||||
|
public array|string $styleSrcAttr = 'self';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Defines the origins from which images can be loaded.
|
* Defines the origins from which images can be loaded.
|
||||||
*
|
*
|
||||||
@@ -169,6 +204,11 @@ class ContentSecurityPolicy extends BaseConfig
|
|||||||
*/
|
*/
|
||||||
public $manifestSrc;
|
public $manifestSrc;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* @var list<string>|string
|
||||||
|
*/
|
||||||
|
public array|string $workerSrc = [];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Limits the kinds of plugins a page may invoke.
|
* Limits the kinds of plugins a page may invoke.
|
||||||
*
|
*
|
||||||
@@ -184,17 +224,17 @@ class ContentSecurityPolicy extends BaseConfig
|
|||||||
public $sandbox;
|
public $sandbox;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Nonce tag for style
|
* Nonce placeholder for style tags.
|
||||||
*/
|
*/
|
||||||
public string $styleNonceTag = '{csp-style-nonce}';
|
public string $styleNonceTag = '{csp-style-nonce}';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Nonce tag for script
|
* Nonce placeholder for script tags.
|
||||||
*/
|
*/
|
||||||
public string $scriptNonceTag = '{csp-script-nonce}';
|
public string $scriptNonceTag = '{csp-script-nonce}';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Replace nonce tag automatically
|
* Replace nonce tag automatically?
|
||||||
*/
|
*/
|
||||||
public bool $autoNonce = true;
|
public bool $autoNonce = true;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -85,7 +85,7 @@ class Cookie extends BaseConfig
|
|||||||
* (empty string) means default SameSite attribute set by browsers (`Lax`)
|
* (empty string) means default SameSite attribute set by browsers (`Lax`)
|
||||||
* will be set on cookies. If set to `None`, `$secure` must also be set.
|
* will be set on cookies. If set to `None`, `$secure` must also be set.
|
||||||
*
|
*
|
||||||
* @phpstan-var 'None'|'Lax'|'Strict'|''
|
* @var ''|'Lax'|'None'|'Strict'
|
||||||
*/
|
*/
|
||||||
public string $samesite = 'Lax';
|
public string $samesite = 'Lax';
|
||||||
|
|
||||||
|
|||||||
@@ -42,6 +42,8 @@ class Database extends Config
|
|||||||
'strictOn' => false,
|
'strictOn' => false,
|
||||||
'failover' => [],
|
'failover' => [],
|
||||||
'port' => 3306,
|
'port' => 3306,
|
||||||
|
'numberNative' => false,
|
||||||
|
'foundRows' => false,
|
||||||
'dateFormat' => [
|
'dateFormat' => [
|
||||||
'date' => 'Y-m-d',
|
'date' => 'Y-m-d',
|
||||||
'datetime' => 'Y-m-d H:i:s',
|
'datetime' => 'Y-m-d H:i:s',
|
||||||
@@ -55,26 +57,27 @@ class Database extends Config
|
|||||||
* @var array<string, mixed>
|
* @var array<string, mixed>
|
||||||
*/
|
*/
|
||||||
public array $tests = [
|
public array $tests = [
|
||||||
'DSN' => '',
|
'DSN' => '',
|
||||||
'hostname' => 'localhost',
|
'hostname' => 'localhost',
|
||||||
'username' => 'admin',
|
'username' => 'admin',
|
||||||
'password' => 'pointofsale',
|
'password' => 'pointofsale',
|
||||||
'database' => 'ospos',
|
'database' => 'ospos',
|
||||||
'DBDriver' => 'MySQLi',
|
'DBDriver' => 'MySQLi',
|
||||||
'DBPrefix' => 'ospos_',
|
'DBPrefix' => 'ospos_',
|
||||||
'pConnect' => false,
|
'pConnect' => false,
|
||||||
'DBDebug' => (ENVIRONMENT !== 'production'),
|
'DBDebug' => (ENVIRONMENT !== 'production'),
|
||||||
'charset' => 'utf8mb4',
|
'charset' => 'utf8mb4',
|
||||||
'DBCollat' => 'utf8mb4_general_ci',
|
'DBCollat' => 'utf8mb4_general_ci',
|
||||||
'swapPre' => '',
|
'swapPre' => '',
|
||||||
'encrypt' => false,
|
'encrypt' => false,
|
||||||
'compress' => false,
|
'compress' => false,
|
||||||
'strictOn' => false,
|
'strictOn' => false,
|
||||||
'failover' => [],
|
'failover' => [],
|
||||||
'port' => 3306,
|
'port' => 3306,
|
||||||
'foreignKeys' => true,
|
'foreignKeys' => true,
|
||||||
'busyTimeout' => 1000,
|
'busyTimeout' => 1000,
|
||||||
'dateFormat' => [
|
'synchronous' => null,
|
||||||
|
'dateFormat' => [
|
||||||
'date' => 'Y-m-d',
|
'date' => 'Y-m-d',
|
||||||
'datetime' => 'Y-m-d H:i:s',
|
'datetime' => 'Y-m-d H:i:s',
|
||||||
'time' => 'H:i:s',
|
'time' => 'H:i:s',
|
||||||
|
|||||||
@@ -2,9 +2,6 @@
|
|||||||
|
|
||||||
namespace Config;
|
namespace Config;
|
||||||
|
|
||||||
/**
|
|
||||||
* @immutable
|
|
||||||
*/
|
|
||||||
class DocTypes
|
class DocTypes
|
||||||
{
|
{
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -30,6 +30,11 @@ class Email extends BaseConfig
|
|||||||
*/
|
*/
|
||||||
public string $SMTPHost = 'mail.mxserver.com';
|
public string $SMTPHost = 'mail.mxserver.com';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Which SMTP authentication method to use: login, plain
|
||||||
|
*/
|
||||||
|
public string $SMTPAuthMethod = 'login';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* SMTP Username
|
* SMTP Username
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -23,6 +23,23 @@ class Encryption extends BaseConfig
|
|||||||
*/
|
*/
|
||||||
public string $key = '';
|
public string $key = '';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
* Previous Encryption Keys
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* When rotating encryption keys, add old keys here to maintain ability
|
||||||
|
* to decrypt data encrypted with previous keys. Encryption always uses
|
||||||
|
* the current $key. Decryption tries current key first, then falls back
|
||||||
|
* to previous keys if decryption fails.
|
||||||
|
*
|
||||||
|
* In .env file, use comma-separated string:
|
||||||
|
* encryption.previousKeys = hex2bin:9be8c64fcea509867...,hex2bin:3f5a1d8e9c2b7a4f6...
|
||||||
|
*
|
||||||
|
* @var list<string>|string
|
||||||
|
*/
|
||||||
|
public array|string $previousKeys = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* --------------------------------------------------------------------------
|
* --------------------------------------------------------------------------
|
||||||
* Encryption Driver to Use
|
* Encryption Driver to Use
|
||||||
|
|||||||
@@ -65,12 +65,15 @@ class Filters extends BaseFilters
|
|||||||
* List of filter aliases that are always
|
* List of filter aliases that are always
|
||||||
* applied before and after every request.
|
* applied before and after every request.
|
||||||
*
|
*
|
||||||
* @var array<string, array<string, array<string, string>>>|array<string, list<string>>
|
* @var array{
|
||||||
|
* before: array<string, array{except: list<string>|string}>|list<string>,
|
||||||
|
* after: array<string, array{except: list<string>|string}>|list<string>
|
||||||
|
* }
|
||||||
*/
|
*/
|
||||||
public array $globals = [
|
public array $globals = [
|
||||||
'before' => [
|
'before' => [
|
||||||
'honeypot',
|
'honeypot',
|
||||||
'csrf' => ['except' => 'login'],
|
'csrf' => ['except' => 'login|migrate'],
|
||||||
'invalidchars',
|
'invalidchars',
|
||||||
],
|
],
|
||||||
'after' => [
|
'after' => [
|
||||||
@@ -100,7 +103,7 @@ class Filters extends BaseFilters
|
|||||||
* before or after URI patterns.
|
* before or after URI patterns.
|
||||||
*
|
*
|
||||||
* Example:
|
* Example:
|
||||||
* isLoggedIn' => ['before' => ['account/*', 'profiles/*']]
|
* 'isLoggedIn' => ['before' => ['account/*', 'profiles/*']]
|
||||||
*
|
*
|
||||||
* @var array<string, array<string, list<string>>>
|
* @var array<string, array<string, list<string>>>
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -61,4 +61,13 @@ class Format extends BaseConfig
|
|||||||
'application/xml' => 0,
|
'application/xml' => 0,
|
||||||
'text/xml' => 0,
|
'text/xml' => 0,
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
* Maximum depth for JSON encoding.
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* This value determines how deep the JSON encoder will traverse nested structures.
|
||||||
|
*/
|
||||||
|
public int $jsonEncodeDepth = 512;
|
||||||
}
|
}
|
||||||
|
|||||||
40
app/Config/Hostnames.php
Normal file
40
app/Config/Hostnames.php
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Config;
|
||||||
|
|
||||||
|
class Hostnames
|
||||||
|
{
|
||||||
|
// List of known two-part TLDs for subdomain extraction
|
||||||
|
public const TWO_PART_TLDS = [
|
||||||
|
'co.uk', 'org.uk', 'gov.uk', 'ac.uk', 'sch.uk', 'ltd.uk', 'plc.uk',
|
||||||
|
'com.au', 'net.au', 'org.au', 'edu.au', 'gov.au', 'asn.au', 'id.au',
|
||||||
|
'co.jp', 'ac.jp', 'go.jp', 'or.jp', 'ne.jp', 'gr.jp',
|
||||||
|
'co.nz', 'org.nz', 'govt.nz', 'ac.nz', 'net.nz', 'geek.nz', 'maori.nz', 'school.nz',
|
||||||
|
'co.in', 'net.in', 'org.in', 'ind.in', 'ac.in', 'gov.in', 'res.in',
|
||||||
|
'com.cn', 'net.cn', 'org.cn', 'gov.cn', 'edu.cn',
|
||||||
|
'com.sg', 'net.sg', 'org.sg', 'gov.sg', 'edu.sg', 'per.sg',
|
||||||
|
'co.za', 'org.za', 'gov.za', 'ac.za', 'net.za',
|
||||||
|
'co.kr', 'or.kr', 'go.kr', 'ac.kr', 'ne.kr', 'pe.kr',
|
||||||
|
'co.th', 'or.th', 'go.th', 'ac.th', 'net.th', 'in.th',
|
||||||
|
'com.my', 'net.my', 'org.my', 'edu.my', 'gov.my', 'mil.my', 'name.my',
|
||||||
|
'com.mx', 'org.mx', 'net.mx', 'edu.mx', 'gob.mx',
|
||||||
|
'com.br', 'net.br', 'org.br', 'gov.br', 'edu.br', 'art.br', 'eng.br',
|
||||||
|
'co.il', 'org.il', 'ac.il', 'gov.il', 'net.il', 'muni.il',
|
||||||
|
'co.id', 'or.id', 'ac.id', 'go.id', 'net.id', 'web.id', 'my.id',
|
||||||
|
'com.hk', 'edu.hk', 'gov.hk', 'idv.hk', 'net.hk', 'org.hk',
|
||||||
|
'com.tw', 'net.tw', 'org.tw', 'edu.tw', 'gov.tw', 'idv.tw',
|
||||||
|
'com.sa', 'net.sa', 'org.sa', 'gov.sa', 'edu.sa', 'sch.sa', 'med.sa',
|
||||||
|
'co.ae', 'net.ae', 'org.ae', 'gov.ae', 'ac.ae', 'sch.ae',
|
||||||
|
'com.tr', 'net.tr', 'org.tr', 'gov.tr', 'edu.tr', 'av.tr', 'gen.tr',
|
||||||
|
'co.ke', 'or.ke', 'go.ke', 'ac.ke', 'sc.ke', 'me.ke', 'mobi.ke', 'info.ke',
|
||||||
|
'com.ng', 'org.ng', 'gov.ng', 'edu.ng', 'net.ng', 'sch.ng', 'name.ng',
|
||||||
|
'com.pk', 'net.pk', 'org.pk', 'gov.pk', 'edu.pk', 'fam.pk',
|
||||||
|
'com.eg', 'edu.eg', 'gov.eg', 'org.eg', 'net.eg',
|
||||||
|
'com.cy', 'net.cy', 'org.cy', 'gov.cy', 'ac.cy',
|
||||||
|
'com.lk', 'org.lk', 'edu.lk', 'gov.lk', 'net.lk', 'int.lk',
|
||||||
|
'com.bd', 'net.bd', 'org.bd', 'ac.bd', 'gov.bd', 'mil.bd',
|
||||||
|
'com.ar', 'net.ar', 'org.ar', 'gov.ar', 'edu.ar', 'mil.ar',
|
||||||
|
'gob.cl', 'com.pl', 'net.pl', 'org.pl', 'gov.pl', 'edu.pl',
|
||||||
|
'co.ir', 'ac.ir', 'org.ir', 'id.ir', 'gov.ir', 'sch.ir', 'net.ir',
|
||||||
|
];
|
||||||
|
}
|
||||||
@@ -16,6 +16,8 @@ class Images extends BaseConfig
|
|||||||
/**
|
/**
|
||||||
* The path to the image library.
|
* The path to the image library.
|
||||||
* Required for ImageMagick, GraphicsMagick, or NetPBM.
|
* Required for ImageMagick, GraphicsMagick, or NetPBM.
|
||||||
|
*
|
||||||
|
* @deprecated 4.7.0 No longer used.
|
||||||
*/
|
*/
|
||||||
public string $libraryPath = '/usr/local/bin/convert';
|
public string $libraryPath = '/usr/local/bin/convert';
|
||||||
|
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ namespace Config;
|
|||||||
|
|
||||||
use CodeIgniter\Config\BaseConfig;
|
use CodeIgniter\Config\BaseConfig;
|
||||||
use CodeIgniter\Log\Handlers\FileHandler;
|
use CodeIgniter\Log\Handlers\FileHandler;
|
||||||
|
use CodeIgniter\Log\Handlers\HandlerInterface;
|
||||||
|
|
||||||
class Logger extends BaseConfig
|
class Logger extends BaseConfig
|
||||||
{
|
{
|
||||||
@@ -73,7 +74,7 @@ class Logger extends BaseConfig
|
|||||||
* Handlers are executed in the order defined in this array, starting with
|
* Handlers are executed in the order defined in this array, starting with
|
||||||
* the handler on top and continuing down.
|
* the handler on top and continuing down.
|
||||||
*
|
*
|
||||||
* @var array<class-string, array<string, int|list<string>|string>>
|
* @var array<class-string<HandlerInterface>, array<string, int|list<string>|string>>
|
||||||
*/
|
*/
|
||||||
public array $handlers = [
|
public array $handlers = [
|
||||||
/*
|
/*
|
||||||
|
|||||||
@@ -47,4 +47,19 @@ class Migrations extends BaseConfig
|
|||||||
* - Y_m_d_His_
|
* - Y_m_d_His_
|
||||||
*/
|
*/
|
||||||
public string $timestampFormat = 'YmdHis_';
|
public string $timestampFormat = 'YmdHis_';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
* Enable/Disable Migration Lock
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* Locking is disabled by default.
|
||||||
|
*
|
||||||
|
* When enabled, it will prevent multiple migration processes
|
||||||
|
* from running at the same time by using a lock mechanism.
|
||||||
|
*
|
||||||
|
* This is useful in production environments to avoid conflicts
|
||||||
|
* or race conditions during concurrent deployments.
|
||||||
|
*/
|
||||||
|
public bool $lock = false;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,8 +3,6 @@
|
|||||||
namespace Config;
|
namespace Config;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Mimes
|
|
||||||
*
|
|
||||||
* This file contains an array of mime types. It is used by the
|
* This file contains an array of mime types. It is used by the
|
||||||
* Upload class to help identify allowed file types.
|
* Upload class to help identify allowed file types.
|
||||||
*
|
*
|
||||||
@@ -15,8 +13,6 @@ namespace Config;
|
|||||||
*
|
*
|
||||||
* When working with mime types, please make sure you have the ´fileinfo´
|
* When working with mime types, please make sure you have the ´fileinfo´
|
||||||
* extension enabled to reliably detect the media types.
|
* extension enabled to reliably detect the media types.
|
||||||
*
|
|
||||||
* @immutable
|
|
||||||
*/
|
*/
|
||||||
class Mimes
|
class Mimes
|
||||||
{
|
{
|
||||||
@@ -482,6 +478,8 @@ class Mimes
|
|||||||
'application/sla',
|
'application/sla',
|
||||||
'application/vnd.ms-pki.stl',
|
'application/vnd.ms-pki.stl',
|
||||||
'application/x-navistyle',
|
'application/x-navistyle',
|
||||||
|
'model/stl',
|
||||||
|
'application/octet-stream',
|
||||||
],
|
],
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -490,7 +488,7 @@ class Mimes
|
|||||||
*
|
*
|
||||||
* @return string|null The mime type found, or none if unable to determine.
|
* @return string|null The mime type found, or none if unable to determine.
|
||||||
*/
|
*/
|
||||||
public static function guessTypeFromExtension(string $extension): array|string|null
|
public static function guessTypeFromExtension(string $extension)
|
||||||
{
|
{
|
||||||
$extension = trim(strtolower($extension), '. ');
|
$extension = trim(strtolower($extension), '. ');
|
||||||
|
|
||||||
@@ -508,7 +506,7 @@ class Mimes
|
|||||||
*
|
*
|
||||||
* @return string|null The extension determined, or null if unable to match.
|
* @return string|null The extension determined, or null if unable to match.
|
||||||
*/
|
*/
|
||||||
public static function guessExtensionFromType(string $type, ?string $proposedExtension = null): ?string
|
public static function guessExtensionFromType(string $type, ?string $proposedExtension = null)
|
||||||
{
|
{
|
||||||
$type = trim(strtolower($type), '. ');
|
$type = trim(strtolower($type), '. ');
|
||||||
|
|
||||||
|
|||||||
@@ -9,8 +9,6 @@ use CodeIgniter\Modules\Modules as BaseModules;
|
|||||||
*
|
*
|
||||||
* NOTE: This class is required prior to Autoloader instantiation,
|
* NOTE: This class is required prior to Autoloader instantiation,
|
||||||
* and does not extend BaseConfig.
|
* and does not extend BaseConfig.
|
||||||
*
|
|
||||||
* @immutable
|
|
||||||
*/
|
*/
|
||||||
class Modules extends BaseModules
|
class Modules extends BaseModules
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ namespace Config;
|
|||||||
use App\Models\Appconfig;
|
use App\Models\Appconfig;
|
||||||
use CodeIgniter\Cache\CacheInterface;
|
use CodeIgniter\Cache\CacheInterface;
|
||||||
use CodeIgniter\Config\BaseConfig;
|
use CodeIgniter\Config\BaseConfig;
|
||||||
|
use Config\Database;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class holds the configuration options stored from the database so that on launch those settings can be cached
|
* This class holds the configuration options stored from the database so that on launch those settings can be cached
|
||||||
@@ -13,7 +14,7 @@ use CodeIgniter\Config\BaseConfig;
|
|||||||
*/
|
*/
|
||||||
class OSPOS extends BaseConfig
|
class OSPOS extends BaseConfig
|
||||||
{
|
{
|
||||||
public array $settings;
|
public array $settings = [];
|
||||||
public string $commit_sha1 = 'dev'; // TODO: Travis scripts need to be updated to replace this with the commit hash on build
|
public string $commit_sha1 = 'dev'; // TODO: Travis scripts need to be updated to replace this with the commit hash on build
|
||||||
private CacheInterface $cache;
|
private CacheInterface $cache;
|
||||||
|
|
||||||
@@ -33,15 +34,37 @@ class OSPOS extends BaseConfig
|
|||||||
|
|
||||||
if ($cache) {
|
if ($cache) {
|
||||||
$this->settings = decode_array($cache);
|
$this->settings = decode_array($cache);
|
||||||
} else {
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$db = Database::connect();
|
||||||
|
|
||||||
|
if (!$db->tableExists('app_config')) {
|
||||||
|
$this->settings = $this->getDefaultSettings();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$appconfig = model(Appconfig::class);
|
$appconfig = model(Appconfig::class);
|
||||||
foreach ($appconfig->get_all()->getResult() as $app_config) {
|
foreach ($appconfig->get_all()->getResult() as $app_config) {
|
||||||
$this->settings[$app_config->key] = $app_config->value;
|
$this->settings[$app_config->key] = $app_config->value;
|
||||||
}
|
}
|
||||||
$this->cache->save('settings', encode_array($this->settings));
|
$this->cache->save('settings', encode_array($this->settings));
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
$this->settings = $this->getDefaultSettings();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function getDefaultSettings(): array
|
||||||
|
{
|
||||||
|
return [
|
||||||
|
'language' => 'english',
|
||||||
|
'language_code' => 'en',
|
||||||
|
'company' => 'Home',
|
||||||
|
'barcode_type' => 'Code39'
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @return void
|
* @return void
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ namespace Config;
|
|||||||
* NOTE: This class does not extend BaseConfig for performance reasons.
|
* NOTE: This class does not extend BaseConfig for performance reasons.
|
||||||
* So you cannot replace the property values with Environment Variables.
|
* So you cannot replace the property values with Environment Variables.
|
||||||
*
|
*
|
||||||
* @immutable
|
* WARNING: Do not use these options when running the app in the Worker Mode.
|
||||||
*/
|
*/
|
||||||
class Optimize
|
class Optimize
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -15,8 +15,6 @@ namespace Config;
|
|||||||
*
|
*
|
||||||
* NOTE: This class is required prior to Autoloader instantiation,
|
* NOTE: This class is required prior to Autoloader instantiation,
|
||||||
* and does not extend BaseConfig.
|
* and does not extend BaseConfig.
|
||||||
*
|
|
||||||
* @immutable
|
|
||||||
*/
|
*/
|
||||||
class Paths
|
class Paths
|
||||||
{
|
{
|
||||||
@@ -77,4 +75,16 @@ class Paths
|
|||||||
* is used when no value is provided to `Services::renderer()`.
|
* is used when no value is provided to `Services::renderer()`.
|
||||||
*/
|
*/
|
||||||
public string $viewDirectory = __DIR__ . '/../Views';
|
public string $viewDirectory = __DIR__ . '/../Views';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ---------------------------------------------------------------
|
||||||
|
* ENVIRONMENT DIRECTORY NAME
|
||||||
|
* ---------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* This variable must contain the name of the directory where
|
||||||
|
* the .env file is located.
|
||||||
|
* Please consider security implications when changing this
|
||||||
|
* value - the directory should not be publicly accessible.
|
||||||
|
*/
|
||||||
|
public string $envDirectory = __DIR__ . '/../../';
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
use CodeIgniter\Router\RouteCollection;
|
use CodeIgniter\Router\RouteCollection;
|
||||||
|
|
||||||
@@ -10,6 +10,41 @@ $routes->setDefaultController('Login');
|
|||||||
$routes->get('/', 'Login::index');
|
$routes->get('/', 'Login::index');
|
||||||
$routes->get('login', 'Login::index');
|
$routes->get('login', 'Login::index');
|
||||||
$routes->post('login', 'Login::index');
|
$routes->post('login', 'Login::index');
|
||||||
|
$routes->post('migrate', 'Login::migrate');
|
||||||
|
|
||||||
|
$routes->get('sales', 'Sales::getIndex');
|
||||||
|
$routes->get('sales/customerDisplay', 'Sales::getCustomerDisplay');
|
||||||
|
$routes->get('sales/itemSearch', 'Sales::getItemSearch');
|
||||||
|
$routes->post('sales/selectCustomer', 'Sales::postSelectCustomer');
|
||||||
|
$routes->post('sales/changeMode', 'Sales::postChangeMode');
|
||||||
|
$routes->post('sales/setComment', 'Sales::postSetComment');
|
||||||
|
$routes->post('sales/setInvoiceNumber', 'Sales::postSetInvoiceNumber');
|
||||||
|
$routes->post('sales/setPaymentType', 'Sales::postSetPaymentType');
|
||||||
|
$routes->post('sales/setPrintAfterSale', 'Sales::postSetPrintAfterSale');
|
||||||
|
$routes->post('sales/setPriceWorkOrders', 'Sales::postSetPriceWorkOrders');
|
||||||
|
$routes->post('sales/setEmailReceipt', 'Sales::postSetEmailReceipt');
|
||||||
|
$routes->post('sales/addPayment', 'Sales::postAddPayment');
|
||||||
|
$routes->post('sales/add', 'Sales::postAdd');
|
||||||
|
$routes->post('sales/editItem/(:segment)', 'Sales::postEditItem/$1');
|
||||||
|
$routes->post('sales/deleteItem/(:segment)', 'Sales::getDeleteItem/$1');
|
||||||
|
$routes->post('sales/deletePayment/(:segment)', 'Sales::getDeletePayment/$1');
|
||||||
|
$routes->post('sales/removeCustomer', 'Sales::getRemoveCustomer');
|
||||||
|
$routes->post('sales/complete', 'Sales::postComplete');
|
||||||
|
$routes->post('sales/cancel', 'Sales::postCancel');
|
||||||
|
$routes->post('sales/suspend', 'Sales::postSuspend');
|
||||||
|
$routes->post('sales/unsuspend', 'Sales::postUnsuspend');
|
||||||
|
$routes->post('sales/checkInvoiceNumber', 'Sales::postCheckInvoiceNumber');
|
||||||
|
$routes->post('sales/changeItemNumber', 'Sales::postChangeItemNumber');
|
||||||
|
$routes->post('sales/changeItemName', 'Sales::postChangeItemName');
|
||||||
|
$routes->post('sales/changeItemDescription', 'Sales::postChangeItemDescription');
|
||||||
|
$routes->get('sales/suspended', 'Sales::getSuspended');
|
||||||
|
$routes->get('sales/discardSuspendedSale', 'Sales::getDiscardSuspendedSale');
|
||||||
|
$routes->get('sales/sales_keyboard_help', 'Sales::getSalesKeyboardHelp');
|
||||||
|
$routes->get('sales/receipt/(:num)', 'Sales::getReceipt/$1');
|
||||||
|
$routes->get('sales/invoice/(:num)', 'Sales::getInvoice/$1');
|
||||||
|
$routes->get('sales/edit/(:num)', 'Sales::getEdit/$1');
|
||||||
|
$routes->post('sales/delete/(:num)', 'Sales::postDelete/$1');
|
||||||
|
$routes->post('sales/save/(:num)', 'Sales::postSave/$1');
|
||||||
|
|
||||||
$routes->add('no_access/index/(:segment)', 'No_access::index/$1');
|
$routes->add('no_access/index/(:segment)', 'No_access::index/$1');
|
||||||
$routes->add('no_access/index/(:segment)/(:segment)', 'No_access::index/$1/$2');
|
$routes->add('no_access/index/(:segment)/(:segment)', 'No_access::index/$1/$2');
|
||||||
|
|||||||
@@ -96,6 +96,15 @@ class Routing extends BaseRouting
|
|||||||
*/
|
*/
|
||||||
public bool $autoRoute = true;
|
public bool $autoRoute = true;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* If TRUE, the system will look for attributes on controller
|
||||||
|
* class and methods that can run before and after the
|
||||||
|
* controller/method.
|
||||||
|
*
|
||||||
|
* If FALSE, will ignore any attributes.
|
||||||
|
*/
|
||||||
|
public bool $useControllerAttributes = true;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* For Defined Routes.
|
* For Defined Routes.
|
||||||
* If TRUE, will enable the use of the 'prioritize' option
|
* If TRUE, will enable the use of the 'prioritize' option
|
||||||
|
|||||||
@@ -13,9 +13,9 @@ class Security extends BaseConfig
|
|||||||
*
|
*
|
||||||
* Protection Method for Cross Site Request Forgery protection.
|
* Protection Method for Cross Site Request Forgery protection.
|
||||||
*
|
*
|
||||||
* @var string|false 'cookie', 'session', or false
|
* @var string 'cookie' or 'session'
|
||||||
*/
|
*/
|
||||||
public string|false $csrfProtection = 'session';
|
public string $csrfProtection = 'session';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* --------------------------------------------------------------------------
|
* --------------------------------------------------------------------------
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace Config;
|
namespace Config;
|
||||||
|
|
||||||
|
use App\Libraries\MY_Language;
|
||||||
use Locale;
|
use Locale;
|
||||||
use HTMLPurifier;
|
use HTMLPurifier;
|
||||||
use HTMLPurifier_Config;
|
use HTMLPurifier_Config;
|
||||||
@@ -38,9 +39,11 @@ class Services extends BaseService
|
|||||||
/**
|
/**
|
||||||
* Responsible for loading the language string translations.
|
* Responsible for loading the language string translations.
|
||||||
*
|
*
|
||||||
|
* @param string|null $locale
|
||||||
|
* @param bool $getShared
|
||||||
* @return MY_Language
|
* @return MY_Language
|
||||||
*/
|
*/
|
||||||
public static function language(?string $locale = null, bool $getShared = true)
|
public static function language(?string $locale = null, bool $getShared = true): MY_Language
|
||||||
{
|
{
|
||||||
if ($getShared) {
|
if ($getShared) {
|
||||||
return static::getSharedInstance('language', $locale)->setLocale($locale);
|
return static::getSharedInstance('language', $locale)->setLocale($locale);
|
||||||
@@ -55,12 +58,12 @@ class Services extends BaseService
|
|||||||
// Use '?:' for empty string check
|
// Use '?:' for empty string check
|
||||||
$locale = $locale ?: $requestLocale;
|
$locale = $locale ?: $requestLocale;
|
||||||
|
|
||||||
return new \App\Libraries\MY_Language($locale);
|
return new MY_Language($locale);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static $htmlPurifier;
|
private static HTMLPurifier $htmlPurifier;
|
||||||
|
|
||||||
public static function htmlPurifier($getShared = true)
|
public static function htmlPurifier($getShared = true): object
|
||||||
{
|
{
|
||||||
if ($getShared) {
|
if ($getShared) {
|
||||||
return static::getSharedInstance('htmlPurifier');
|
return static::getSharedInstance('htmlPurifier');
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ namespace Config;
|
|||||||
use CodeIgniter\Config\BaseConfig;
|
use CodeIgniter\Config\BaseConfig;
|
||||||
use CodeIgniter\Session\Handlers\BaseHandler;
|
use CodeIgniter\Session\Handlers\BaseHandler;
|
||||||
use CodeIgniter\Session\Handlers\DatabaseHandler;
|
use CodeIgniter\Session\Handlers\DatabaseHandler;
|
||||||
|
use CodeIgniter\Session\Handlers\FileHandler;
|
||||||
|
|
||||||
class Session extends BaseConfig
|
class Session extends BaseConfig
|
||||||
{
|
{
|
||||||
@@ -124,4 +125,27 @@ class Session extends BaseConfig
|
|||||||
* seconds.
|
* seconds.
|
||||||
*/
|
*/
|
||||||
public int $lockMaxRetries = 300;
|
public int $lockMaxRetries = 300;
|
||||||
|
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
|
||||||
|
if ($this->driver === DatabaseHandler::class) {
|
||||||
|
try {
|
||||||
|
$db = Database::connect();
|
||||||
|
|
||||||
|
if (!$db->tableExists($this->savePath)) {
|
||||||
|
$this->driver = FileHandler::class;
|
||||||
|
$this->savePath = WRITEPATH . 'session';
|
||||||
|
}
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
// Database not available yet (e.g. fresh install before migrations).
|
||||||
|
// Fall back to file-based sessions so the login/migration page
|
||||||
|
// can still be served. Catches mysqli_sql_exception which is
|
||||||
|
// not a subclass of DatabaseException but is a RuntimeException.
|
||||||
|
$this->driver = FileHandler::class;
|
||||||
|
$this->savePath = WRITEPATH . 'session';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -119,4 +119,29 @@ class Toolbar extends BaseConfig
|
|||||||
public array $watchedExtensions = [
|
public array $watchedExtensions = [
|
||||||
'php', 'css', 'js', 'html', 'svg', 'json', 'env',
|
'php', 'css', 'js', 'html', 'svg', 'json', 'env',
|
||||||
];
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
* Ignored HTTP Headers
|
||||||
|
* --------------------------------------------------------------------------
|
||||||
|
*
|
||||||
|
* CodeIgniter Debug Toolbar normally injects HTML and JavaScript into every
|
||||||
|
* HTML response. This is correct for full page loads, but it breaks requests
|
||||||
|
* that expect only a clean HTML fragment.
|
||||||
|
*
|
||||||
|
* Libraries like HTMX, Unpoly, and Hotwire (Turbo) update parts of the page or
|
||||||
|
* manage navigation on the client side. Injecting the Debug Toolbar into their
|
||||||
|
* responses can cause invalid HTML, duplicated scripts, or JavaScript errors
|
||||||
|
* (such as infinite loops or "Maximum call stack size exceeded").
|
||||||
|
*
|
||||||
|
* Any request containing one of the following headers is treated as a
|
||||||
|
* client-managed or partial request, and the Debug Toolbar injection is skipped.
|
||||||
|
*
|
||||||
|
* @var array<string, string|null>
|
||||||
|
*/
|
||||||
|
public array $disableOnHeaders = [
|
||||||
|
'X-Requested-With' => 'xmlhttprequest', // AJAX requests
|
||||||
|
'HX-Request' => 'true', // HTMX requests
|
||||||
|
'X-Up-Version' => null, // Unpoly partial requests
|
||||||
|
];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -230,9 +230,13 @@ class UserAgents extends BaseConfig
|
|||||||
*/
|
*/
|
||||||
public array $robots = [
|
public array $robots = [
|
||||||
'googlebot' => 'Googlebot',
|
'googlebot' => 'Googlebot',
|
||||||
|
'google-pagerenderer' => 'Google Page Renderer',
|
||||||
|
'google-read-aloud' => 'Google Read Aloud',
|
||||||
|
'google-safety' => 'Google Safety Bot',
|
||||||
'msnbot' => 'MSNBot',
|
'msnbot' => 'MSNBot',
|
||||||
'baiduspider' => 'Baiduspider',
|
'baiduspider' => 'Baiduspider',
|
||||||
'bingbot' => 'Bing',
|
'bingbot' => 'Bing',
|
||||||
|
'bingpreview' => 'BingPreview',
|
||||||
'slurp' => 'Inktomi Slurp',
|
'slurp' => 'Inktomi Slurp',
|
||||||
'yahoo' => 'Yahoo',
|
'yahoo' => 'Yahoo',
|
||||||
'ask jeeves' => 'Ask Jeeves',
|
'ask jeeves' => 'Ask Jeeves',
|
||||||
@@ -248,5 +252,11 @@ class UserAgents extends BaseConfig
|
|||||||
'ia_archiver' => 'Alexa Crawler',
|
'ia_archiver' => 'Alexa Crawler',
|
||||||
'MJ12bot' => 'Majestic-12',
|
'MJ12bot' => 'Majestic-12',
|
||||||
'Uptimebot' => 'Uptimebot',
|
'Uptimebot' => 'Uptimebot',
|
||||||
|
'duckduckbot' => 'DuckDuckBot',
|
||||||
|
'sogou' => 'Sogou Spider',
|
||||||
|
'exabot' => 'Exabot',
|
||||||
|
'bot' => 'Generic Bot',
|
||||||
|
'crawler' => 'Generic Crawler',
|
||||||
|
'spider' => 'Generic Spider',
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -135,4 +135,19 @@ class OSPOSRules
|
|||||||
{
|
{
|
||||||
return parse_decimals($candidate) !== false;
|
return parse_decimals($candidate) !== false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates that a locale-aware decimal value is non-negative (>= 0).
|
||||||
|
*
|
||||||
|
* @param string $candidate
|
||||||
|
* @param string|null $error
|
||||||
|
* @return bool
|
||||||
|
* @noinspection PhpUnused
|
||||||
|
*/
|
||||||
|
public function nonNegativeDecimal(string $candidate, ?string &$error = null): bool
|
||||||
|
{
|
||||||
|
$value = parse_decimals($candidate);
|
||||||
|
|
||||||
|
return $value !== false && $value >= 0;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -59,4 +59,21 @@ class View extends BaseView
|
|||||||
* @var list<class-string<ViewDecoratorInterface>>
|
* @var list<class-string<ViewDecoratorInterface>>
|
||||||
*/
|
*/
|
||||||
public array $decorators = [];
|
public array $decorators = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Subdirectory within app/Views for namespaced view overrides.
|
||||||
|
*
|
||||||
|
* Namespaced views will be searched in:
|
||||||
|
*
|
||||||
|
* app/Views/{$appOverridesFolder}/{Namespace}/{view_path}.{php|html...}
|
||||||
|
*
|
||||||
|
* This allows application-level overrides for package or module views
|
||||||
|
* without modifying vendor source files.
|
||||||
|
*
|
||||||
|
* Examples:
|
||||||
|
* 'overrides' -> app/Views/overrides/Example/Blog/post/card.php
|
||||||
|
* 'vendor' -> app/Views/vendor/Example/Blog/post/card.php
|
||||||
|
* '' -> app/Views/Example/Blog/post/card.php (direct mapping)
|
||||||
|
*/
|
||||||
|
public string $appOverridesFolder = 'overrides';
|
||||||
}
|
}
|
||||||
|
|||||||
62
app/Config/WorkerMode.php
Normal file
62
app/Config/WorkerMode.php
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace Config;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This configuration controls how CodeIgniter behaves when running
|
||||||
|
* in worker mode (with FrankenPHP).
|
||||||
|
*/
|
||||||
|
class WorkerMode
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Persistent Services
|
||||||
|
*
|
||||||
|
* List of service names that should persist across requests.
|
||||||
|
* These services will NOT be reset between requests.
|
||||||
|
*
|
||||||
|
* Services not in this list will be reset for each request to prevent
|
||||||
|
* state leakage.
|
||||||
|
*
|
||||||
|
* Recommended persistent services:
|
||||||
|
* - `autoloader`: PSR-4 autoloading configuration
|
||||||
|
* - `locator`: File locator
|
||||||
|
* - `exceptions`: Exception handler
|
||||||
|
* - `commands`: CLI commands registry
|
||||||
|
* - `codeigniter`: Main application instance
|
||||||
|
* - `superglobals`: Superglobals wrapper
|
||||||
|
* - `routes`: Router configuration
|
||||||
|
* - `cache`: Cache instance
|
||||||
|
*
|
||||||
|
* @var list<string>
|
||||||
|
*/
|
||||||
|
public array $persistentServices = [
|
||||||
|
'autoloader',
|
||||||
|
'locator',
|
||||||
|
'exceptions',
|
||||||
|
'commands',
|
||||||
|
'codeigniter',
|
||||||
|
'superglobals',
|
||||||
|
'routes',
|
||||||
|
'cache',
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reset Event Listeners
|
||||||
|
*
|
||||||
|
* List of event names whose listeners should be removed between requests.
|
||||||
|
* Use this if you register event listeners inside other event callbacks
|
||||||
|
* (rather than at the top level of Config/Events.php), which would cause
|
||||||
|
* them to accumulate across requests in worker mode.
|
||||||
|
*
|
||||||
|
* @var list<string>
|
||||||
|
*/
|
||||||
|
public array $resetEventListeners = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Force Garbage Collection
|
||||||
|
*
|
||||||
|
* Whether to force garbage collection after each request.
|
||||||
|
* Helps prevent memory leaks at a small performance cost.
|
||||||
|
*/
|
||||||
|
public bool $forceGarbageCollection = true;
|
||||||
|
}
|
||||||
@@ -132,7 +132,7 @@ class Attributes extends Secure_Controller
|
|||||||
|
|
||||||
$definition_name = $definition_data['definition_name'];
|
$definition_name = $definition_data['definition_name'];
|
||||||
|
|
||||||
if ($this->attribute->save_definition($definition_data, $definition_id)) {
|
if ($this->attribute->saveDefinition($definition_data, $definition_id)) {
|
||||||
// New definition
|
// New definition
|
||||||
if ($definition_id == NO_DEFINITION_ID) {
|
if ($definition_id == NO_DEFINITION_ID) {
|
||||||
$definition_values = json_decode(html_entity_decode($this->request->getPost('definition_values')));
|
$definition_values = json_decode(html_entity_decode($this->request->getPost('definition_values')));
|
||||||
|
|||||||
@@ -3,44 +3,28 @@
|
|||||||
namespace App\Controllers;
|
namespace App\Controllers;
|
||||||
|
|
||||||
use CodeIgniter\Controller;
|
use CodeIgniter\Controller;
|
||||||
use CodeIgniter\HTTP\CLIRequest;
|
|
||||||
use CodeIgniter\HTTP\IncomingRequest;
|
|
||||||
use CodeIgniter\HTTP\RequestInterface;
|
use CodeIgniter\HTTP\RequestInterface;
|
||||||
use CodeIgniter\HTTP\ResponseInterface;
|
use CodeIgniter\HTTP\ResponseInterface;
|
||||||
use Psr\Log\LoggerInterface;
|
use Psr\Log\LoggerInterface;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Class BaseController
|
|
||||||
*
|
|
||||||
* BaseController provides a convenient place for loading components
|
* BaseController provides a convenient place for loading components
|
||||||
* and performing functions that are needed by all your controllers.
|
* and performing functions that are needed by all your controllers.
|
||||||
* Extend this class in any new controllers:
|
|
||||||
* class Home extends BaseController
|
|
||||||
*
|
*
|
||||||
* For security be sure to declare any new methods as protected or private.
|
* Extend this class in any new controllers:
|
||||||
|
* ```
|
||||||
|
* class Home extends BaseController
|
||||||
|
* ```
|
||||||
|
*
|
||||||
|
* For security, be sure to declare any new methods as protected or private.
|
||||||
*/
|
*/
|
||||||
abstract class BaseController extends Controller
|
abstract class BaseController extends Controller
|
||||||
{
|
{
|
||||||
/**
|
|
||||||
* Instance of the main Request object.
|
|
||||||
*
|
|
||||||
* @var CLIRequest|IncomingRequest
|
|
||||||
*/
|
|
||||||
protected $request;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* An array of helpers to be loaded automatically upon
|
|
||||||
* class instantiation. These helpers will be available
|
|
||||||
* to all other controllers that extend BaseController.
|
|
||||||
*
|
|
||||||
* @var list<string>
|
|
||||||
*/
|
|
||||||
protected $helpers = [];
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Be sure to declare properties for any property fetch you initialized.
|
* Be sure to declare properties for any property fetch you initialized.
|
||||||
* The creation of dynamic property is deprecated in PHP 8.2.
|
* The creation of dynamic property is deprecated in PHP 8.2.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
// protected $session;
|
// protected $session;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -48,11 +32,14 @@ abstract class BaseController extends Controller
|
|||||||
*/
|
*/
|
||||||
public function initController(RequestInterface $request, ResponseInterface $response, LoggerInterface $logger)
|
public function initController(RequestInterface $request, ResponseInterface $response, LoggerInterface $logger)
|
||||||
{
|
{
|
||||||
// Do Not Edit This Line
|
// Load here all helpers you want to be available in your controllers that extend BaseController.
|
||||||
|
// Caution: Do not put the this below the parent::initController() call below.
|
||||||
|
// $this->helpers = ['form', 'url'];
|
||||||
|
|
||||||
|
// Caution: Do not edit this line.
|
||||||
parent::initController($request, $response, $logger);
|
parent::initController($request, $response, $logger);
|
||||||
|
|
||||||
// Preload any models, libraries, etc, here.
|
// Preload any models, libraries, etc, here.
|
||||||
|
// $this->session = service('session');
|
||||||
// E.g.: $this->session = service('session');
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -221,6 +221,7 @@ class Config extends Secure_Controller
|
|||||||
*/
|
*/
|
||||||
public function getIndex(): string
|
public function getIndex(): string
|
||||||
{
|
{
|
||||||
|
$data['config'] = $this->config;
|
||||||
$data['stock_locations'] = $this->stock_location->get_all()->getResultArray();
|
$data['stock_locations'] = $this->stock_location->get_all()->getResultArray();
|
||||||
$data['dinner_tables'] = $this->dinner_table->get_all()->getResultArray();
|
$data['dinner_tables'] = $this->dinner_table->get_all()->getResultArray();
|
||||||
$data['customer_rewards'] = $this->customer_rewards->get_all()->getResultArray();
|
$data['customer_rewards'] = $this->customer_rewards->get_all()->getResultArray();
|
||||||
@@ -231,6 +232,8 @@ class Config extends Secure_Controller
|
|||||||
$data['line_sequence_options'] = $this->sale_lib->get_line_sequence_options();
|
$data['line_sequence_options'] = $this->sale_lib->get_line_sequence_options();
|
||||||
$data['register_mode_options'] = $this->sale_lib->get_register_mode_options();
|
$data['register_mode_options'] = $this->sale_lib->get_register_mode_options();
|
||||||
$data['invoice_type_options'] = $this->sale_lib->get_invoice_type_options();
|
$data['invoice_type_options'] = $this->sale_lib->get_invoice_type_options();
|
||||||
|
$data['keyboardShortcutOptions'] = $this->sale_lib->getKeyShortcutsOptions();
|
||||||
|
$data['keyboardShortcuts'] = $this->sale_lib->getKeyShortcuts();
|
||||||
$data['rounding_options'] = rounding_mode::get_rounding_options();
|
$data['rounding_options'] = rounding_mode::get_rounding_options();
|
||||||
$data['tax_code_options'] = $this->tax_lib->get_tax_code_options();
|
$data['tax_code_options'] = $this->tax_lib->get_tax_code_options();
|
||||||
$data['tax_category_options'] = $this->tax_lib->get_tax_category_options();
|
$data['tax_category_options'] = $this->tax_lib->get_tax_category_options();
|
||||||
@@ -238,6 +241,28 @@ class Config extends Secure_Controller
|
|||||||
$data['show_office_group'] = $this->module->get_show_office_group();
|
$data['show_office_group'] = $this->module->get_show_office_group();
|
||||||
$data['currency_code'] = $this->config['currency_code'] ?? '';
|
$data['currency_code'] = $this->config['currency_code'] ?? '';
|
||||||
$data['dbVersion'] = mysqli_get_server_info($this->db->getConnection());
|
$data['dbVersion'] = mysqli_get_server_info($this->db->getConnection());
|
||||||
|
$data['scale_export_formats'] = [
|
||||||
|
'txt' => 'TXT',
|
||||||
|
'csv' => 'CSV',
|
||||||
|
];
|
||||||
|
$data['scale_export_charsets'] = [
|
||||||
|
'windows-1256' => 'Windows-1256',
|
||||||
|
'utf-8' => 'UTF-8',
|
||||||
|
'windows-1252' => 'Windows-1252',
|
||||||
|
];
|
||||||
|
$data['scale_export_delimiters'] = [
|
||||||
|
';' => ';',
|
||||||
|
',' => ',',
|
||||||
|
"\t" => 'Tab',
|
||||||
|
];
|
||||||
|
$data['scale_export_fields_options'] = [
|
||||||
|
'legacy_code' => lang('Items.item_number'),
|
||||||
|
'item_number' => lang('Items.item_number'),
|
||||||
|
'repeat_item_number' => lang('Items.item_number'),
|
||||||
|
'name' => lang('Items.name'),
|
||||||
|
'unit_price' => lang('Items.unit_price'),
|
||||||
|
'legacy_tail' => lang('Items.item_number'),
|
||||||
|
];
|
||||||
|
|
||||||
// Load all the license statements, they are already XSS cleaned in the private function
|
// Load all the license statements, they are already XSS cleaned in the private function
|
||||||
$data['licenses'] = $this->_licenses();
|
$data['licenses'] = $this->_licenses();
|
||||||
@@ -367,7 +392,7 @@ class Config extends Secure_Controller
|
|||||||
*/
|
*/
|
||||||
public function postSaveGeneral(): ResponseInterface
|
public function postSaveGeneral(): ResponseInterface
|
||||||
{
|
{
|
||||||
$batch_save_data = [
|
$batchSaveData = [
|
||||||
'theme' => $this->request->getPost('theme'),
|
'theme' => $this->request->getPost('theme'),
|
||||||
'login_form' => $this->request->getPost('login_form'),
|
'login_form' => $this->request->getPost('login_form'),
|
||||||
'default_sales_discount_type' => $this->request->getPost('default_sales_discount_type') != null,
|
'default_sales_discount_type' => $this->request->getPost('default_sales_discount_type') != null,
|
||||||
@@ -391,6 +416,7 @@ class Config extends Secure_Controller
|
|||||||
'suggestions_third_column' => $this->validateSuggestionsColumn($this->request->getPost('suggestions_third_column'), 'other'),
|
'suggestions_third_column' => $this->validateSuggestionsColumn($this->request->getPost('suggestions_third_column'), 'other'),
|
||||||
'giftcard_number' => $this->request->getPost('giftcard_number'),
|
'giftcard_number' => $this->request->getPost('giftcard_number'),
|
||||||
'derive_sale_quantity' => $this->request->getPost('derive_sale_quantity') != null,
|
'derive_sale_quantity' => $this->request->getPost('derive_sale_quantity') != null,
|
||||||
|
'customer_display_enabled' => $this->request->getPost('customer_display_enabled') != null,
|
||||||
'multi_pack_enabled' => $this->request->getPost('multi_pack_enabled') != null,
|
'multi_pack_enabled' => $this->request->getPost('multi_pack_enabled') != null,
|
||||||
'include_hsn' => $this->request->getPost('include_hsn') != null,
|
'include_hsn' => $this->request->getPost('include_hsn') != null,
|
||||||
'category_dropdown' => $this->request->getPost('category_dropdown') != null
|
'category_dropdown' => $this->request->getPost('category_dropdown') != null
|
||||||
@@ -398,19 +424,26 @@ class Config extends Secure_Controller
|
|||||||
|
|
||||||
$this->module->set_show_office_group($this->request->getPost('show_office_group') != null);
|
$this->module->set_show_office_group($this->request->getPost('show_office_group') != null);
|
||||||
|
|
||||||
if ($batch_save_data['category_dropdown'] == 1) {
|
$this->db->transStart();
|
||||||
$definition_data['definition_name'] = 'ospos_category';
|
|
||||||
$definition_data['definition_flags'] = 0;
|
|
||||||
$definition_data['definition_type'] = 'DROPDOWN';
|
|
||||||
$definition_data['definition_id'] = CATEGORY_DEFINITION_ID;
|
|
||||||
$definition_data['deleted'] = 0;
|
|
||||||
|
|
||||||
$this->attribute->save_definition($definition_data, CATEGORY_DEFINITION_ID);
|
$attributeSuccess = true;
|
||||||
} elseif ($batch_save_data['category_dropdown'] == NO_DEFINITION_ID) {
|
if ($batchSaveData['category_dropdown']) {
|
||||||
$this->attribute->deleteDefinition(CATEGORY_DEFINITION_ID);
|
$definitionData['definition_name'] = 'ospos_category';
|
||||||
|
$definitionData['definition_flags'] = 0;
|
||||||
|
$definitionData['definition_type'] = 'DROPDOWN';
|
||||||
|
$definitionData['definition_id'] = CATEGORY_DEFINITION_ID;
|
||||||
|
$definitionData['deleted'] = 0;
|
||||||
|
|
||||||
|
$attributeSuccess = $this->attribute->saveDefinition($definitionData, CATEGORY_DEFINITION_ID);
|
||||||
|
} elseif ($batchSaveData['category_dropdown'] == NO_DEFINITION_ID) {
|
||||||
|
$attributeSuccess = $this->attribute->deleteDefinition(CATEGORY_DEFINITION_ID);
|
||||||
}
|
}
|
||||||
|
|
||||||
$success = $this->appconfig->batch_save($batch_save_data);
|
$success = $attributeSuccess && $this->appconfig->batch_save($batchSaveData);
|
||||||
|
|
||||||
|
$this->db->transComplete();
|
||||||
|
|
||||||
|
$success = $success && $this->db->transStatus();
|
||||||
|
|
||||||
return $this->response->setJSON(['success' => $success, 'message' => lang('Config.saved_' . ($success ? '' : 'un') . 'successfully')]);
|
return $this->response->setJSON(['success' => $success, 'message' => lang('Config.saved_' . ($success ? '' : 'un') . 'successfully')]);
|
||||||
}
|
}
|
||||||
@@ -423,32 +456,35 @@ class Config extends Secure_Controller
|
|||||||
*/
|
*/
|
||||||
public function postCheckNumberLocale(): ResponseInterface
|
public function postCheckNumberLocale(): ResponseInterface
|
||||||
{
|
{
|
||||||
$number_locale = $this->request->getPost('number_locale');
|
$numberLocale = $this->request->getPost('number_locale');
|
||||||
$save_number_locale = $this->request->getPost('save_number_locale');
|
$saveNumberLocale = $this->request->getPost('save_number_locale');
|
||||||
|
$postedCurrencySymbol = $this->request->getPost('currency_symbol');
|
||||||
|
$postedCurrencyCode = $this->request->getPost('currency_code');
|
||||||
|
|
||||||
$fmt = new NumberFormatter($number_locale, NumberFormatter::CURRENCY);
|
$fmt = new NumberFormatter($numberLocale, NumberFormatter::CURRENCY);
|
||||||
if ($number_locale != $save_number_locale) {
|
|
||||||
$currency_symbol = $fmt->getSymbol(NumberFormatter::CURRENCY_SYMBOL);
|
// Use posted values if provided, otherwise fall back to locale defaults
|
||||||
$currency_code = $fmt->getTextAttribute(NumberFormatter::CURRENCY_CODE);
|
$currencySymbol = $postedCurrencySymbol !== '' ? $postedCurrencySymbol : $fmt->getSymbol(NumberFormatter::CURRENCY_SYMBOL);
|
||||||
$save_number_locale = $number_locale;
|
$currencyCode = $postedCurrencyCode !== '' ? $postedCurrencyCode : $fmt->getTextAttribute(NumberFormatter::CURRENCY_CODE);
|
||||||
} else {
|
|
||||||
$currency_symbol = empty($this->request->getPost('currency_symbol')) ? $fmt->getSymbol(NumberFormatter::CURRENCY_SYMBOL) : $this->request->getPost('currency_symbol');
|
// Update saved locale if it changed
|
||||||
$currency_code = empty($this->request->getPost('currency_code')) ? $fmt->getTextAttribute(NumberFormatter::CURRENCY_CODE) : $this->request->getPost('currency_code');
|
if ($numberLocale !== $saveNumberLocale) {
|
||||||
|
$saveNumberLocale = $numberLocale;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($this->request->getPost('thousands_separator') == 'false') {
|
if ($this->request->getPost('thousands_separator') == 'false') {
|
||||||
$fmt->setTextAttribute(NumberFormatter::GROUPING_SEPARATOR_SYMBOL, '');
|
$fmt->setTextAttribute(NumberFormatter::GROUPING_SEPARATOR_SYMBOL, '');
|
||||||
}
|
}
|
||||||
|
|
||||||
$fmt->setSymbol(NumberFormatter::CURRENCY_SYMBOL, $currency_symbol);
|
$fmt->setSymbol(NumberFormatter::CURRENCY_SYMBOL, $currencySymbol);
|
||||||
$number_local_example = $fmt->format(1234567890.12300);
|
$numberLocaleExample = $fmt->format(1234567890.12300);
|
||||||
|
|
||||||
return $this->response->setJSON([
|
return $this->response->setJSON([
|
||||||
'success' => $number_local_example != false,
|
'success' => $numberLocaleExample != false,
|
||||||
'save_number_locale' => $save_number_locale,
|
'save_number_locale' => $saveNumberLocale,
|
||||||
'number_locale_example' => $number_local_example,
|
'number_locale_example' => $numberLocaleExample,
|
||||||
'currency_symbol' => $currency_symbol,
|
'currency_symbol' => $currencySymbol,
|
||||||
'currency_code' => $currency_code,
|
'currency_code' => $currencyCode,
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -461,13 +497,36 @@ class Config extends Secure_Controller
|
|||||||
*/
|
*/
|
||||||
public function postSaveLocale(): ResponseInterface
|
public function postSaveLocale(): ResponseInterface
|
||||||
{
|
{
|
||||||
$exploded = explode(":", $this->request->getPost('language'));
|
$language = trim((string) $this->request->getPost('language'));
|
||||||
|
$languageCode = 'en';
|
||||||
|
$languageName = 'english';
|
||||||
|
|
||||||
|
if ($language !== '' && str_contains($language, ':')) {
|
||||||
|
$exploded = array_map('trim', explode(':', $language, 2));
|
||||||
|
|
||||||
|
if (count($exploded) === 2) {
|
||||||
|
$languageCode = htmlspecialchars($exploded[0]);
|
||||||
|
$languageName = htmlspecialchars($exploded[1]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
$currency_symbol = $this->request->getPost('currency_symbol');
|
$currency_symbol = $this->request->getPost('currency_symbol');
|
||||||
|
$secondaryCurrencyCode = strtoupper(trim((string) $this->request->getPost('secondary_currency_code')));
|
||||||
|
|
||||||
|
if (!preg_match('/^[A-Z]{3}$/', $secondaryCurrencyCode)) {
|
||||||
|
$secondaryCurrencyCode = '';
|
||||||
|
}
|
||||||
|
|
||||||
$batch_save_data = [
|
$batch_save_data = [
|
||||||
'currency_symbol' => htmlspecialchars($currency_symbol ?? ''),
|
'currency_symbol' => htmlspecialchars($currency_symbol ?? ''),
|
||||||
'currency_code' => $this->request->getPost('currency_code'),
|
'currency_code' => $this->request->getPost('currency_code'),
|
||||||
'language_code' => $exploded[0],
|
'secondary_currency_enabled' => $this->request->getPost('secondary_currency_enabled') != null,
|
||||||
'language' => $exploded[1],
|
'secondary_currency_symbol' => htmlspecialchars($this->request->getPost('secondary_currency_symbol') ?? ''),
|
||||||
|
'secondary_currency_code' => $secondaryCurrencyCode,
|
||||||
|
'secondary_currency_rate' => $this->request->getPost('secondary_currency_rate', FILTER_SANITIZE_NUMBER_FLOAT, FILTER_FLAG_ALLOW_FRACTION),
|
||||||
|
'secondary_currency_decimals' => $this->request->getPost('secondary_currency_decimals', FILTER_SANITIZE_NUMBER_INT),
|
||||||
|
'language_code' => $languageCode,
|
||||||
|
'language' => $languageName,
|
||||||
'timezone' => $this->request->getPost('timezone'),
|
'timezone' => $this->request->getPost('timezone'),
|
||||||
'dateformat' => $this->request->getPost('dateformat'),
|
'dateformat' => $this->request->getPost('dateformat'),
|
||||||
'timeformat' => $this->request->getPost('timeformat'),
|
'timeformat' => $this->request->getPost('timeformat'),
|
||||||
@@ -504,9 +563,24 @@ class Config extends Secure_Controller
|
|||||||
$password = $this->encrypter->encrypt($this->request->getPost('smtp_pass'));
|
$password = $this->encrypter->encrypt($this->request->getPost('smtp_pass'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$protocol = $this->request->getPost('protocol');
|
||||||
|
$mailpath = $this->request->getPost('mailpath');
|
||||||
|
|
||||||
|
// Validate mailpath: required for sendmail, optional for others but must be safe if provided
|
||||||
|
$isMailpathRequired = ($protocol === 'sendmail');
|
||||||
|
$isMailpathProvided = !empty($mailpath);
|
||||||
|
$isMailpathValid = $isMailpathProvided && preg_match('/^[a-zA-Z0-9_\-\/.]+$/', $mailpath);
|
||||||
|
|
||||||
|
if (($isMailpathRequired && !$isMailpathProvided) || ($isMailpathProvided && !$isMailpathValid)) {
|
||||||
|
return $this->response->setJSON([
|
||||||
|
'success' => false,
|
||||||
|
'message' => lang('Config.mailpath_invalid')
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
$batch_save_data = [
|
$batch_save_data = [
|
||||||
'protocol' => $this->request->getPost('protocol'),
|
'protocol' => $protocol,
|
||||||
'mailpath' => $this->request->getPost('mailpath'),
|
'mailpath' => $mailpath,
|
||||||
'smtp_host' => $this->request->getPost('smtp_host'),
|
'smtp_host' => $this->request->getPost('smtp_host'),
|
||||||
'smtp_user' => $this->request->getPost('smtp_user'),
|
'smtp_user' => $this->request->getPost('smtp_user'),
|
||||||
'smtp_pass' => $password,
|
'smtp_pass' => $password,
|
||||||
@@ -906,6 +980,7 @@ class Config extends Secure_Controller
|
|||||||
'receipt_show_tax_ind' => $this->request->getPost('receipt_show_tax_ind') != null,
|
'receipt_show_tax_ind' => $this->request->getPost('receipt_show_tax_ind') != null,
|
||||||
'receipt_show_total_discount' => $this->request->getPost('receipt_show_total_discount') != null,
|
'receipt_show_total_discount' => $this->request->getPost('receipt_show_total_discount') != null,
|
||||||
'receipt_show_description' => $this->request->getPost('receipt_show_description') != null,
|
'receipt_show_description' => $this->request->getPost('receipt_show_description') != null,
|
||||||
|
'receipt_show_secondary_currency' => $this->request->getPost('receipt_show_secondary_currency') != null,
|
||||||
'receipt_show_serialnumber' => $this->request->getPost('receipt_show_serialnumber') != null,
|
'receipt_show_serialnumber' => $this->request->getPost('receipt_show_serialnumber') != null,
|
||||||
'print_silently' => $this->request->getPost('print_silently') != null,
|
'print_silently' => $this->request->getPost('print_silently') != null,
|
||||||
'print_header' => $this->request->getPost('print_header') != null,
|
'print_header' => $this->request->getPost('print_header') != null,
|
||||||
@@ -921,6 +996,44 @@ class Config extends Secure_Controller
|
|||||||
return $this->response->setJSON(['success' => $success, 'message' => lang('Config.saved_' . ($success ? '' : 'un') . 'successfully')]);
|
return $this->response->setJSON(['success' => $success, 'message' => lang('Config.saved_' . ($success ? '' : 'un') . 'successfully')]);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Saves keyboard shortcut bindings.
|
||||||
|
*
|
||||||
|
* @return ResponseInterface
|
||||||
|
* @noinspection PhpUnused
|
||||||
|
*/
|
||||||
|
public function postSaveShortcuts(): ResponseInterface
|
||||||
|
{
|
||||||
|
$allowedShortcuts = array_keys($this->sale_lib->getKeyShortcutsOptions());
|
||||||
|
$currentShortcuts = $this->sale_lib->getKeyShortcuts();
|
||||||
|
$batchSaveData = [];
|
||||||
|
|
||||||
|
foreach ($currentShortcuts as $name => $shortcut) {
|
||||||
|
$postedValue = trim((string) $this->request->getPost('key_' . $name));
|
||||||
|
|
||||||
|
if (!in_array($postedValue, $allowedShortcuts, true)) {
|
||||||
|
$postedValue = $shortcut['value'];
|
||||||
|
}
|
||||||
|
|
||||||
|
$batchSaveData['key_' . $name] = $postedValue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$duplicateValues = array_filter(array_count_values($batchSaveData), static fn(int $count): bool => $count > 1);
|
||||||
|
if (!empty($duplicateValues)) {
|
||||||
|
return $this->response->setJSON([
|
||||||
|
'success' => false,
|
||||||
|
'message' => lang('Config.shortcuts_duplicate_bindings')
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$success = $this->appconfig->batch_save($batchSaveData);
|
||||||
|
|
||||||
|
return $this->response->setJSON([
|
||||||
|
'success' => $success,
|
||||||
|
'message' => lang('Config.saved_' . ($success ? '' : 'un') . 'successfully')
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Saves invoice configuration. Used in app/Views/configs/invoice_config.php.
|
* Saves invoice configuration. Used in app/Views/configs/invoice_config.php.
|
||||||
*
|
*
|
||||||
@@ -1000,3 +1113,6 @@ class Config extends Secure_Controller
|
|||||||
return in_array($column, $allowed, true) ? $column : $fallback;
|
return in_array($column, $allowed, true) ? $column : $fallback;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Controllers;
|
namespace App\Controllers;
|
||||||
|
|
||||||
|
use App\Libraries\MY_Migration;
|
||||||
use CodeIgniter\HTTP\RedirectResponse;
|
use CodeIgniter\HTTP\RedirectResponse;
|
||||||
use CodeIgniter\HTTP\ResponseInterface;
|
use CodeIgniter\HTTP\ResponseInterface;
|
||||||
|
|
||||||
@@ -34,15 +35,15 @@ class Home extends Secure_Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load "change employee password" form
|
* Load the "change employee password" form
|
||||||
*
|
*
|
||||||
|
* @param int $employeeId
|
||||||
* @return ResponseInterface|string
|
* @return ResponseInterface|string
|
||||||
* @noinspection PhpUnused
|
|
||||||
*/
|
*/
|
||||||
public function getChangePassword(int $employeeId = NEW_ENTRY)
|
public function getChangePassword(int $employeeId = NEW_ENTRY): ResponseInterface|string
|
||||||
{
|
{
|
||||||
$loggedInEmployee = $this->employee->get_logged_in_employee_info();
|
$loggedInEmployee = $this->employee->get_logged_in_employee_info();
|
||||||
$currentPersonId = $loggedInEmployee->person_id;
|
$currentPersonId = (int) $loggedInEmployee->person_id;
|
||||||
|
|
||||||
$employeeId = $employeeId === NEW_ENTRY ? $currentPersonId : $employeeId;
|
$employeeId = $employeeId === NEW_ENTRY ? $currentPersonId : $employeeId;
|
||||||
|
|
||||||
@@ -67,10 +68,11 @@ class Home extends Secure_Controller
|
|||||||
public function postSave(int $employeeId = NEW_ENTRY): ResponseInterface
|
public function postSave(int $employeeId = NEW_ENTRY): ResponseInterface
|
||||||
{
|
{
|
||||||
$currentUser = $this->employee->get_logged_in_employee_info();
|
$currentUser = $this->employee->get_logged_in_employee_info();
|
||||||
|
$currentPersonId = (int) $currentUser->person_id;
|
||||||
|
|
||||||
$employeeId = $employeeId === NEW_ENTRY ? $currentUser->person_id : $employeeId;
|
$employeeId = $employeeId === NEW_ENTRY ? $currentPersonId : $employeeId;
|
||||||
|
|
||||||
if (!$this->employee->isAdmin($currentUser->person_id) && $employeeId !== $currentUser->person_id) {
|
if (!$this->employee->isAdmin($currentPersonId) && $employeeId !== $currentPersonId) {
|
||||||
return $this->response->setStatusCode(403)->setJSON([
|
return $this->response->setStatusCode(403)->setJSON([
|
||||||
'success' => false,
|
'success' => false,
|
||||||
'message' => lang('Employees.unauthorized_modify')
|
'message' => lang('Employees.unauthorized_modify')
|
||||||
@@ -81,7 +83,7 @@ class Home extends Secure_Controller
|
|||||||
if ($this->employee->check_password($this->request->getPost('username', FILTER_SANITIZE_FULL_SPECIAL_CHARS), $this->request->getPost('current_password'))) {
|
if ($this->employee->check_password($this->request->getPost('username', FILTER_SANITIZE_FULL_SPECIAL_CHARS), $this->request->getPost('current_password'))) {
|
||||||
// Validate password length BEFORE hashing
|
// Validate password length BEFORE hashing
|
||||||
$new_password = $this->request->getPost('password');
|
$new_password = $this->request->getPost('password');
|
||||||
|
|
||||||
if (strlen($new_password) < 8) {
|
if (strlen($new_password) < 8) {
|
||||||
return $this->response->setJSON([
|
return $this->response->setJSON([
|
||||||
'success' => false,
|
'success' => false,
|
||||||
@@ -89,7 +91,7 @@ class Home extends Secure_Controller
|
|||||||
'id' => NEW_ENTRY
|
'id' => NEW_ENTRY
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$employee_data = [
|
$employee_data = [
|
||||||
'username' => $this->request->getPost('username', FILTER_SANITIZE_FULL_SPECIAL_CHARS),
|
'username' => $this->request->getPost('username', FILTER_SANITIZE_FULL_SPECIAL_CHARS),
|
||||||
'password' => password_hash($new_password, PASSWORD_DEFAULT),
|
'password' => password_hash($new_password, PASSWORD_DEFAULT),
|
||||||
@@ -124,4 +126,4 @@ class Home extends Secure_Controller
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,7 +4,6 @@ namespace App\Controllers;
|
|||||||
|
|
||||||
use App\Libraries\Barcode_lib;
|
use App\Libraries\Barcode_lib;
|
||||||
use App\Libraries\Item_lib;
|
use App\Libraries\Item_lib;
|
||||||
|
|
||||||
use App\Models\Attribute;
|
use App\Models\Attribute;
|
||||||
use App\Models\Inventory;
|
use App\Models\Inventory;
|
||||||
use App\Models\Item;
|
use App\Models\Item;
|
||||||
@@ -14,7 +13,6 @@ use App\Models\Item_taxes;
|
|||||||
use App\Models\Stock_location;
|
use App\Models\Stock_location;
|
||||||
use App\Models\Supplier;
|
use App\Models\Supplier;
|
||||||
use App\Models\Tax_category;
|
use App\Models\Tax_category;
|
||||||
|
|
||||||
use CodeIgniter\HTTP\ResponseInterface;
|
use CodeIgniter\HTTP\ResponseInterface;
|
||||||
use CodeIgniter\Images\Handlers\BaseHandler;
|
use CodeIgniter\Images\Handlers\BaseHandler;
|
||||||
use CodeIgniter\HTTP\DownloadResponse;
|
use CodeIgniter\HTTP\DownloadResponse;
|
||||||
@@ -73,7 +71,7 @@ class Items extends Secure_Controller
|
|||||||
$this->session->set('allow_temp_items', 0);
|
$this->session->set('allow_temp_items', 0);
|
||||||
|
|
||||||
$data['table_headers'] = get_items_manage_table_headers();
|
$data['table_headers'] = get_items_manage_table_headers();
|
||||||
|
|
||||||
// Restore stock_location from URL or session
|
// Restore stock_location from URL or session
|
||||||
$stockLocation = $this->request->getGet('stock_location', FILTER_SANITIZE_NUMBER_INT);
|
$stockLocation = $this->request->getGet('stock_location', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$data['stock_location'] = $stockLocation
|
$data['stock_location'] = $stockLocation
|
||||||
@@ -156,8 +154,23 @@ class Items extends Secure_Controller
|
|||||||
{
|
{
|
||||||
helper('file');
|
helper('file');
|
||||||
|
|
||||||
$pic_filename = rawurldecode($pic_filename);
|
// Security: Sanitize filename to prevent path traversal
|
||||||
$file_extension = pathinfo($pic_filename, PATHINFO_EXTENSION);
|
// Use basename() to strip directory components and prevent '../' attacks
|
||||||
|
$pic_filename = basename(rawurldecode($pic_filename));
|
||||||
|
$file_extension = strtolower(pathinfo($pic_filename, PATHINFO_EXTENSION));
|
||||||
|
|
||||||
|
// Validate file extension against system-configured allowed image types
|
||||||
|
// Handle both legacy pipe-separated and current comma-separated formats
|
||||||
|
// Fallback to types that GD library can process for thumbnail generation
|
||||||
|
$allowed_types = $this->config['image_allowed_types'] ?? 'jpg,jpeg,gif,png,webp,bmp,tif,tiff';
|
||||||
|
$allowed_extensions = strpos($allowed_types, '|') !== false
|
||||||
|
? explode('|', $allowed_types)
|
||||||
|
: explode(',', $allowed_types);
|
||||||
|
|
||||||
|
if (!in_array($file_extension, $allowed_extensions, true)) {
|
||||||
|
return $this->response->setStatusCode(400)->setBody('Invalid file type');
|
||||||
|
}
|
||||||
|
|
||||||
$images = glob("./uploads/item_pics/$pic_filename");
|
$images = glob("./uploads/item_pics/$pic_filename");
|
||||||
$base_path = './uploads/item_pics/' . pathinfo($pic_filename, PATHINFO_FILENAME);
|
$base_path = './uploads/item_pics/' . pathinfo($pic_filename, PATHINFO_FILENAME);
|
||||||
|
|
||||||
@@ -508,7 +521,7 @@ class Items extends Secure_Controller
|
|||||||
$data['definition_names'] = $this->attribute->get_definition_names();
|
$data['definition_names'] = $this->attribute->get_definition_names();
|
||||||
|
|
||||||
foreach ($data['definition_values'] as $definition_id => $definition_value) {
|
foreach ($data['definition_values'] as $definition_id => $definition_value) {
|
||||||
$attribute_value = $this->attribute->get_attribute_value($item_id, $definition_id);
|
$attribute_value = $this->attribute->getAttributeValue($item_id, $definition_id);
|
||||||
$attribute_id = (empty($attribute_value) || empty($attribute_value->attribute_id)) ? null : $attribute_value->attribute_id;
|
$attribute_id = (empty($attribute_value) || empty($attribute_value->attribute_id)) ? null : $attribute_value->attribute_id;
|
||||||
$values = &$data['definition_values'][$definition_id];
|
$values = &$data['definition_values'][$definition_id];
|
||||||
$values['attribute_id'] = $attribute_id;
|
$values['attribute_id'] = $attribute_id;
|
||||||
@@ -544,7 +557,7 @@ class Items extends Secure_Controller
|
|||||||
$data['definition_names'] = $this->attribute->get_definition_names();
|
$data['definition_names'] = $this->attribute->get_definition_names();
|
||||||
|
|
||||||
foreach ($data['definition_values'] as $definition_id => $definition_value) {
|
foreach ($data['definition_values'] as $definition_id => $definition_value) {
|
||||||
$attribute_value = $this->attribute->get_attribute_value($item_id, $definition_id);
|
$attribute_value = $this->attribute->getAttributeValue($item_id, $definition_id);
|
||||||
$attribute_id = (empty($attribute_value) || empty($attribute_value->attribute_id)) ? null : $attribute_value->attribute_id;
|
$attribute_id = (empty($attribute_value) || empty($attribute_value->attribute_id)) ? null : $attribute_value->attribute_id;
|
||||||
$values = &$data['definition_values'][$definition_id];
|
$values = &$data['definition_values'][$definition_id];
|
||||||
$values['attribute_id'] = $attribute_id;
|
$values['attribute_id'] = $attribute_id;
|
||||||
@@ -713,7 +726,7 @@ class Items extends Secure_Controller
|
|||||||
$item_quantity = $this->item_quantity->get_item_quantity($item_id, $location['location_id']);
|
$item_quantity = $this->item_quantity->get_item_quantity($item_id, $location['location_id']);
|
||||||
|
|
||||||
if ($item_quantity->quantity != $updated_quantity || $new_item) {
|
if ($item_quantity->quantity != $updated_quantity || $new_item) {
|
||||||
$success &= $this->item_quantity->save_value($location_detail, $item_id, $location['location_id']);
|
$success = $success && $this->item_quantity->save_value($location_detail, $item_id, $location['location_id']);
|
||||||
|
|
||||||
$inv_data = [
|
$inv_data = [
|
||||||
'trans_date' => date('Y-m-d H:i:s'),
|
'trans_date' => date('Y-m-d H:i:s'),
|
||||||
@@ -724,10 +737,10 @@ class Items extends Secure_Controller
|
|||||||
'trans_inventory' => $updated_quantity - $item_quantity->quantity
|
'trans_inventory' => $updated_quantity - $item_quantity->quantity
|
||||||
];
|
];
|
||||||
|
|
||||||
$success &= $this->inventory->insert($inv_data, false);
|
$success = $success && $this->inventory->insert($inv_data, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$this->saveItemAttributes($item_id);
|
$success = $success && $this->saveItemAttributes($item_id);
|
||||||
|
|
||||||
if ($success && $upload_success) {
|
if ($success && $upload_success) {
|
||||||
$message = lang('Items.successful_' . ($new_item ? 'adding' : 'updating')) . ' ' . $item_data['name'];
|
$message = lang('Items.successful_' . ($new_item ? 'adding' : 'updating')) . ' ' . $item_data['name'];
|
||||||
@@ -777,7 +790,7 @@ class Items extends Secure_Controller
|
|||||||
|
|
||||||
$filename = $file->getClientName();
|
$filename = $file->getClientName();
|
||||||
$info = pathinfo($filename);
|
$info = pathinfo($filename);
|
||||||
|
|
||||||
// Sanitize filename to remove problematic characters like spaces
|
// Sanitize filename to remove problematic characters like spaces
|
||||||
$sanitized_name = preg_replace('/[^a-zA-Z0-9_\-\.]/', '_', $info['filename']);
|
$sanitized_name = preg_replace('/[^a-zA-Z0-9_\-\.]/', '_', $info['filename']);
|
||||||
|
|
||||||
@@ -940,7 +953,7 @@ class Items extends Secure_Controller
|
|||||||
*/
|
*/
|
||||||
public function getGenerateCsvFile(): DownloadResponse
|
public function getGenerateCsvFile(): DownloadResponse
|
||||||
{
|
{
|
||||||
helper('importfile_helper');
|
helper('importfile');
|
||||||
$name = 'import_items.csv';
|
$name = 'import_items.csv';
|
||||||
$allowed_locations = $this->stock_location->get_allowed_locations();
|
$allowed_locations = $this->stock_location->get_allowed_locations();
|
||||||
$allowed_attributes = $this->attribute->get_definition_names();
|
$allowed_attributes = $this->attribute->get_definition_names();
|
||||||
@@ -959,14 +972,13 @@ class Items extends Secure_Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Imports items from CSV formatted file.
|
* Imports items from a CSV formatted file.
|
||||||
* @return ResponseInterface
|
* @return ResponseInterface
|
||||||
* @throws ReflectionException
|
|
||||||
* @noinspection PhpUnused
|
* @noinspection PhpUnused
|
||||||
*/
|
*/
|
||||||
public function postImportCsvFile(): ResponseInterface
|
public function postImportCsvFile(): ResponseInterface
|
||||||
{
|
{
|
||||||
helper('importfile_helper');
|
helper('importfile');
|
||||||
try {
|
try {
|
||||||
if ($_FILES['file_path']['error'] !== UPLOAD_ERR_OK) {
|
if ($_FILES['file_path']['error'] !== UPLOAD_ERR_OK) {
|
||||||
return $this->response->setJSON(['success' => false, 'message' => lang('Items.csv_import_failed')]);
|
return $this->response->setJSON(['success' => false, 'message' => lang('Items.csv_import_failed')]);
|
||||||
@@ -975,33 +987,33 @@ class Items extends Secure_Controller
|
|||||||
set_time_limit(240);
|
set_time_limit(240);
|
||||||
|
|
||||||
$failCodes = [];
|
$failCodes = [];
|
||||||
$csv_rows = get_csv_file($_FILES['file_path']['tmp_name']);
|
$csvRows = get_csv_file($_FILES['file_path']['tmp_name']);
|
||||||
$employee_id = $this->employee->get_logged_in_employee_info()->person_id;
|
$employeeId = $this->employee->get_logged_in_employee_info()->person_id;
|
||||||
$allowed_stock_locations = $this->stock_location->get_allowed_locations();
|
$allowedStockLocations = $this->stock_location->get_allowed_locations();
|
||||||
$attribute_definition_names = $this->attribute->get_definition_names();
|
$attributeDefinitionNames = $this->attribute->get_definition_names();
|
||||||
|
|
||||||
unset($attribute_definition_names[NEW_ENTRY]); // Removes the common_none_selected_text from the array
|
unset($attributeDefinitionNames[NEW_ENTRY]); // Removes the common_none_selected_text from the array
|
||||||
|
|
||||||
$attribute_data = [];
|
$attributeData = [];
|
||||||
|
|
||||||
foreach ($attribute_definition_names as $definition_name) {
|
foreach ($attributeDefinitionNames as $definitionName) {
|
||||||
$attribute_data[$definition_name] = $this->attribute->get_definition_by_name($definition_name)[0];
|
$attributeData[$definitionName] = $this->attribute->get_definition_by_name($definitionName)[0];
|
||||||
|
|
||||||
if ($attribute_data[$definition_name]['definition_type'] === DROPDOWN) {
|
if ($attributeData[$definitionName]['definition_type'] === DROPDOWN) {
|
||||||
$attribute_data[$definition_name]['dropdown_values'] = $this->attribute->get_definition_values($attribute_data[$definition_name]['definition_id']);
|
$attributeData[$definitionName]['dropdown_values'] = $this->attribute->get_definition_values($attributeData[$definitionName]['definition_id']);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$db = db_connect();
|
$db = db_connect();
|
||||||
$db->transBegin(); // TODO: This section needs to be reworked so that the data array is being created then passed to the Item model because $db doesn't exist in the controller without being instantiated, but database operations should be restricted to the model
|
$db->transBegin(); // TODO: This section needs to be reworked so that the data array is being created then passed to the Item model because $db doesn't exist in the controller without being instantiated, but database operations should be restricted to the model
|
||||||
|
|
||||||
foreach ($csv_rows as $key => $row) {
|
foreach ($csvRows as $key => $row) {
|
||||||
$is_failed_row = false;
|
$isFailedRow = false;
|
||||||
$item_id = (int)$row['Id'];
|
$itemId = (int)$row['Id'];
|
||||||
$is_update = ($item_id > 0);
|
$isUpdate = ($itemId > 0);
|
||||||
$item_data = [
|
$itemData = [
|
||||||
'item_id' => $item_id,
|
'item_id' => $itemId,
|
||||||
'name' => $row['Item Name'],
|
'name' => $row['Item Name'],
|
||||||
'description' => $row['Description'],
|
'description' => filter_var($row['Description'], FILTER_SANITIZE_FULL_SPECIAL_CHARS),
|
||||||
'category' => $row['Category'],
|
'category' => $row['Category'],
|
||||||
'cost_price' => $row['Cost Price'],
|
'cost_price' => $row['Cost Price'],
|
||||||
'unit_price' => $row['Unit Price'],
|
'unit_price' => $row['Unit Price'],
|
||||||
@@ -1011,25 +1023,26 @@ class Items extends Secure_Controller
|
|||||||
'pic_filename' => $row['Image']
|
'pic_filename' => $row['Image']
|
||||||
];
|
];
|
||||||
|
|
||||||
if (!empty($row['supplier ID'])) {
|
if (!empty($row['Supplier ID'])) {
|
||||||
$item_data['supplier_id'] = $this->supplier->exists($row['Supplier ID']) ? $row['Supplier ID'] : null;
|
$itemData['supplier_id'] = $this->supplier->exists($row['Supplier ID']) ? $row['Supplier ID'] : null;
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($is_update) {
|
if ($isUpdate) {
|
||||||
$item_data['allow_alt_description'] = empty($row['Allow Alt Description']) ? null : $row['Allow Alt Description'];
|
$itemData['allow_alt_description'] = $row['Allow Alt Description'] === '' ? null : $row['Allow Alt Description'];
|
||||||
$item_data['is_serialized'] = empty($row['Item has Serial Number']) ? null : $row['Item has Serial Number'];
|
$itemData['is_serialized'] = $row['Item has Serial Number'] === '' ? null : $row['Item has Serial Number'];
|
||||||
} else {
|
} else {
|
||||||
$item_data['allow_alt_description'] = empty($row['Allow Alt Description']) ? '0' : '1';
|
$itemData['allow_alt_description'] = $row['Allow Alt Description'] === '' ? '0' : '1';
|
||||||
$item_data['is_serialized'] = empty($row['Item has Serial Number']) ? '0' : '1';
|
$itemData['is_serialized'] = $row['Item has Serial Number'] === '' ? '0' : '1';
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!empty($row['Barcode']) && !$is_update) {
|
if (!empty($row['Barcode'])) {
|
||||||
$item_data['item_number'] = $row['Barcode'];
|
$itemData['item_number'] = $row['Barcode'];
|
||||||
$is_failed_row = $this->item->item_number_exists($item_data['item_number']);
|
$isFailedRow = $this->item->item_number_exists($itemData['item_number'], $itemId);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$is_failed_row) {
|
if (!$isFailedRow) {
|
||||||
$invalidLocations = $this->validateCSVStockLocations($row, $allowedStockLocations);
|
$allowedStockLocations = $this->stock_location->get_allowed_locations();
|
||||||
|
$isFailedRow = $this->validateCSVData($row, $itemData, $allowedStockLocations, $attributeDefinitionNames, $attributeData);
|
||||||
if (!empty($invalidLocations)) {
|
if (!empty($invalidLocations)) {
|
||||||
$isFailedRow = true;
|
$isFailedRow = true;
|
||||||
log_message('error', 'CSV import: Invalid stock location(s) found: ' . implode(', ', $invalidLocations));
|
log_message('error', 'CSV import: Invalid stock location(s) found: ' . implode(', ', $invalidLocations));
|
||||||
@@ -1037,28 +1050,41 @@ class Items extends Secure_Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Remove false, null, '' and empty strings but keep 0
|
// Remove false, null, '' and empty strings but keep 0
|
||||||
$item_data = array_filter($item_data, function ($value) {
|
$itemData = array_filter($itemData, function ($value) {
|
||||||
return $value !== null && strlen($value);
|
return $value !== null && strlen($value);
|
||||||
});
|
});
|
||||||
|
|
||||||
if (!$is_failed_row && $this->item->save_value($item_data, $item_id)) {
|
if (!$isFailedRow && $this->item->save_value($itemData, $itemId)) {
|
||||||
$this->save_tax_data($row, $item_data);
|
if (!$this->save_tax_data($row, $itemData)) {
|
||||||
$this->save_inventory_quantities($row, $item_data, $allowed_stock_locations, $employee_id);
|
$isFailedRow = true;
|
||||||
$is_failed_row = $this->save_attribute_data($row, $item_data, $attribute_data); // TODO: $is_failed_row never gets used after this.
|
}
|
||||||
|
if (!$this->save_inventory_quantities($row, $itemData, $allowedStockLocations, $employeeId)) {
|
||||||
|
$isFailedRow = true;
|
||||||
|
}
|
||||||
|
$csvAttributeValues = $this->extractAttributeData($row);
|
||||||
|
if (!$this->attribute->saveCSVRowAttributeData($csvAttributeValues, $itemData, $attributeData)) {
|
||||||
|
$isFailedRow = true;
|
||||||
|
}
|
||||||
|
if ($isFailedRow) {
|
||||||
|
$failedRow = $key + 2;
|
||||||
|
$failCodes[] = $failedRow;
|
||||||
|
log_message('error', "CSV Item import failed on line $failedRow while saving item.");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
if ($is_update) {
|
if ($isUpdate) {
|
||||||
$item_data = array_merge($item_data, get_object_vars($this->item->get_info_by_id_or_number($item_id)));
|
$itemData = array_merge($itemData, get_object_vars($this->item->get_info_by_id_or_number($itemId)));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
$failed_row = $key + 2;
|
$failedRow = $key + 2;
|
||||||
$failCodes[] = $failed_row;
|
$failCodes[] = $failedRow;
|
||||||
log_message('error', "CSV Item import failed on line $failed_row. This item was not imported.");
|
log_message('error', "CSV Item import failed on line $failedRow. This item was not imported.");
|
||||||
}
|
}
|
||||||
|
|
||||||
unset($csv_rows[$key]);
|
unset($csvRows[$key]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$csv_rows = null;
|
$csvRows = null;
|
||||||
|
|
||||||
if (count($failCodes) > 0) {
|
if (count($failCodes) > 0) {
|
||||||
$message = lang('Items.csv_import_partially_failed', [count($failCodes), implode(', ', $failCodes)]);
|
$message = lang('Items.csv_import_partially_failed', [count($failCodes), implode(', ', $failCodes)]);
|
||||||
@@ -1066,6 +1092,7 @@ class Items extends Secure_Controller
|
|||||||
return $this->response->setJSON(['success' => false, 'message' => $message]);
|
return $this->response->setJSON(['success' => false, 'message' => $message]);
|
||||||
} else {
|
} else {
|
||||||
$db->transCommit();
|
$db->transCommit();
|
||||||
|
$this->attribute->deleteOrphanedValues();
|
||||||
|
|
||||||
return $this->response->setJSON(['success' => true, 'message' => lang('Items.csv_import_success')]);
|
return $this->response->setJSON(['success' => true, 'message' => lang('Items.csv_import_success')]);
|
||||||
}
|
}
|
||||||
@@ -1079,6 +1106,20 @@ class Items extends Secure_Controller
|
|||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function extractAttributeData(array $row): array
|
||||||
|
{
|
||||||
|
$attributeData = [];
|
||||||
|
|
||||||
|
foreach ($row as $key => $value) {
|
||||||
|
if (str_starts_with($key, 'attribute_')) {
|
||||||
|
$definitionName = substr($key, 10);
|
||||||
|
$attributeData[$definitionName] = $value;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return $attributeData;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates that stock location columns in CSV row are valid locations
|
* Validates that stock location columns in CSV row are valid locations
|
||||||
*
|
*
|
||||||
@@ -1107,87 +1148,99 @@ class Items extends Secure_Controller
|
|||||||
* Checks the entire line of data in an import file for errors
|
* Checks the entire line of data in an import file for errors
|
||||||
*
|
*
|
||||||
* @param array $row
|
* @param array $row
|
||||||
* @param array $item_data
|
* @param array $itemData
|
||||||
* @param array $allowed_locations
|
* @param array $allowedStockLocations
|
||||||
* @param array $definition_names
|
* @param array $definitionNames
|
||||||
* @param array $attribute_data
|
* @param array $attributeData
|
||||||
* @return bool Returns false if all data checks out and true when there is an error in the data
|
* @return bool Returns false if all data checks out and true when there is an error in the data
|
||||||
*/
|
*/
|
||||||
private function data_error_check(array $row, array $item_data, array $allowed_locations, array $definition_names, array $attribute_data): bool // TODO: Long function and large number of parameters in the declaration... perhaps refactoring is needed
|
private function validateCSVData(array $row, array $itemData, array $allowedStockLocations, array $definitionNames, array $attributeData): bool // TODO: Long function and large number of parameters in the declaration... perhaps refactoring is needed
|
||||||
{
|
{
|
||||||
$item_id = $row['Id'];
|
$itemId = $row['Id'];
|
||||||
$is_update = (bool)$item_id;
|
$isUpdate = (bool)$itemId;
|
||||||
|
|
||||||
// Check for empty required fields
|
// Check for empty required fields
|
||||||
$check_for_empty = [
|
$valuesToCheckForEmpty = [
|
||||||
'name' => $item_data['name'],
|
'name' => $itemData['name'],
|
||||||
'category' => $item_data['category'],
|
'category' => $itemData['category'],
|
||||||
'unit_price' => $item_data['unit_price']
|
'unit_price' => $itemData['unit_price']
|
||||||
];
|
];
|
||||||
|
|
||||||
foreach ($check_for_empty as $key => $val) {
|
foreach ($valuesToCheckForEmpty as $key => $value) {
|
||||||
if (empty($val) && !$is_update) {
|
if (($value === null || $value === '') && !$isUpdate) {
|
||||||
log_message('error', "Empty required value in $key.");
|
log_message('error', "Empty required value in $key.");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$is_update) {
|
if (!$isUpdate) {
|
||||||
$item_data['cost_price'] = empty($item_data['cost_price']) ? 0 : $item_data['cost_price']; // Allow for zero wholesale price
|
$itemData['cost_price'] = empty($itemData['cost_price']) ? 0 : $itemData['cost_price']; // Allow for zero wholesale price
|
||||||
} else {
|
} else {
|
||||||
if (!$this->item->exists($item_id)) {
|
if (!$this->item->exists($itemId)) {
|
||||||
log_message('error', "non-existent item_id: '$item_id' when either existing item_id or no item_id is required.");
|
log_message('error', "non-existent item_id: '$itemId' when either existing item_id or no item_id is required.");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Build array of fields to check for numerics
|
// Build array of fields to check for numerics
|
||||||
$check_for_numeric_values = [
|
$valuesToCheckForNumeric = [
|
||||||
'cost_price' => $item_data['cost_price'],
|
'cost_price' => $itemData['cost_price'],
|
||||||
'unit_price' => $item_data['unit_price'],
|
'unit_price' => $itemData['unit_price'],
|
||||||
'reorder_level' => $item_data['reorder_level'],
|
'reorder_level' => $itemData['reorder_level'],
|
||||||
'supplier_id' => $row['Supplier ID'],
|
'supplier_id' => $row['Supplier ID'],
|
||||||
'Tax 1 Percent' => $row['Tax 1 Percent'],
|
'Tax 1 Percent' => $row['Tax 1 Percent'],
|
||||||
'Tax 2 Percent' => $row['Tax 2 Percent']
|
'Tax 2 Percent' => $row['Tax 2 Percent']
|
||||||
];
|
];
|
||||||
|
|
||||||
foreach ($allowed_locations as $location_name) {
|
foreach ($allowedStockLocations as $location_name) {
|
||||||
$check_for_numeric_values[] = $row["location_$location_name"];
|
$valuesToCheckForNumeric[] = $row["location_$location_name"];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check for non-numeric values which require numeric
|
// Check for non-numeric values which require numeric
|
||||||
foreach ($check_for_numeric_values as $key => $value) {
|
foreach ($valuesToCheckForNumeric as $key => $value) {
|
||||||
if (!is_numeric($value) && !empty($value)) {
|
if (!is_numeric($value) && !empty($value)) {
|
||||||
log_message('error', "non-numeric: '$value' for '$key' when numeric is required");
|
log_message('error', "non-numeric: '$value' for '$key' when numeric is required");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Check stock locations
|
||||||
|
$invalidLocations = $this->validateCSVStockLocations($row, $allowedStockLocations);
|
||||||
|
if (!empty($invalidLocations)) {
|
||||||
|
log_message('error', 'CSV import: Invalid stock location(s) found: ' . implode(', ', $invalidLocations));
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
// Check Attribute Data
|
// Check Attribute Data
|
||||||
foreach ($definition_names as $definition_name) {
|
foreach ($definitionNames as $definitionName) {
|
||||||
if (!empty($row["attribute_$definition_name"])) {
|
$attributeColumn = "attribute_$definitionName";
|
||||||
$definition_type = $attribute_data[$definition_name]['definition_type'];
|
if (array_key_exists($attributeColumn, $row) && $row[$attributeColumn] != '') {
|
||||||
$attribute_value = $row["attribute_$definition_name"];
|
$definitionType = $attributeData[$definitionName]['definition_type'];
|
||||||
|
$attributeValue = $row[$attributeColumn];
|
||||||
|
|
||||||
switch ($definition_type) {
|
if (strcasecmp($attributeValue, '_DELETE_') === 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch ($definitionType) {
|
||||||
case DROPDOWN:
|
case DROPDOWN:
|
||||||
$dropdown_values = $attribute_data[$definition_name]['dropdown_values'];
|
$dropdownValues = $attributeData[$definitionName]['dropdown_values'];
|
||||||
$dropdown_values[] = '';
|
$dropdownValues[] = '';
|
||||||
|
|
||||||
if (!empty($attribute_value) && !in_array($attribute_value, $dropdown_values)) {
|
if (!empty($attributeValue) && !in_array($attributeValue, $dropdownValues)) {
|
||||||
log_message('error', "Value: '$attribute_value' is not an acceptable DROPDOWN value");
|
log_message('error', "Value: '$attributeValue' is not an acceptable DROPDOWN value");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case DECIMAL:
|
case DECIMAL:
|
||||||
if (!is_numeric($attribute_value) && !empty($attribute_value)) {
|
if (!is_numeric($attributeValue) && !empty($attributeValue)) {
|
||||||
log_message('error', "'$attribute_value' is not an acceptable DECIMAL value");
|
log_message('error', "'$attributeValue' is not an acceptable DECIMAL value");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
case DATE:
|
case DATE:
|
||||||
if (!valid_date($attribute_value) && !empty($attribute_value)) {
|
if (!valid_date($attributeValue) && !empty($attributeValue)) {
|
||||||
log_message('error', "'$attribute_value' is not an acceptable DATE value. The value must match the set locale.");
|
log_message('error', "'$attributeValue' is not an acceptable DATE value. The value must match the set locale.");
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -1198,59 +1251,6 @@ class Items extends Secure_Controller
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Saves attribute data found in the CSV import.
|
|
||||||
*
|
|
||||||
* @param array $row
|
|
||||||
* @param array $item_data
|
|
||||||
* @param array $definitions
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
private function save_attribute_data(array $row, array $item_data, array $definitions): bool
|
|
||||||
{
|
|
||||||
foreach ($definitions as $definition) {
|
|
||||||
$attribute_name = $definition['definition_name'];
|
|
||||||
$attribute_value = $row["attribute_$attribute_name"];
|
|
||||||
|
|
||||||
// Create attribute value
|
|
||||||
if (!empty($attribute_value) || $attribute_value === '0') {
|
|
||||||
if ($definition['definition_type'] === CHECKBOX) {
|
|
||||||
$checkbox_is_unchecked = (strcasecmp($attribute_value, 'false') === 0 || $attribute_value === '0');
|
|
||||||
$attribute_value = $checkbox_is_unchecked ? '0' : '1';
|
|
||||||
|
|
||||||
$attribute_id = $this->store_attribute_value($attribute_value, $definition, $item_data['item_id']);
|
|
||||||
} elseif (!empty($attribute_value)) {
|
|
||||||
$attribute_id = $this->store_attribute_value($attribute_value, $definition, $item_data['item_id']);
|
|
||||||
} else {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!$attribute_id) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Saves the attribute_value and attribute_link if necessary
|
|
||||||
*/
|
|
||||||
private function store_attribute_value(string $value, array $attribute_data, int $item_id)
|
|
||||||
{
|
|
||||||
$attribute_id = $this->attribute->attributeValueExists($value, $attribute_data['definition_type']);
|
|
||||||
|
|
||||||
$this->attribute->deleteAttributeLinks($item_id, $attribute_data['definition_id']);
|
|
||||||
|
|
||||||
if (!$attribute_id) {
|
|
||||||
$attribute_id = $this->attribute->saveAttributeValue($value, $attribute_data['definition_id'], $item_id, false, $attribute_data['definition_type']);
|
|
||||||
} elseif (!$this->attribute->saveAttributeLink($item_id, $attribute_data['definition_id'], $attribute_id)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return $attribute_id;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Saves inventory quantities for the row in the appropriate stock locations.
|
* Saves inventory quantities for the row in the appropriate stock locations.
|
||||||
*
|
*
|
||||||
@@ -1258,13 +1258,15 @@ class Items extends Secure_Controller
|
|||||||
* @param array $item_data
|
* @param array $item_data
|
||||||
* @param array $allowed_locations
|
* @param array $allowed_locations
|
||||||
* @param int $employee_id
|
* @param int $employee_id
|
||||||
|
* @return bool Returns true on success, false on failure
|
||||||
* @throws ReflectionException
|
* @throws ReflectionException
|
||||||
*/
|
*/
|
||||||
private function save_inventory_quantities(array $row, array $item_data, array $allowed_locations, int $employee_id): void
|
private function save_inventory_quantities(array $row, array $item_data, array $allowed_locations, int $employee_id): bool
|
||||||
{
|
{
|
||||||
// Quantities & Inventory Section
|
// Quantities & Inventory Section
|
||||||
$comment = lang('Items.inventory_CSV_import_quantity');
|
$comment = lang('Items.inventory_CSV_import_quantity');
|
||||||
$is_update = (bool)$row['Id'];
|
$is_update = (bool)$row['Id'];
|
||||||
|
$success = true;
|
||||||
|
|
||||||
foreach ($allowed_locations as $location_id => $location_name) {
|
foreach ($allowed_locations as $location_id => $location_name) {
|
||||||
$item_quantity_data = ['item_id' => $item_data['item_id'], 'location_id' => $location_id];
|
$item_quantity_data = ['item_id' => $item_data['item_id'], 'location_id' => $location_id];
|
||||||
@@ -1278,20 +1280,22 @@ class Items extends Secure_Controller
|
|||||||
|
|
||||||
if (!empty($row["location_$location_name"]) || $row["location_$location_name"] === '0') {
|
if (!empty($row["location_$location_name"]) || $row["location_$location_name"] === '0') {
|
||||||
$item_quantity_data['quantity'] = $row["location_$location_name"];
|
$item_quantity_data['quantity'] = $row["location_$location_name"];
|
||||||
$this->item_quantity->save_value($item_quantity_data, $item_data['item_id'], $location_id);
|
$success &= $this->item_quantity->save_value($item_quantity_data, $item_data['item_id'], $location_id);
|
||||||
|
|
||||||
$csv_data['trans_inventory'] = $row["location_$location_name"];
|
$csv_data['trans_inventory'] = $row["location_$location_name"];
|
||||||
$this->inventory->insert($csv_data, false);
|
$success &= (bool)$this->inventory->insert($csv_data, false);
|
||||||
} elseif ($is_update) {
|
} elseif ($is_update) {
|
||||||
return;
|
continue;
|
||||||
} else {
|
} else {
|
||||||
$item_quantity_data['quantity'] = 0;
|
$item_quantity_data['quantity'] = 0;
|
||||||
$this->item_quantity->save_value($item_quantity_data, $item_data['item_id'], $location_id);
|
$success &= $this->item_quantity->save_value($item_quantity_data, $item_data['item_id'], $location_id);
|
||||||
|
|
||||||
$csv_data['trans_inventory'] = 0;
|
$csv_data['trans_inventory'] = 0;
|
||||||
$this->inventory->insert($csv_data, false);
|
$success &= (bool)$this->inventory->insert($csv_data, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return (bool)$success;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1299,8 +1303,9 @@ class Items extends Secure_Controller
|
|||||||
*
|
*
|
||||||
* @param array $row
|
* @param array $row
|
||||||
* @param array $item_data
|
* @param array $item_data
|
||||||
|
* @return bool Returns true on success, false on failure
|
||||||
*/
|
*/
|
||||||
private function save_tax_data(array $row, array $item_data): void
|
private function save_tax_data(array $row, array $item_data): bool
|
||||||
{
|
{
|
||||||
$items_taxes_data = [];
|
$items_taxes_data = [];
|
||||||
|
|
||||||
@@ -1312,9 +1317,11 @@ class Items extends Secure_Controller
|
|||||||
$items_taxes_data[] = ['name' => $row['Tax 2 Name'], 'percent' => $row['Tax 2 Percent']];
|
$items_taxes_data[] = ['name' => $row['Tax 2 Name'], 'percent' => $row['Tax 2 Percent']];
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isset($items_taxes_data)) {
|
if (!empty($items_taxes_data)) {
|
||||||
$this->item_taxes->save_value($items_taxes_data, $item_data['item_id']);
|
return $this->item_taxes->save_value($items_taxes_data, $item_data['item_id']);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1344,10 +1351,11 @@ class Items extends Secure_Controller
|
|||||||
* Saves item attributes for a given item.
|
* Saves item attributes for a given item.
|
||||||
*
|
*
|
||||||
* @param int $itemId The item for which attributes need to be saved to.
|
* @param int $itemId The item for which attributes need to be saved to.
|
||||||
* @return void
|
* @return bool Returns true when item attributes are successfully saved and false on error.
|
||||||
*/
|
*/
|
||||||
public function saveItemAttributes(int $itemId): void
|
public function saveItemAttributes(int $itemId): bool
|
||||||
{
|
{
|
||||||
|
$success = true;
|
||||||
$attributeLinks = $this->request->getPost('attribute_links') ?? [];
|
$attributeLinks = $this->request->getPost('attribute_links') ?? [];
|
||||||
$attributeIds = $this->request->getPost('attribute_ids');
|
$attributeIds = $this->request->getPost('attribute_ids');
|
||||||
|
|
||||||
@@ -1359,16 +1367,18 @@ class Items extends Secure_Controller
|
|||||||
switch ($definitionType) {
|
switch ($definitionType) {
|
||||||
case DROPDOWN:
|
case DROPDOWN:
|
||||||
$attributeId = $attributeValue;
|
$attributeId = $attributeValue;
|
||||||
|
$success = $success && $this->attribute->saveAttributeLink($itemId, $definitionId, $attributeId);
|
||||||
break;
|
break;
|
||||||
case DECIMAL:
|
case DECIMAL:
|
||||||
$attributeValue = parse_decimals($attributeValue);
|
$attributeValue = parse_decimals($attributeValue);
|
||||||
// Fall through to save the attribute value
|
// no break
|
||||||
default:
|
default:
|
||||||
$attributeId = $this->attribute->saveAttributeValue($attributeValue, $definitionId, $itemId, $attributeIds[$definitionId], $definitionType);
|
$attributeId = $this->attribute->saveAttributeValue($attributeValue, $definitionId, $itemId, $attributeIds[$definitionId], $definitionType);
|
||||||
|
$success = $success && ($attributeId > 0);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->attribute->saveAttributeLink($itemId, $definitionId, $attributeId);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return $success && $this->attribute->deleteOrphanedValues();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ namespace App\Controllers;
|
|||||||
use App\Libraries\MY_Migration;
|
use App\Libraries\MY_Migration;
|
||||||
use App\Models\Employee;
|
use App\Models\Employee;
|
||||||
use CodeIgniter\HTTP\RedirectResponse;
|
use CodeIgniter\HTTP\RedirectResponse;
|
||||||
|
use CodeIgniter\HTTP\ResponseInterface;
|
||||||
use CodeIgniter\Model;
|
use CodeIgniter\Model;
|
||||||
use Config\OSPOS;
|
use Config\OSPOS;
|
||||||
use Config\Services;
|
use Config\Services;
|
||||||
@@ -36,6 +37,7 @@ class Login extends BaseController
|
|||||||
|
|
||||||
$data = [
|
$data = [
|
||||||
'has_errors' => false,
|
'has_errors' => false,
|
||||||
|
'is_new_install' => !(MY_Migration::get_current_version()),
|
||||||
'is_latest' => $migration->is_latest(),
|
'is_latest' => $migration->is_latest(),
|
||||||
'latest_version' => $migration->get_latest_migration(),
|
'latest_version' => $migration->get_latest_migration(),
|
||||||
'gcaptcha_enabled' => $gcaptcha_enabled,
|
'gcaptcha_enabled' => $gcaptcha_enabled,
|
||||||
@@ -47,6 +49,13 @@ class Login extends BaseController
|
|||||||
return view('login', $data);
|
return view('login', $data);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!$data['is_latest'] || $data['is_new_install']) {
|
||||||
|
set_time_limit(3600);
|
||||||
|
|
||||||
|
$migration->setNamespace('App')->latest();
|
||||||
|
return redirect()->to('login');
|
||||||
|
}
|
||||||
|
|
||||||
$rules = ['username' => 'required|login_check[data]'];
|
$rules = ['username' => 'required|login_check[data]'];
|
||||||
$messages = [
|
$messages = [
|
||||||
'username' => [
|
'username' => [
|
||||||
@@ -60,15 +69,32 @@ class Login extends BaseController
|
|||||||
|
|
||||||
return view('login', $data);
|
return view('login', $data);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$data['is_latest']) {
|
|
||||||
set_time_limit(3600);
|
|
||||||
|
|
||||||
$migration->setNamespace('App')->latest();
|
|
||||||
return redirect()->to('login');
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->to('home');
|
return redirect()->to('home');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function migrate(): ResponseInterface
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$migration = new MY_Migration(config('Migrations'));
|
||||||
|
$migration->migrate_to_ci4();
|
||||||
|
|
||||||
|
set_time_limit(3600);
|
||||||
|
$migration->setNamespace('App')->latest();
|
||||||
|
|
||||||
|
return $this->response->setJSON([
|
||||||
|
'success' => true,
|
||||||
|
'message' => 'Migration completed successfully'
|
||||||
|
]);
|
||||||
|
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
log_message('error', 'Migration failed: ' . $e->getMessage());
|
||||||
|
|
||||||
|
return $this->response->setJSON([
|
||||||
|
'success' => false,
|
||||||
|
'message' => 'Migration failed: ' . $e->getMessage()
|
||||||
|
])->setStatusCode(500);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -190,11 +190,11 @@ class Receivings extends Secure_Controller
|
|||||||
/**
|
/**
|
||||||
* Edit line item in current receiving. Used in app/Views/receivings/receiving.php
|
* Edit line item in current receiving. Used in app/Views/receivings/receiving.php
|
||||||
*
|
*
|
||||||
* @param string|int|null $item_id
|
* @param int|string|null $item_id
|
||||||
* @return string
|
* @return string
|
||||||
* @noinspection PhpUnused
|
* @noinspection PhpUnused
|
||||||
*/
|
*/
|
||||||
public function postEditItem($item_id): string
|
public function postEditItem(int|string|null $item_id): string
|
||||||
{
|
{
|
||||||
$data = [];
|
$data = [];
|
||||||
|
|
||||||
@@ -242,7 +242,7 @@ class Receivings extends Secure_Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
$receiving_info = $this->receiving->get_info($receiving_id)->getRowArray();
|
$receiving_info = $this->receiving->get_info($receiving_id)->getRowArray();
|
||||||
|
|
||||||
$current_employee_id = $this->employee->get_logged_in_employee_info()->person_id;
|
$current_employee_id = $this->employee->get_logged_in_employee_info()->person_id;
|
||||||
$can_assign_employee = $this->employee->has_grant('employees', $current_employee_id);
|
$can_assign_employee = $this->employee->has_grant('employees', $current_employee_id);
|
||||||
|
|
||||||
@@ -280,8 +280,10 @@ class Receivings extends Secure_Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @throws ReflectionException
|
* @param int $receiving_id
|
||||||
|
* @param bool $update_inventory
|
||||||
* @return ResponseInterface
|
* @return ResponseInterface
|
||||||
|
* @throws ReflectionException
|
||||||
*/
|
*/
|
||||||
public function postDelete(int $receiving_id = -1, bool $update_inventory = true): ResponseInterface
|
public function postDelete(int $receiving_id = -1, bool $update_inventory = true): ResponseInterface
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -1246,13 +1246,15 @@ class Reports extends Secure_Controller
|
|||||||
public function get_payment_type(): array
|
public function get_payment_type(): array
|
||||||
{
|
{
|
||||||
return [
|
return [
|
||||||
'all' => lang('Common.none_selected_text'),
|
'all' => lang('Common.none_selected_text'),
|
||||||
'cash' => lang('Sales.cash'),
|
'cash' => lang('Sales.cash'),
|
||||||
'due' => lang('Sales.due'),
|
'due' => lang('Sales.due'),
|
||||||
'check' => lang('Sales.check'),
|
'check' => lang('Sales.check'),
|
||||||
'credit' => lang('Sales.credit'),
|
'credit' => lang('Sales.credit'),
|
||||||
'debit' => lang('Sales.debit'),
|
'debit' => lang('Sales.debit'),
|
||||||
'invoices' => lang('Sales.invoice')
|
'bank_transfer' => lang('Sales.bank_transfer'),
|
||||||
|
'wallet' => lang('Sales.wallet'),
|
||||||
|
'invoices' => lang('Sales.invoice')
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -66,12 +66,168 @@ class Sales extends Secure_Controller
|
|||||||
$this->employee = model(Employee::class);
|
$this->employee = model(Employee::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Adds the shared secondary currency context to a view data array.
|
||||||
|
*
|
||||||
|
* @param array $data
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
private function _append_secondary_currency(array &$data): void
|
||||||
|
{
|
||||||
|
$secondaryCurrency = secondary_currency_context($this->config);
|
||||||
|
$data['secondaryCurrency'] = $secondaryCurrency;
|
||||||
|
|
||||||
|
if (!$secondaryCurrency['show']) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$displayFields = [
|
||||||
|
'total' => 'secondaryTotalDisplay',
|
||||||
|
'amount_due' => 'secondaryAmountDueDisplay',
|
||||||
|
'cash_amount_due' => 'secondaryCashAmountDueDisplay',
|
||||||
|
'non_cash_total' => 'secondaryNonCashTotalDisplay',
|
||||||
|
'non_cash_amount_due' => 'secondaryNonCashAmountDueDisplay'
|
||||||
|
];
|
||||||
|
|
||||||
|
foreach ($displayFields as $sourceField => $targetField) {
|
||||||
|
if (array_key_exists($sourceField, $data)) {
|
||||||
|
$data[$targetField] = to_secondary_currency((float) $data[$sourceField], $secondaryCurrency);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
public function getIndex(): ResponseInterface|string
|
public function getIndex(): ResponseInterface|string
|
||||||
{
|
{
|
||||||
$this->session->set('allow_temp_items', 1);
|
$this->session->set('allow_temp_items', 1);
|
||||||
return $this->_reload(); // TODO: Hungarian Notation
|
return $this->_reload(); // TODO: Hungarian Notation
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Load the customer display popup.
|
||||||
|
*
|
||||||
|
* @return ResponseInterface|string
|
||||||
|
* @noinspection PhpUnused
|
||||||
|
*/
|
||||||
|
public function getCustomerDisplay(): ResponseInterface|string
|
||||||
|
{
|
||||||
|
if (($this->config['customer_display_enabled'] ?? false) != 1) {
|
||||||
|
return $this->response->setStatusCode(404)->setBody('');
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->session->get('sale_id') == '') {
|
||||||
|
$this->session->set('sale_id', NEW_ENTRY);
|
||||||
|
}
|
||||||
|
|
||||||
|
$secondaryCurrency = secondary_currency_context($this->config);
|
||||||
|
$secondaryCurrencyEnabled = (($this->config['secondary_currency_enabled'] ?? false) == 1);
|
||||||
|
$cashRounding = $this->sale_lib->reset_cash_rounding();
|
||||||
|
$showCustomerDisplay = $secondaryCurrencyEnabled && !empty($secondaryCurrency['rate']) && (float) $secondaryCurrency['rate'] > 0;
|
||||||
|
$companyLines = preg_split("/\r\n|\r|\n/", (string) ($this->config['company'] ?? '')) ?: [];
|
||||||
|
$companyName = array_shift($companyLines) ?? '';
|
||||||
|
$companyDetails = trim(implode("\n", $companyLines));
|
||||||
|
$secondaryCurrencySymbol = trim((string) ($this->config['secondary_currency_symbol'] ?? ''));
|
||||||
|
$secondaryCurrencyCode = trim((string) ($this->config['secondary_currency_code'] ?? ''));
|
||||||
|
$originalCurrencySymbol = trim((string) ($this->config['currency_symbol'] ?? ''));
|
||||||
|
$customerDisplayCurrencyLabel = $secondaryCurrencyCode !== '' ? $secondaryCurrencyCode : ($secondaryCurrencySymbol !== '' ? $secondaryCurrencySymbol : 'LBP');
|
||||||
|
$originalCurrencyLabel = $originalCurrencySymbol !== '' ? $originalCurrencySymbol : '$';
|
||||||
|
$cartHasCustomerDisplay = $showCustomerDisplay;
|
||||||
|
$cartColspan = $cartHasCustomerDisplay ? 6 : 5;
|
||||||
|
$cartItemWidth = $cartHasCustomerDisplay ? 32 : 44;
|
||||||
|
$cartPriceWidth = $cartHasCustomerDisplay ? 18 : 0;
|
||||||
|
$cartOriginalWidth = $cartHasCustomerDisplay ? 18 : 26;
|
||||||
|
$cartQuantityWidth = $cartHasCustomerDisplay ? 12 : 10;
|
||||||
|
$cartDiscountWidth = $cartHasCustomerDisplay ? 10 : 9;
|
||||||
|
$cartTotalWidth = $cartHasCustomerDisplay ? 10 : 11;
|
||||||
|
|
||||||
|
$data = [
|
||||||
|
'cash_rounding' => $cashRounding,
|
||||||
|
'cart' => $this->sale_lib->get_cart()
|
||||||
|
];
|
||||||
|
$customer_info = $this->_load_customer_data($this->sale_lib->get_customer(), $data, true);
|
||||||
|
$data += [
|
||||||
|
'customer_name' => $data['customer'] ?? lang('Sales.walk_in_customer'),
|
||||||
|
'customer_reward_points' => (int) ($data['customer_rewards']['points'] ?? 0),
|
||||||
|
'customer_reward_package' => $data['customer_rewards']['package_name'] ?? '',
|
||||||
|
'giftcard_remainder' => $this->sale_lib->get_giftcard_remainder(),
|
||||||
|
'rewards_remainder' => $this->sale_lib->get_rewards_remainder(),
|
||||||
|
'customerName' => $data['customer'] ?? lang('Sales.walk_in_customer'),
|
||||||
|
'customerRewardPoints' => (int) ($data['customer_rewards']['points'] ?? 0),
|
||||||
|
'giftcardRemainder' => $this->sale_lib->get_giftcard_remainder()
|
||||||
|
];
|
||||||
|
|
||||||
|
$tax_details = $this->tax_lib->get_taxes($data['cart']);
|
||||||
|
$data += [
|
||||||
|
'tax_exclusive_subtotal' => $this->sale_lib->get_subtotal(true, true),
|
||||||
|
'taxes' => $tax_details[0],
|
||||||
|
'discount' => $this->sale_lib->get_discount(),
|
||||||
|
'payments' => $this->sale_lib->get_payments()
|
||||||
|
];
|
||||||
|
|
||||||
|
$totals = $this->sale_lib->get_totals($tax_details[0]);
|
||||||
|
$data += [
|
||||||
|
'item_count' => $totals['item_count'],
|
||||||
|
'total_units' => $totals['total_units'],
|
||||||
|
'subtotal' => $totals['subtotal'],
|
||||||
|
'total' => $totals['total'],
|
||||||
|
'payments_total' => $totals['payment_total'],
|
||||||
|
'payments_cover_total' => $totals['payments_cover_total'],
|
||||||
|
'prediscount_subtotal' => $totals['prediscount_subtotal'],
|
||||||
|
'cash_total' => $totals['cash_total'],
|
||||||
|
'non_cash_total' => $totals['total'],
|
||||||
|
'cash_amount_due' => $totals['cash_amount_due'],
|
||||||
|
'non_cash_amount_due' => $totals['amount_due'],
|
||||||
|
'cash_mode' => $this->session->get('cash_mode'),
|
||||||
|
'selected_payment_type' => $this->sale_lib->get_payment_type(),
|
||||||
|
'comment' => $this->sale_lib->get_comment(),
|
||||||
|
'email_receipt' => $this->sale_lib->is_email_receipt(),
|
||||||
|
'config' => $this->config,
|
||||||
|
'mode' => $this->sale_lib->get_mode(),
|
||||||
|
'rate' => (float) ($secondaryCurrency['rate'] ?? $this->config['secondary_currency_rate'] ?? 0),
|
||||||
|
'secondaryCurrency' => $secondaryCurrency,
|
||||||
|
'secondaryCurrencyEnabled' => $secondaryCurrencyEnabled,
|
||||||
|
'showCustomerDisplay' => $showCustomerDisplay,
|
||||||
|
'companyName' => $companyName,
|
||||||
|
'companyDetails' => $companyDetails,
|
||||||
|
'secondaryCurrencySymbol' => $secondaryCurrencySymbol,
|
||||||
|
'secondaryCurrencyCode' => $secondaryCurrencyCode,
|
||||||
|
'originalCurrencySymbol' => $originalCurrencySymbol,
|
||||||
|
'customerDisplayCurrencyLabel' => $customerDisplayCurrencyLabel,
|
||||||
|
'originalCurrencyLabel' => $originalCurrencyLabel,
|
||||||
|
'cartHasCustomerDisplay' => $cartHasCustomerDisplay,
|
||||||
|
'cartColspan' => $cartColspan,
|
||||||
|
'cartItemWidth' => $cartItemWidth,
|
||||||
|
'cartPriceWidth' => $cartPriceWidth,
|
||||||
|
'cartOriginalWidth' => $cartOriginalWidth,
|
||||||
|
'cartQuantityWidth' => $cartQuantityWidth,
|
||||||
|
'cartDiscountWidth' => $cartDiscountWidth,
|
||||||
|
'cartTotalWidth' => $cartTotalWidth,
|
||||||
|
'items_module_allowed' => $this->employee->has_grant('items', $this->employee->get_logged_in_employee_info()->person_id),
|
||||||
|
'change_price' => $this->employee->has_grant('sales_change_price', $this->employee->get_logged_in_employee_info()->person_id)
|
||||||
|
];
|
||||||
|
|
||||||
|
$invoice_number = $this->sale_lib->get_invoice_number();
|
||||||
|
if ($invoice_number == null || $invoice_number == '') {
|
||||||
|
$invoice_number = $this->token_lib->render($this->config['sales_invoice_format'], [], false);
|
||||||
|
}
|
||||||
|
|
||||||
|
$data += [
|
||||||
|
'invoice_number' => $invoice_number,
|
||||||
|
'print_after_sale' => $this->sale_lib->is_print_after_sale(),
|
||||||
|
'price_work_orders' => $this->sale_lib->is_price_work_orders(),
|
||||||
|
'pos_mode' => $data['mode'] == 'sale' || $data['mode'] == 'return',
|
||||||
|
'quote_number' => $this->sale_lib->get_quote_number(),
|
||||||
|
'work_order_number' => $this->sale_lib->get_work_order_number(),
|
||||||
|
'amount_due' => $data['cash_mode'] && ($data['selected_payment_type'] === lang('Sales.cash') || $data['payments_total'] > 0) ? $totals['cash_amount_due'] : $totals['amount_due']
|
||||||
|
];
|
||||||
|
$data['amount_change'] = $data['amount_due'] * -1;
|
||||||
|
$data['payment_change_due'] = ((float) $data['amount_due'] < 0)
|
||||||
|
? abs((float) $data['amount_due'])
|
||||||
|
: max(((float) $data['payments_total']) - ((float) $data['amount_due']), 0);
|
||||||
|
$data['paymentChangeDue'] = $data['payment_change_due'];
|
||||||
|
|
||||||
|
return view('sales/customer_display', $data);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Load the sale edit modal. Used in app/Views/sales/register.php.
|
* Load the sale edit modal. Used in app/Views/sales/register.php.
|
||||||
*
|
*
|
||||||
@@ -425,7 +581,7 @@ class Sales extends Secure_Controller
|
|||||||
$new_giftcard_value = $giftcard->get_giftcard_value($giftcard_num) - $this->sale_lib->get_amount_due();
|
$new_giftcard_value = $giftcard->get_giftcard_value($giftcard_num) - $this->sale_lib->get_amount_due();
|
||||||
$new_giftcard_value = max($new_giftcard_value, 0);
|
$new_giftcard_value = max($new_giftcard_value, 0);
|
||||||
$this->sale_lib->set_giftcard_remainder($new_giftcard_value);
|
$this->sale_lib->set_giftcard_remainder($new_giftcard_value);
|
||||||
$new_giftcard_value = str_replace('$', '\$', to_currency($new_giftcard_value));
|
$new_giftcard_value = to_currency($new_giftcard_value);
|
||||||
$data['warning'] = lang('Giftcards.remaining_balance', [$giftcard_num, $new_giftcard_value]);
|
$data['warning'] = lang('Giftcards.remaining_balance', [$giftcard_num, $new_giftcard_value]);
|
||||||
$amount_tendered = min($this->sale_lib->get_amount_due(), $giftcard->get_giftcard_value($giftcard_num));
|
$amount_tendered = min($this->sale_lib->get_amount_due(), $giftcard->get_giftcard_value($giftcard_num));
|
||||||
|
|
||||||
@@ -582,12 +738,21 @@ class Sales extends Secure_Controller
|
|||||||
$data = [];
|
$data = [];
|
||||||
|
|
||||||
$rules = [
|
$rules = [
|
||||||
'price' => 'trim|required|decimal_locale',
|
'price' => 'trim|required|decimal_locale|nonNegativeDecimal',
|
||||||
'quantity' => 'trim|required|decimal_locale',
|
'quantity' => 'trim|required|decimal_locale',
|
||||||
'discount' => 'trim|permit_empty|decimal_locale',
|
'discount' => 'trim|permit_empty|decimal_locale|nonNegativeDecimal',
|
||||||
];
|
];
|
||||||
|
|
||||||
if ($this->validate($rules)) {
|
$messages = [
|
||||||
|
'price' => [
|
||||||
|
'nonNegativeDecimal' => lang('Sales.negative_price_invalid'),
|
||||||
|
],
|
||||||
|
'discount' => [
|
||||||
|
'nonNegativeDecimal' => lang('Sales.negative_discount_invalid'),
|
||||||
|
],
|
||||||
|
];
|
||||||
|
|
||||||
|
if ($this->validate($rules, $messages)) {
|
||||||
$description = $this->request->getPost('description', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$description = $this->request->getPost('description', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
$serialnumber = $this->request->getPost('serialnumber', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$serialnumber = $this->request->getPost('serialnumber', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
$price = parse_decimals($this->request->getPost('price'));
|
$price = parse_decimals($this->request->getPost('price'));
|
||||||
@@ -596,20 +761,38 @@ class Sales extends Secure_Controller
|
|||||||
$discount = $discount_type
|
$discount = $discount_type
|
||||||
? parse_quantity($this->request->getPost('discount'))
|
? parse_quantity($this->request->getPost('discount'))
|
||||||
: parse_decimals($this->request->getPost('discount'));
|
: parse_decimals($this->request->getPost('discount'));
|
||||||
|
$discount = $discount ?: 0;
|
||||||
|
|
||||||
|
// Return mode legitimately uses negative quantities for refunds
|
||||||
|
if ($this->sale_lib->get_mode() != 'return' && $quantity < 0) {
|
||||||
|
$data['error'] = lang('Sales.negative_quantity_invalid');
|
||||||
|
return $this->_reload($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Business logic: discount bounds depend on discount_type and item values
|
||||||
|
if ($discount_type == PERCENT && $discount > 100) {
|
||||||
|
$data['error'] = lang('Sales.discount_percent_exceeds_100');
|
||||||
|
return $this->_reload($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($discount_type == FIXED && bccomp((string)$discount, bcmul((string)abs($quantity), (string)$price, 2), 2) > 0) {
|
||||||
|
$data['error'] = lang('Sales.discount_exceeds_item_total');
|
||||||
|
return $this->_reload($data);
|
||||||
|
}
|
||||||
|
|
||||||
$item_location = $this->request->getPost('location', FILTER_SANITIZE_NUMBER_INT);
|
$item_location = $this->request->getPost('location', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$discounted_total = $this->request->getPost('discounted_total') != ''
|
$discounted_total = $this->request->getPost('discounted_total') != ''
|
||||||
? parse_decimals($this->request->getPost('discounted_total') ?? '')
|
? parse_decimals($this->request->getPost('discounted_total') ?? '')
|
||||||
: null;
|
: null;
|
||||||
|
|
||||||
|
|
||||||
$this->sale_lib->edit_item($line, $description, $serialnumber, $quantity, $discount, $discount_type, $price, $discounted_total);
|
$this->sale_lib->edit_item($line, $description, $serialnumber, $quantity, $discount, $discount_type, $price, $discounted_total);
|
||||||
|
|
||||||
$this->sale_lib->empty_payments();
|
$this->sale_lib->empty_payments();
|
||||||
|
|
||||||
$data['warning'] = $this->sale_lib->out_of_stock($this->sale_lib->get_item_id($line), $item_location);
|
$data['warning'] = $this->sale_lib->out_of_stock($this->sale_lib->get_item_id($line), $item_location);
|
||||||
} else {
|
} else {
|
||||||
$data['error'] = lang('Sales.error_editing_item');
|
$errors = $this->validator->getErrors();
|
||||||
|
$data['error'] = $errors ? reset($errors) : lang('Sales.error_editing_item');
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->_reload($data);
|
return $this->_reload($data);
|
||||||
@@ -723,6 +906,12 @@ class Sales extends Secure_Controller
|
|||||||
$data['cash_amount_due'] = $totals['cash_amount_due'];
|
$data['cash_amount_due'] = $totals['cash_amount_due'];
|
||||||
$data['non_cash_amount_due'] = $totals['amount_due'];
|
$data['non_cash_amount_due'] = $totals['amount_due'];
|
||||||
|
|
||||||
|
// Prevent negative total sales (fraud/theft vector) - returns can have negative totals for legitimate refunds
|
||||||
|
if ($this->sale_lib->get_mode() != 'return' && bccomp($totals['total'], '0') < 0) {
|
||||||
|
$data['error'] = lang('Sales.negative_total_invalid');
|
||||||
|
return $this->_reload($data);
|
||||||
|
}
|
||||||
|
|
||||||
if ($data['cash_mode']) { // TODO: Convert this to ternary notation
|
if ($data['cash_mode']) { // TODO: Convert this to ternary notation
|
||||||
$data['amount_due'] = $totals['cash_amount_due'];
|
$data['amount_due'] = $totals['cash_amount_due'];
|
||||||
} else {
|
} else {
|
||||||
@@ -763,7 +952,7 @@ class Sales extends Secure_Controller
|
|||||||
|
|
||||||
if ($sale_id == NEW_ENTRY && $this->sale->check_invoice_number_exists($invoice_number)) {
|
if ($sale_id == NEW_ENTRY && $this->sale->check_invoice_number_exists($invoice_number)) {
|
||||||
$data['error'] = lang('Sales.invoice_number_duplicate', [$invoice_number]);
|
$data['error'] = lang('Sales.invoice_number_duplicate', [$invoice_number]);
|
||||||
$this->_reload($data);
|
return $this->_reload($data);
|
||||||
} else {
|
} else {
|
||||||
$data['invoice_number'] = $invoice_number;
|
$data['invoice_number'] = $invoice_number;
|
||||||
$data['sale_status'] = COMPLETED;
|
$data['sale_status'] = COMPLETED;
|
||||||
@@ -781,13 +970,15 @@ class Sales extends Secure_Controller
|
|||||||
|
|
||||||
// Resort and filter cart lines for printing
|
// Resort and filter cart lines for printing
|
||||||
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
||||||
|
$this->_append_secondary_currency($data);
|
||||||
|
|
||||||
if ($data['sale_id_num'] == NEW_ENTRY) {
|
if ($data['sale_id_num'] == NEW_ENTRY) {
|
||||||
$data['error_message'] = lang('Sales.transaction_failed');
|
$data['error_message'] = lang('Sales.transaction_failed');
|
||||||
|
return $this->_reload($data);
|
||||||
} else {
|
} else {
|
||||||
$data['barcode'] = $this->barcode_lib->generate_receipt_barcode($data['sale_id']);
|
$data['barcode'] = $this->barcode_lib->generate_receipt_barcode($data['sale_id']);
|
||||||
return view('sales/' . $invoice_view, $data);
|
|
||||||
$this->sale_lib->clear_all();
|
$this->sale_lib->clear_all();
|
||||||
|
return view('sales/' . $invoice_view, $data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} elseif ($this->sale_lib->is_work_order_mode()) {
|
} elseif ($this->sale_lib->is_work_order_mode()) {
|
||||||
@@ -807,7 +998,7 @@ class Sales extends Secure_Controller
|
|||||||
|
|
||||||
if ($sale_id == NEW_ENTRY && $this->sale->check_work_order_number_exists($work_order_number)) {
|
if ($sale_id == NEW_ENTRY && $this->sale->check_work_order_number_exists($work_order_number)) {
|
||||||
$data['error'] = lang('Sales.work_order_number_duplicate');
|
$data['error'] = lang('Sales.work_order_number_duplicate');
|
||||||
$this->_reload($data);
|
return $this->_reload($data);
|
||||||
} else {
|
} else {
|
||||||
$data['work_order_number'] = $work_order_number;
|
$data['work_order_number'] = $work_order_number;
|
||||||
$data['sale_status'] = SUSPENDED;
|
$data['sale_status'] = SUSPENDED;
|
||||||
@@ -819,10 +1010,10 @@ class Sales extends Secure_Controller
|
|||||||
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
||||||
|
|
||||||
$data['barcode'] = null;
|
$data['barcode'] = null;
|
||||||
|
$this->_append_secondary_currency($data);
|
||||||
|
|
||||||
return view('sales/work_order', $data);
|
|
||||||
$this->sale_lib->clear_mode();
|
|
||||||
$this->sale_lib->clear_all();
|
$this->sale_lib->clear_all();
|
||||||
|
return view('sales/work_order', $data);
|
||||||
}
|
}
|
||||||
} elseif ($this->sale_lib->is_quote_mode()) {
|
} elseif ($this->sale_lib->is_quote_mode()) {
|
||||||
$data['sales_quote'] = lang('Sales.quote');
|
$data['sales_quote'] = lang('Sales.quote');
|
||||||
@@ -836,7 +1027,7 @@ class Sales extends Secure_Controller
|
|||||||
|
|
||||||
if ($sale_id == NEW_ENTRY && $this->sale->check_quote_number_exists($quote_number)) {
|
if ($sale_id == NEW_ENTRY && $this->sale->check_quote_number_exists($quote_number)) {
|
||||||
$data['error'] = lang('Sales.quote_number_duplicate');
|
$data['error'] = lang('Sales.quote_number_duplicate');
|
||||||
$this->_reload($data);
|
return $this->_reload($data);
|
||||||
} else {
|
} else {
|
||||||
$data['quote_number'] = $quote_number;
|
$data['quote_number'] = $quote_number;
|
||||||
$data['sale_status'] = SUSPENDED;
|
$data['sale_status'] = SUSPENDED;
|
||||||
@@ -847,10 +1038,10 @@ class Sales extends Secure_Controller
|
|||||||
|
|
||||||
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
||||||
$data['barcode'] = null;
|
$data['barcode'] = null;
|
||||||
|
$this->_append_secondary_currency($data);
|
||||||
|
|
||||||
return view('sales/quote', $data);
|
|
||||||
$this->sale_lib->clear_mode();
|
|
||||||
$this->sale_lib->clear_all();
|
$this->sale_lib->clear_all();
|
||||||
|
return view('sales/quote', $data);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Save the data to the sales table
|
// Save the data to the sales table
|
||||||
@@ -866,13 +1057,15 @@ class Sales extends Secure_Controller
|
|||||||
$data['sale_id'] = 'POS ' . $data['sale_id_num'];
|
$data['sale_id'] = 'POS ' . $data['sale_id_num'];
|
||||||
|
|
||||||
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
$data['cart'] = $this->sale_lib->sort_and_filter_cart($data['cart']);
|
||||||
|
$this->_append_secondary_currency($data);
|
||||||
|
|
||||||
if ($data['sale_id_num'] == NEW_ENTRY) {
|
if ($data['sale_id_num'] == NEW_ENTRY) {
|
||||||
$data['error_message'] = lang('Sales.transaction_failed');
|
$data['error_message'] = lang('Sales.transaction_failed');
|
||||||
|
return $this->_reload($data);
|
||||||
} else {
|
} else {
|
||||||
$data['barcode'] = $this->barcode_lib->generate_receipt_barcode($data['sale_id']);
|
$data['barcode'] = $this->barcode_lib->generate_receipt_barcode($data['sale_id']);
|
||||||
return view('sales/receipt', $data);
|
|
||||||
$this->sale_lib->clear_all();
|
$this->sale_lib->clear_all();
|
||||||
|
return view('sales/receipt', $data);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -904,7 +1097,10 @@ class Sales extends Secure_Controller
|
|||||||
new Token_customer((array)$sale_data)
|
new Token_customer((array)$sale_data)
|
||||||
];
|
];
|
||||||
$text = $this->token_lib->render($text, $tokens);
|
$text = $this->token_lib->render($text, $tokens);
|
||||||
$sale_data['mimetype'] = mime_content_type(FCPATH . 'uploads/' . $this->config['company_logo']);
|
$sale_data['mimetype'] = $this->email_lib->getLogoMimeType();
|
||||||
|
|
||||||
|
// Build img_tag for email views that need it (receipt_email.php)
|
||||||
|
$sale_data['img_tag'] = $this->email_lib->buildLogoImgTag();
|
||||||
|
|
||||||
// Generate email attachment: invoice in PDF format
|
// Generate email attachment: invoice in PDF format
|
||||||
$view = Services::renderer();
|
$view = Services::renderer();
|
||||||
@@ -941,13 +1137,7 @@ class Sales extends Secure_Controller
|
|||||||
|
|
||||||
if (!empty($sale_data['customer_email'])) {
|
if (!empty($sale_data['customer_email'])) {
|
||||||
$sale_data['barcode'] = $this->barcode_lib->generate_receipt_barcode($sale_data['sale_id']);
|
$sale_data['barcode'] = $this->barcode_lib->generate_receipt_barcode($sale_data['sale_id']);
|
||||||
$sale_data['img_tag'] = '';
|
$sale_data['img_tag'] = $this->email_lib->buildLogoImgTag();
|
||||||
|
|
||||||
$logo_path = FCPATH . 'uploads/' . $this->config['company_logo'];
|
|
||||||
if (!empty($this->config['company_logo']) && file_exists($logo_path)) {
|
|
||||||
$logo_data = base64_encode(file_get_contents($logo_path));
|
|
||||||
$sale_data['img_tag'] = '<img id="image" src="data:image/png;base64,' . $logo_data . '" alt="company_logo">';
|
|
||||||
}
|
|
||||||
|
|
||||||
$to = $sale_data['customer_email'];
|
$to = $sale_data['customer_email'];
|
||||||
$subject = lang('Sales.receipt');
|
$subject = lang('Sales.receipt');
|
||||||
@@ -1128,6 +1318,7 @@ class Sales extends Secure_Controller
|
|||||||
$invoice_type = 'invoice';
|
$invoice_type = 'invoice';
|
||||||
}
|
}
|
||||||
$data['invoice_view'] = $invoice_type;
|
$data['invoice_view'] = $invoice_type;
|
||||||
|
$this->_append_secondary_currency($data);
|
||||||
|
|
||||||
return $data;
|
return $data;
|
||||||
}
|
}
|
||||||
@@ -1194,6 +1385,7 @@ class Sales extends Secure_Controller
|
|||||||
}
|
}
|
||||||
|
|
||||||
$data['amount_change'] = $data['amount_due'] * -1;
|
$data['amount_change'] = $data['amount_due'] * -1;
|
||||||
|
$this->_append_secondary_currency($data);
|
||||||
|
|
||||||
$data['comment'] = $this->sale_lib->get_comment();
|
$data['comment'] = $this->sale_lib->get_comment();
|
||||||
$data['email_receipt'] = $this->sale_lib->is_email_receipt();
|
$data['email_receipt'] = $this->sale_lib->is_email_receipt();
|
||||||
@@ -1662,10 +1854,11 @@ class Sales extends Secure_Controller
|
|||||||
$this->item->update_item_number($item_id, $item_number);
|
$this->item->update_item_number($item_id, $item_number);
|
||||||
$cart = $this->sale_lib->get_cart();
|
$cart = $this->sale_lib->get_cart();
|
||||||
$x = $this->search_cart_for_item_id($item_id, $cart);
|
$x = $this->search_cart_for_item_id($item_id, $cart);
|
||||||
if ($x != null) {
|
if ($x !== null) {
|
||||||
$cart[$x]['item_number'] = $item_number;
|
$cart[$x]['item_number'] = $item_number;
|
||||||
}
|
}
|
||||||
$this->sale_lib->set_cart($cart);
|
$this->sale_lib->set_cart($cart);
|
||||||
|
return $this->response->setJSON(['success' => true]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1684,11 +1877,12 @@ class Sales extends Secure_Controller
|
|||||||
$cart = $this->sale_lib->get_cart();
|
$cart = $this->sale_lib->get_cart();
|
||||||
$x = $this->search_cart_for_item_id($item_id, $cart);
|
$x = $this->search_cart_for_item_id($item_id, $cart);
|
||||||
|
|
||||||
if ($x != null) {
|
if ($x !== null) {
|
||||||
$cart[$x]['name'] = $name;
|
$cart[$x]['name'] = $name;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->sale_lib->set_cart($cart);
|
$this->sale_lib->set_cart($cart);
|
||||||
|
return $this->response->setJSON(['success' => true]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1707,11 +1901,12 @@ class Sales extends Secure_Controller
|
|||||||
$cart = $this->sale_lib->get_cart();
|
$cart = $this->sale_lib->get_cart();
|
||||||
$x = $this->search_cart_for_item_id($item_id, $cart);
|
$x = $this->search_cart_for_item_id($item_id, $cart);
|
||||||
|
|
||||||
if ($x != null) {
|
if ($x !== null) {
|
||||||
$cart[$x]['description'] = $description;
|
$cart[$x]['description'] = $description;
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->sale_lib->set_cart($cart);
|
$this->sale_lib->set_cart($cart);
|
||||||
|
return $this->response->setJSON(['success' => true]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -1730,3 +1925,5 @@ class Sales extends Secure_Controller
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -40,7 +40,7 @@ class Tax_categories extends Secure_Controller
|
|||||||
$search = $this->request->getGet('search');
|
$search = $this->request->getGet('search');
|
||||||
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$sort = $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$sort = $this->sanitizeSortColumn(get_tax_categories_table_headers(), $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS), 'tax_category_id');
|
||||||
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
|
|
||||||
$tax_categories = $this->tax_category->search($search, $limit, $offset, $sort, $order);
|
$tax_categories = $this->tax_category->search($search, $limit, $offset, $sort, $order);
|
||||||
|
|||||||
@@ -50,7 +50,7 @@ class Tax_codes extends Secure_Controller
|
|||||||
$search = $this->request->getGet('search');
|
$search = $this->request->getGet('search');
|
||||||
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$sort = $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$sort = $this->sanitizeSortColumn(get_tax_code_table_headers(), $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS), 'tax_code');
|
||||||
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
|
|
||||||
$tax_codes = $this->tax_code->search($search, $limit, $offset, $sort, $order);
|
$tax_codes = $this->tax_code->search($search, $limit, $offset, $sort, $order);
|
||||||
|
|||||||
@@ -43,7 +43,7 @@ class Tax_jurisdictions extends Secure_Controller
|
|||||||
$search = $this->request->getGet('search');
|
$search = $this->request->getGet('search');
|
||||||
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$sort = $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$sort = $this->sanitizeSortColumn(get_tax_jurisdictions_table_headers(), $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS), 'jurisdiction_id');
|
||||||
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
|
|
||||||
$tax_jurisdictions = $this->tax_jurisdiction->search($search, $limit, $offset, $sort, $order);
|
$tax_jurisdictions = $this->tax_jurisdiction->search($search, $limit, $offset, $sort, $order);
|
||||||
|
|||||||
@@ -81,7 +81,7 @@ class Taxes extends Secure_Controller
|
|||||||
$search = $this->request->getGet('search');
|
$search = $this->request->getGet('search');
|
||||||
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
$limit = $this->request->getGet('limit', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
$offset = $this->request->getGet('offset', FILTER_SANITIZE_NUMBER_INT);
|
||||||
$sort = $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$sort = $this->sanitizeSortColumn(get_tax_rates_manage_table_headers(), $this->request->getGet('sort', FILTER_SANITIZE_FULL_SPECIAL_CHARS), 'tax_rate_id');
|
||||||
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$order = $this->request->getGet('order', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
|
|
||||||
$tax_rates = $this->tax->search($search, $limit, $offset, $sort, $order);
|
$tax_rates = $this->tax->search($search, $limit, $offset, $sort, $order);
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
FROM alpine:3.14
|
FROM alpine:3.14
|
||||||
MAINTAINER jekkos
|
LABEL maintainer="jekkos"
|
||||||
|
|
||||||
ADD database.sql /docker-entrypoint-initdb.d/database.sql
|
ADD database.sql /docker-entrypoint-initdb.d/database.sql
|
||||||
VOLUME /docker-entrypoint-initdb.d
|
VOLUME /docker-entrypoint-initdb.d
|
||||||
|
|||||||
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
namespace App\Database\Migrations;
|
namespace App\Database\Migrations;
|
||||||
|
|
||||||
|
use CodeIgniter\Database\Exceptions\DatabaseException;
|
||||||
use CodeIgniter\Database\Migration;
|
use CodeIgniter\Database\Migration;
|
||||||
|
|
||||||
class Migration_Upgrade_To_3_1_1 extends Migration
|
class Migration_Upgrade_To_3_1_1 extends Migration
|
||||||
@@ -17,7 +18,37 @@ class Migration_Upgrade_To_3_1_1 extends Migration
|
|||||||
public function up(): void
|
public function up(): void
|
||||||
{
|
{
|
||||||
helper('migration');
|
helper('migration');
|
||||||
execute_script(APPPATH . 'Database/Migrations/sqlscripts/3.0.2_to_3.1.1.sql');
|
|
||||||
|
// MariaDB blocks CONVERT TO CHARACTER SET on tables with FK constraints.
|
||||||
|
// Drop all FKs across affected tables before running the SQL script, recreate after.
|
||||||
|
$fkColumns = [
|
||||||
|
['modules', 'module_id'],
|
||||||
|
['stock_locations', 'location_id'],
|
||||||
|
['permissions', 'permission_id'],
|
||||||
|
['people', 'person_id'],
|
||||||
|
['suppliers', 'supplier_id'],
|
||||||
|
['items', 'item_id'],
|
||||||
|
['item_kits', 'item_kit_id'],
|
||||||
|
['sales', 'sale_id'],
|
||||||
|
['receivings', 'receiving_id'],
|
||||||
|
['employees', 'employee_id'],
|
||||||
|
['customers', 'person_id'],
|
||||||
|
];
|
||||||
|
|
||||||
|
$constraints = [];
|
||||||
|
foreach ($fkColumns as [$table, $column]) {
|
||||||
|
foreach (dropAllForeignKeyConstraints($table, $column) as $c) {
|
||||||
|
$constraints[$c['constraintName']] = $c;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!execute_script(APPPATH . 'Database/Migrations/sqlscripts/3.0.2_to_3.1.1.sql')) {
|
||||||
|
throw new DatabaseException('Migration script 3.0.2_to_3.1.1.sql failed. Check logs for details.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$droppedTables = ['sales_suspended', 'sales_suspended_items', 'sales_suspended_items_taxes', 'sales_suspended_payments'];
|
||||||
|
$toRecreate = array_filter($constraints, fn($c) => !in_array($c['tableName'], $droppedTables, true));
|
||||||
|
recreateForeignKeyConstraints(array_values($toRecreate));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -20,7 +20,7 @@ class Migration_database_optimizations extends Migration
|
|||||||
|
|
||||||
$attribute = model(Attribute::class);
|
$attribute = model(Attribute::class);
|
||||||
|
|
||||||
$attribute->delete_orphaned_values();
|
$attribute->deleteOrphanedValues();
|
||||||
|
|
||||||
$this->migrate_duplicate_attribute_values(DECIMAL);
|
$this->migrate_duplicate_attribute_values(DECIMAL);
|
||||||
$this->migrate_duplicate_attribute_values(DATE);
|
$this->migrate_duplicate_attribute_values(DATE);
|
||||||
|
|||||||
46
app/Database/Migrations/20260506000000_AddShortcutKeys.php
Normal file
46
app/Database/Migrations/20260506000000_AddShortcutKeys.php
Normal file
@@ -0,0 +1,46 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Database\Migrations;
|
||||||
|
|
||||||
|
use CodeIgniter\Database\Migration;
|
||||||
|
|
||||||
|
class AddShortcutKeys extends Migration
|
||||||
|
{
|
||||||
|
public function up(): void
|
||||||
|
{
|
||||||
|
$shortcutValues = [
|
||||||
|
['key' => 'key_cancel', 'value' => '27 | ESC'],
|
||||||
|
['key' => 'key_items', 'value' => '49 | ALT + 1'],
|
||||||
|
['key' => 'key_customers', 'value' => '50 | ALT + 2'],
|
||||||
|
['key' => 'key_suspend', 'value' => '51 | ALT + 3'],
|
||||||
|
['key' => 'key_suspended', 'value' => '52 | ALT + 4'],
|
||||||
|
['key' => 'key_amount', 'value' => '53 | ALT + 5'],
|
||||||
|
['key' => 'key_payment', 'value' => '54 | ALT + 6'],
|
||||||
|
['key' => 'key_complete', 'value' => '55 | ALT + 7'],
|
||||||
|
['key' => 'key_finish', 'value' => '56 | ALT + 8'],
|
||||||
|
['key' => 'key_help', 'value' => '57 | ALT + 9'],
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->db->table('app_config')->ignore(true)->insertBatch($shortcutValues);
|
||||||
|
}
|
||||||
|
|
||||||
|
public function down(): void
|
||||||
|
{
|
||||||
|
$shortcutKeys = [
|
||||||
|
'key_cancel',
|
||||||
|
'key_items',
|
||||||
|
'key_customers',
|
||||||
|
'key_suspend',
|
||||||
|
'key_suspended',
|
||||||
|
'key_amount',
|
||||||
|
'key_payment',
|
||||||
|
'key_complete',
|
||||||
|
'key_finish',
|
||||||
|
'key_help',
|
||||||
|
];
|
||||||
|
|
||||||
|
$this->db->table('app_config')
|
||||||
|
->whereIn('key', $shortcutKeys)
|
||||||
|
->delete();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -327,19 +327,6 @@ INSERT INTO `ospos_sales_items` (sale_id, item_id, description, serialnumber, li
|
|||||||
INSERT INTO `ospos_sales_payments` (sale_id, payment_type, payment_amount) SELECT sale_id, payment_type, payment_amount FROM `ospos_sales_suspended_payments`;
|
INSERT INTO `ospos_sales_payments` (sale_id, payment_type, payment_amount) SELECT sale_id, payment_type, payment_amount FROM `ospos_sales_suspended_payments`;
|
||||||
INSERT INTO `ospos_sales_items_taxes` (sale_id, item_id, line, name, percent) SELECT sale_id, item_id, line, name, percent FROM `ospos_sales_suspended_items_taxes`;
|
INSERT INTO `ospos_sales_items_taxes` (sale_id, item_id, line, name, percent) SELECT sale_id, item_id, line, name, percent FROM `ospos_sales_suspended_items_taxes`;
|
||||||
|
|
||||||
ALTER TABLE `ospos_sales_suspended_payments` DROP FOREIGN KEY `ospos_sales_suspended_payments_ibfk_1`;
|
|
||||||
|
|
||||||
ALTER TABLE `ospos_sales_suspended_items_taxes` DROP FOREIGN KEY `ospos_sales_suspended_items_taxes_ibfk_1`;
|
|
||||||
ALTER TABLE `ospos_sales_suspended_items_taxes` DROP FOREIGN KEY `ospos_sales_suspended_items_taxes_ibfk_2`;
|
|
||||||
|
|
||||||
ALTER TABLE `ospos_sales_suspended_items` DROP FOREIGN KEY `ospos_sales_suspended_items_ibfk_1`;
|
|
||||||
ALTER TABLE `ospos_sales_suspended_items` DROP FOREIGN KEY `ospos_sales_suspended_items_ibfk_2`;
|
|
||||||
ALTER TABLE `ospos_sales_suspended_items` DROP FOREIGN KEY `ospos_sales_suspended_items_ibfk_3`;
|
|
||||||
|
|
||||||
ALTER TABLE `ospos_sales_suspended` DROP FOREIGN KEY `ospos_sales_suspended_ibfk_1`;
|
|
||||||
ALTER TABLE `ospos_sales_suspended` DROP FOREIGN KEY `ospos_sales_suspended_ibfk_2`;
|
|
||||||
ALTER TABLE `ospos_sales_suspended` DROP FOREIGN KEY `ospos_sales_suspended_ibfk_3`;
|
|
||||||
|
|
||||||
DROP TABLE `ospos_sales_suspended_payments`, `ospos_sales_suspended_items_taxes`, `ospos_sales_suspended_items`, `ospos_sales_suspended`;
|
DROP TABLE `ospos_sales_suspended_payments`, `ospos_sales_suspended_items_taxes`, `ospos_sales_suspended_items`, `ospos_sales_suspended`;
|
||||||
|
|
||||||
--
|
--
|
||||||
|
|||||||
@@ -140,7 +140,7 @@ CREATE TABLE IF NOT EXISTS `ospos_expense_categories` (
|
|||||||
`category_name` varchar(255) DEFAULT NULL,
|
`category_name` varchar(255) DEFAULT NULL,
|
||||||
`category_description` varchar(255) NOT NULL,
|
`category_description` varchar(255) NOT NULL,
|
||||||
`deleted` int(1) NOT NULL DEFAULT '0'
|
`deleted` int(1) NOT NULL DEFAULT '0'
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
|
|
||||||
-- Table structure for table `ospos_expenses`
|
-- Table structure for table `ospos_expenses`
|
||||||
@@ -154,7 +154,7 @@ CREATE TABLE IF NOT EXISTS `ospos_expenses` (
|
|||||||
`description` varchar(255) NOT NULL,
|
`description` varchar(255) NOT NULL,
|
||||||
`employee_id` int(10) NOT NULL,
|
`employee_id` int(10) NOT NULL,
|
||||||
`deleted` int(1) NOT NULL DEFAULT '0'
|
`deleted` int(1) NOT NULL DEFAULT '0'
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
|
|
||||||
-- Indexes for table `ospos_expense_categories`
|
-- Indexes for table `ospos_expense_categories`
|
||||||
|
|||||||
@@ -75,7 +75,7 @@ CREATE TABLE `ospos_cash_up` (
|
|||||||
`open_employee_id` int(10) NOT NULL,
|
`open_employee_id` int(10) NOT NULL,
|
||||||
`close_employee_id` int(10) NOT NULL,
|
`close_employee_id` int(10) NOT NULL,
|
||||||
`deleted` int(1) NOT NULL DEFAULT '0'
|
`deleted` int(1) NOT NULL DEFAULT '0'
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
-- Indexes for table `ospos_cash_up`
|
-- Indexes for table `ospos_cash_up`
|
||||||
|
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ CREATE TABLE IF NOT EXISTS `ospos_tax_codes` (
|
|||||||
`state` varchar(255) NOT NULL DEFAULT '',
|
`state` varchar(255) NOT NULL DEFAULT '',
|
||||||
`deleted` int(1) NOT NULL DEFAULT 0,
|
`deleted` int(1) NOT NULL DEFAULT 0,
|
||||||
PRIMARY KEY (`tax_code_id`)
|
PRIMARY KEY (`tax_code_id`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
ALTER TABLE `ospos_customers`
|
ALTER TABLE `ospos_customers`
|
||||||
ADD COLUMN `tax_id` varchar(32) NOT NULL DEFAULT '' AFTER `taxable`,
|
ADD COLUMN `tax_id` varchar(32) NOT NULL DEFAULT '' AFTER `taxable`,
|
||||||
@@ -59,7 +59,7 @@ CREATE TABLE `ospos_sales_taxes` (
|
|||||||
`rounding_code` tinyint(2) NOT NULL DEFAULT 0,
|
`rounding_code` tinyint(2) NOT NULL DEFAULT 0,
|
||||||
PRIMARY KEY (`sales_taxes_id`),
|
PRIMARY KEY (`sales_taxes_id`),
|
||||||
KEY `print_sequence` (`sale_id`,`print_sequence`,`tax_group`)
|
KEY `print_sequence` (`sale_id`,`print_sequence`,`tax_group`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
CREATE TABLE IF NOT EXISTS `ospos_tax_jurisdictions` (
|
CREATE TABLE IF NOT EXISTS `ospos_tax_jurisdictions` (
|
||||||
`jurisdiction_id` int(11) NOT NULL AUTO_INCREMENT,
|
`jurisdiction_id` int(11) NOT NULL AUTO_INCREMENT,
|
||||||
@@ -71,7 +71,7 @@ CREATE TABLE IF NOT EXISTS `ospos_tax_jurisdictions` (
|
|||||||
`cascade_sequence` tinyint(2) NOT NULL DEFAULT 0,
|
`cascade_sequence` tinyint(2) NOT NULL DEFAULT 0,
|
||||||
`deleted` int(1) NOT NULL DEFAULT 0,
|
`deleted` int(1) NOT NULL DEFAULT 0,
|
||||||
PRIMARY KEY (`jurisdiction_id`)
|
PRIMARY KEY (`jurisdiction_id`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8 AUTO_INCREMENT=1;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci AUTO_INCREMENT=1;
|
||||||
|
|
||||||
ALTER TABLE `ospos_suppliers`
|
ALTER TABLE `ospos_suppliers`
|
||||||
ADD COLUMN `tax_id` varchar(32) DEFAULT NULL AFTER `account_number`;
|
ADD COLUMN `tax_id` varchar(32) DEFAULT NULL AFTER `account_number`;
|
||||||
@@ -89,7 +89,7 @@ CREATE TABLE IF NOT EXISTS `ospos_tax_rates` (
|
|||||||
`tax_rate` decimal(15,4) NOT NULL DEFAULT 0.0000,
|
`tax_rate` decimal(15,4) NOT NULL DEFAULT 0.0000,
|
||||||
`tax_rounding_code` tinyint(2) NOT NULL DEFAULT 0,
|
`tax_rounding_code` tinyint(2) NOT NULL DEFAULT 0,
|
||||||
PRIMARY KEY (`tax_rate_id`)
|
PRIMARY KEY (`tax_rate_id`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
-- Add support for sales tax report
|
-- Add support for sales tax report
|
||||||
|
|
||||||
|
|||||||
@@ -12,7 +12,7 @@ CREATE TABLE `ospos_sales_payments` (
|
|||||||
`reference_code` varchar(40) NOT NULL DEFAULT '',
|
`reference_code` varchar(40) NOT NULL DEFAULT '',
|
||||||
PRIMARY KEY (`payment_id`),
|
PRIMARY KEY (`payment_id`),
|
||||||
KEY `payment_sale` (`sale_id`, `payment_type`)
|
KEY `payment_sale` (`sale_id`, `payment_type`)
|
||||||
) ENGINE=InnoDB DEFAULT CHARSET=utf8;
|
) ENGINE=InnoDB DEFAULT CHARSET=utf8 COLLATE=utf8_general_ci;
|
||||||
|
|
||||||
INSERT INTO ospos_sales_payments (sale_id, payment_type, payment_amount, payment_user)
|
INSERT INTO ospos_sales_payments (sale_id, payment_type, payment_amount, payment_user)
|
||||||
SELECT payments.sale_id, payments.payment_type, payments.payment_amount, sales.employee_id
|
SELECT payments.sale_id, payments.payment_type, payments.payment_amount, sales.employee_id
|
||||||
|
|||||||
37
app/Database/Seeds/TestDatabaseBootstrapSeeder.php
Normal file
37
app/Database/Seeds/TestDatabaseBootstrapSeeder.php
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Database\Seeds;
|
||||||
|
|
||||||
|
use CodeIgniter\Database\Seeder;
|
||||||
|
use Config\Database;
|
||||||
|
|
||||||
|
class TestDatabaseBootstrapSeeder extends Seeder
|
||||||
|
{
|
||||||
|
public function run(): void
|
||||||
|
{
|
||||||
|
if (ENVIRONMENT !== 'testing') {
|
||||||
|
throw new \RuntimeException('TestDatabaseBootstrapSeeder can only run in the testing environment.');
|
||||||
|
}
|
||||||
|
|
||||||
|
$config = config('Database');
|
||||||
|
$group = $config->tests;
|
||||||
|
$dbName = $group['database'];
|
||||||
|
|
||||||
|
if ($dbName === '' || !str_contains(strtolower($dbName), 'test')) {
|
||||||
|
throw new \RuntimeException("Refusing to reset non-test database: {$dbName}");
|
||||||
|
}
|
||||||
|
|
||||||
|
$serverConn = Database::connect([
|
||||||
|
'hostname' => $group['hostname'],
|
||||||
|
'username' => $group['username'],
|
||||||
|
'password' => $group['password'],
|
||||||
|
'DBDriver' => $group['DBDriver'],
|
||||||
|
'database' => null,
|
||||||
|
'charset' => $group['charset'] ?? 'utf8mb4',
|
||||||
|
'DBCollat' => $group['DBCollat'] ?? 'utf8mb4_general_ci',
|
||||||
|
], false);
|
||||||
|
|
||||||
|
$serverConn->query("DROP DATABASE IF EXISTS `{$dbName}`");
|
||||||
|
$serverConn->query("CREATE DATABASE IF NOT EXISTS `{$dbName}`");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -36,21 +36,26 @@ class Db_log
|
|||||||
private function generate_message(): string
|
private function generate_message(): string
|
||||||
{
|
{
|
||||||
$db = Database::connect();
|
$db = Database::connect();
|
||||||
$last_query = $db->getLastQuery();
|
$lastQuery = $db->getLastQuery();
|
||||||
$affected_rows = $db->affectedRows();
|
|
||||||
$execution_time = $this->convert_time($last_query->getDuration());
|
if ($lastQuery === null) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$affectedRows = $db->affectedRows();
|
||||||
|
$executionTime = $this->convert_time($lastQuery->getDuration());
|
||||||
|
|
||||||
$message = '*** Query: ' . date('Y-m-d H:i:s T') . ' *******************'
|
$message = '*** Query: ' . date('Y-m-d H:i:s T') . ' *******************'
|
||||||
. "\n" . $last_query->getQuery()
|
. "\n" . $lastQuery->getQuery()
|
||||||
. "\n Affected rows: $affected_rows"
|
. "\n Affected rows: $affectedRows"
|
||||||
. "\n Execution Time: " . $execution_time['time'] . ' ' . $execution_time['unit'];
|
. "\n Execution Time: " . $executionTime['time'] . ' ' . $executionTime['unit'];
|
||||||
|
|
||||||
$long_query = ($execution_time['unit'] === 's') && ($execution_time['time'] > 0.5);
|
$longQuery = ($executionTime['unit'] === 's') && ($executionTime['time'] > 0.5);
|
||||||
if ($long_query) {
|
if ($longQuery) {
|
||||||
$message .= ' [LONG RUNNING QUERY]';
|
$message .= ' [LONG RUNNING QUERY]';
|
||||||
}
|
}
|
||||||
|
|
||||||
return $this->config->db_log_only_long && !$long_query ? '' : $message;
|
return $this->config->db_log_only_long && !$longQuery ? '' : $message;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@@ -4,6 +4,8 @@ namespace App\Events;
|
|||||||
|
|
||||||
use App\Libraries\MY_Migration;
|
use App\Libraries\MY_Migration;
|
||||||
use App\Models\Appconfig;
|
use App\Models\Appconfig;
|
||||||
|
use CodeIgniter\Session\Handlers\DatabaseHandler;
|
||||||
|
use CodeIgniter\Session\Handlers\FileHandler;
|
||||||
use CodeIgniter\Session\Session;
|
use CodeIgniter\Session\Session;
|
||||||
use Config\OSPOS;
|
use Config\OSPOS;
|
||||||
use Config\Services;
|
use Config\Services;
|
||||||
@@ -19,38 +21,47 @@ class Load_config
|
|||||||
{
|
{
|
||||||
public Session $session;
|
public Session $session;
|
||||||
|
|
||||||
/**
|
|
||||||
* Loads configuration from database into App CI config and then applies those settings
|
|
||||||
*/
|
|
||||||
public function load_config(): void
|
public function load_config(): void
|
||||||
{
|
{
|
||||||
// Migrations
|
|
||||||
$migration_config = config('Migrations');
|
$migration_config = config('Migrations');
|
||||||
$migration = new MY_Migration($migration_config);
|
$migration = new MY_Migration($migration_config);
|
||||||
|
|
||||||
$this->session = session();
|
$this->session = session();
|
||||||
|
|
||||||
// Database Configuration
|
|
||||||
$config = config(OSPOS::class);
|
$config = config(OSPOS::class);
|
||||||
|
|
||||||
if (!$migration->is_latest()) {
|
if (!$migration->is_latest()) {
|
||||||
$this->session->destroy();
|
$this->session->destroy();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Language
|
$this->setDefaultLanguage($config);
|
||||||
$language_exists = file_exists('../app/Language/' . current_language_code());
|
|
||||||
|
|
||||||
if (current_language_code() == null || current_language() == null || !$language_exists) { // TODO: current_language() is undefined
|
|
||||||
$config->settings['language'] = 'english';
|
|
||||||
$config->settings['language_code'] = 'en';
|
|
||||||
}
|
|
||||||
|
|
||||||
$language = Services::language();
|
$language = Services::language();
|
||||||
$language->setLocale($config->settings['language_code']);
|
$language->setLocale(current_language_code());
|
||||||
|
|
||||||
// Time Zone
|
|
||||||
date_default_timezone_set($config->settings['timezone'] ?? ini_get('date.timezone'));
|
date_default_timezone_set($config->settings['timezone'] ?? ini_get('date.timezone'));
|
||||||
|
|
||||||
bcscale(max(2, totals_decimals() + tax_decimals()));
|
bcscale(max(2, totals_decimals() + tax_decimals()));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function setDefaultLanguage(OSPOS $config): void
|
||||||
|
{
|
||||||
|
$languageCode = $config->settings['language_code'] ?? null;
|
||||||
|
|
||||||
|
if (empty($config->settings) || $languageCode === null) {
|
||||||
|
$config->settings['language'] = 'english';
|
||||||
|
$config->settings['language_code'] = 'en';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!$this->languageExists($languageCode)) {
|
||||||
|
$config->settings['language'] = 'english';
|
||||||
|
$config->settings['language_code'] = 'en';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function languageExists(string $languageCode): bool
|
||||||
|
{
|
||||||
|
return file_exists(APPPATH . 'Language/' . $languageCode);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
35
app/Helpers/attribute_helper.php
Normal file
35
app/Helpers/attribute_helper.php
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Translates the attribute type to the corresponding database column name.
|
||||||
|
*
|
||||||
|
* Maps attribute type constants to their corresponding attribute_values table columns.
|
||||||
|
* Defaults to 'attribute_value' for TEXT, DROPDOWN and CHECKBOX attribute types.
|
||||||
|
*
|
||||||
|
* @param string $input The attribute type constant (DATE, DECIMAL, etc.)
|
||||||
|
* @return string The database column name for storing this attribute type
|
||||||
|
*/
|
||||||
|
function getAttributeDataType(string $input): string
|
||||||
|
{
|
||||||
|
$columnMap = [
|
||||||
|
DATE => 'attribute_date',
|
||||||
|
DECIMAL => 'attribute_decimal',
|
||||||
|
];
|
||||||
|
|
||||||
|
return $columnMap[$input] ?? 'attribute_value';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates that the provided data type is an allowed attribute value type.
|
||||||
|
*
|
||||||
|
* @param string $dataType
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
function validateAttributeValueType(string $dataType): void
|
||||||
|
{
|
||||||
|
$attributeValueTypes = ['attribute_value', 'attribute_decimal', 'attribute_date'];
|
||||||
|
|
||||||
|
if (!in_array($dataType, $attributeValueTypes, true)) {
|
||||||
|
throw new InvalidArgumentException('Invalid data type');
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,10 +1,10 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param array $stock_locations
|
* @param array $stock_locations
|
||||||
* @param array $attributes
|
* @param array $attributes
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
|
|
||||||
function generate_import_items_csv(array $stock_locations, array $attributes): string
|
function generate_import_items_csv(array $stock_locations, array $attributes): string
|
||||||
{
|
{
|
||||||
$csv_headers = pack('CCC', 0xef, 0xbb, 0xbf); // Encode the Byte-Order Mark (BOM) so that UTF-8 File headers display properly in Microsoft Excel
|
$csv_headers = pack('CCC', 0xef, 0xbb, 0xbf); // Encode the Byte-Order Mark (BOM) so that UTF-8 File headers display properly in Microsoft Excel
|
||||||
|
|||||||
@@ -22,9 +22,7 @@ function current_language_code(bool $load_system_language = false): string
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$language_code = $config['language_code'];
|
return $config['language_code'] ?? DEFAULT_LANGUAGE_CODE;
|
||||||
|
|
||||||
return empty($language_code) ? DEFAULT_LANGUAGE_CODE : $language_code;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -45,9 +43,7 @@ function current_language(bool $load_system_language = false): string
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$language = $config['language'];
|
return $config['language'] ?? DEFAULT_LANGUAGE;
|
||||||
|
|
||||||
return empty($language) ? DEFAULT_LANGUAGE : $language;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -369,6 +365,74 @@ function to_currency_no_money(?string $number): string
|
|||||||
return to_decimals($number, 'currency_decimals');
|
return to_decimals($number, 'currency_decimals');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build the secondary currency rendering context from app config values.
|
||||||
|
*
|
||||||
|
* @param array $config
|
||||||
|
* @return array{show:bool,rate:float,symbol:string,code:string,decimals:int}
|
||||||
|
*/
|
||||||
|
function secondary_currency_context(array $config): array
|
||||||
|
{
|
||||||
|
$rate = (float) ($config['secondary_currency_rate'] ?? 0);
|
||||||
|
$symbol = trim((string) ($config['secondary_currency_symbol'] ?? ''));
|
||||||
|
$code = trim((string) ($config['secondary_currency_code'] ?? ''));
|
||||||
|
$decimals = (int) ($config['secondary_currency_decimals'] ?? ($config['currency_decimals'] ?? DEFAULT_PRECISION));
|
||||||
|
|
||||||
|
return [
|
||||||
|
'show' => (($config['secondary_currency_enabled'] ?? false) == 1) && $rate > 0,
|
||||||
|
'rate' => $rate,
|
||||||
|
'symbol' => $symbol,
|
||||||
|
'code' => $code,
|
||||||
|
'decimals' => $decimals,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render a value in the secondary currency.
|
||||||
|
*
|
||||||
|
* @param float|int|string|null $number
|
||||||
|
* @param array{show:bool,rate:float,symbol:string,code:string,decimals:int} $secondaryCurrency
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function to_secondary_currency(float|int|string|null $number, array $secondaryCurrency): string
|
||||||
|
{
|
||||||
|
if (!isset($number) || !$secondaryCurrency['show']) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$config = config(OSPOS::class)->settings;
|
||||||
|
$amount = (float) $number * (float) $secondaryCurrency['rate'];
|
||||||
|
$fmt = new NumberFormatter($config['number_locale'], NumberFormatter::CURRENCY);
|
||||||
|
$fmt->setAttribute(NumberFormatter::MIN_FRACTION_DIGITS, $secondaryCurrency['decimals']);
|
||||||
|
$fmt->setAttribute(NumberFormatter::MAX_FRACTION_DIGITS, $secondaryCurrency['decimals']);
|
||||||
|
|
||||||
|
if (empty($config['thousands_separator'])) {
|
||||||
|
$fmt->setTextAttribute(NumberFormatter::GROUPING_SEPARATOR_SYMBOL, '');
|
||||||
|
}
|
||||||
|
|
||||||
|
$fmt->setSymbol(NumberFormatter::CURRENCY_SYMBOL, $secondaryCurrency['symbol'] !== '' ? $secondaryCurrency['symbol'] : ($secondaryCurrency['code'] !== '' ? $secondaryCurrency['code'] : ''));
|
||||||
|
|
||||||
|
return $fmt->format($amount);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render the secondary and primary currency amounts together.
|
||||||
|
*
|
||||||
|
* @param float|int|string|null $number
|
||||||
|
* @param array{show:bool,rate:float,symbol:string,code:string,decimals:int} $secondaryCurrency
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
function to_secondary_currency_dual(float|int|string|null $number, array $secondaryCurrency): string
|
||||||
|
{
|
||||||
|
$secondary = to_secondary_currency($number, $secondaryCurrency);
|
||||||
|
|
||||||
|
if ($secondary === '') {
|
||||||
|
return to_currency((string) $number);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $secondary . '<br>' . to_currency((string) $number);
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @param string|null $number
|
* @param string|null $number
|
||||||
* @return string
|
* @return string
|
||||||
|
|||||||
@@ -172,6 +172,7 @@ function dropAllForeignKeyConstraints(string $table, string $column): array {
|
|||||||
WHERE kcu.TABLE_SCHEMA = DATABASE()
|
WHERE kcu.TABLE_SCHEMA = DATABASE()
|
||||||
AND ((kcu.REFERENCED_TABLE_NAME = '" . $db->getPrefix() . "$table' AND kcu.REFERENCED_COLUMN_NAME = '$column')
|
AND ((kcu.REFERENCED_TABLE_NAME = '" . $db->getPrefix() . "$table' AND kcu.REFERENCED_COLUMN_NAME = '$column')
|
||||||
OR (kcu.TABLE_NAME = '" . $db->getPrefix() . "$table' AND kcu.COLUMN_NAME = '$column'))
|
OR (kcu.TABLE_NAME = '" . $db->getPrefix() . "$table' AND kcu.COLUMN_NAME = '$column'))
|
||||||
|
AND rc.CONSTRAINT_NAME IS NOT NULL
|
||||||
");
|
");
|
||||||
|
|
||||||
$deletedConstraints = [];
|
$deletedConstraints = [];
|
||||||
|
|||||||
@@ -11,56 +11,54 @@ function check_encryption(): bool
|
|||||||
$old_key = config('Encryption')->key;
|
$old_key = config('Encryption')->key;
|
||||||
|
|
||||||
if ((empty($old_key)) || (strlen($old_key) < 64)) {
|
if ((empty($old_key)) || (strlen($old_key) < 64)) {
|
||||||
// Create Key
|
|
||||||
$encryption = new Encryption();
|
$encryption = new Encryption();
|
||||||
$key = bin2hex($encryption->createKey());
|
$key = bin2hex($encryption->createKey());
|
||||||
config('Encryption')->key = $key;
|
config('Encryption')->key = $key;
|
||||||
|
|
||||||
// Write to .env
|
|
||||||
$config_path = ROOTPATH . '.env';
|
$config_path = ROOTPATH . '.env';
|
||||||
$new_config_path = WRITEPATH . '/backup/.env';
|
|
||||||
$backup_path = WRITEPATH . '/backup/.env.bak';
|
$backup_path = WRITEPATH . '/backup/.env.bak';
|
||||||
|
|
||||||
$backup_folder = WRITEPATH . '/backup';
|
$backup_folder = WRITEPATH . '/backup';
|
||||||
|
|
||||||
if (!file_exists($backup_folder) && !mkdir($backup_folder)) {
|
if (!file_exists($backup_folder)) {
|
||||||
log_message('error', 'Could not create backup folder');
|
@mkdir($backup_folder, 0750, true);
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!copy($config_path, $backup_path)) {
|
if (!file_exists($config_path)) {
|
||||||
log_message('error', "Unable to copy $config_path to $backup_path");
|
$example_path = ROOTPATH . '.env.example';
|
||||||
|
if (file_exists($example_path)) {
|
||||||
|
@copy($example_path, $config_path);
|
||||||
|
} else {
|
||||||
|
@file_put_contents($config_path, "# OSPOS Configuration\n\n");
|
||||||
|
}
|
||||||
|
@chmod($config_path, 0640);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy to backup
|
if (file_exists($config_path)) {
|
||||||
@chmod($config_path, 0660);
|
@copy($config_path, $backup_path);
|
||||||
@chmod($backup_path, 0660);
|
@chmod($backup_path, 0640);
|
||||||
|
@chmod($config_path, 0640);
|
||||||
|
|
||||||
$config_file = file_get_contents($config_path);
|
$config_file = file_get_contents($config_path);
|
||||||
$config_file = preg_replace("/(encryption\.key.*=.*)('.*')/", "$1'$key'", $config_file);
|
|
||||||
|
|
||||||
if (!empty($old_key)) {
|
if (strpos($config_file, 'encryption.key') !== false) {
|
||||||
$old_line = "# encryption.key = '$old_key' REMOVE IF UNNEEDED\r\n";
|
$config_file = preg_replace("/(encryption\.key.*=.*)('.*')/", "$1'$key'", $config_file);
|
||||||
$insertion_point = stripos($config_file, 'encryption.key');
|
} else {
|
||||||
$config_file = substr_replace($config_file, $old_line, $insertion_point, 0);
|
$config_file .= "\nencryption.key = '$key'\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!empty($old_key)) {
|
||||||
|
$old_line = "# encryption.key = '$old_key' REMOVE IF UNNEEDED\r\n";
|
||||||
|
$insertion_point = stripos($config_file, 'encryption.key');
|
||||||
|
if ($insertion_point !== false) {
|
||||||
|
$config_file = substr_replace($config_file, $old_line, $insertion_point, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@file_put_contents($config_path, $config_file);
|
||||||
|
@chmod($config_path, 0640);
|
||||||
|
|
||||||
|
log_message('info', "Updated encryption key in $config_path");
|
||||||
}
|
}
|
||||||
|
|
||||||
$handle = @fopen($config_path, 'w+');
|
|
||||||
|
|
||||||
if (empty($handle)) {
|
|
||||||
log_message('error', "Unable to open $config_path for updating");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@chmod($config_path, 0660);
|
|
||||||
$write_failed = !fwrite($handle, $config_file);
|
|
||||||
fclose($handle);
|
|
||||||
|
|
||||||
if ($write_failed) {
|
|
||||||
log_message('error', "Unable to write to $config_path for updating.");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
log_message('info', "File $config_path has been updated.");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@@ -74,23 +72,14 @@ function abort_encryption_conversion(): void
|
|||||||
$config_path = ROOTPATH . '.env';
|
$config_path = ROOTPATH . '.env';
|
||||||
$backup_path = WRITEPATH . '/backup/.env.bak';
|
$backup_path = WRITEPATH . '/backup/.env.bak';
|
||||||
|
|
||||||
$config_file = file_get_contents($backup_path);
|
if (!file_exists($backup_path)) {
|
||||||
|
return;
|
||||||
$handle = @fopen($config_path, 'w+');
|
|
||||||
|
|
||||||
if (empty($handle)) {
|
|
||||||
log_message('error', "Unable to open $config_path to undo encryption conversion");
|
|
||||||
} else {
|
|
||||||
@chmod($config_path, 0660);
|
|
||||||
$write_failed = !fwrite($handle, $config_file);
|
|
||||||
fclose($handle);
|
|
||||||
|
|
||||||
if ($write_failed) {
|
|
||||||
log_message('error', "Unable to write to $config_path to undo encryption conversion.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
log_message('info', "File $config_path has been updated to undo encryption conversion");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@chmod($config_path, 0640);
|
||||||
|
$config_file = file_get_contents($backup_path);
|
||||||
|
@file_put_contents($config_path, $config_file);
|
||||||
|
log_message('info', "Restored $config_path from backup");
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -99,13 +88,9 @@ function abort_encryption_conversion(): void
|
|||||||
function remove_backup(): void
|
function remove_backup(): void
|
||||||
{
|
{
|
||||||
$backup_path = WRITEPATH . '/backup/.env.bak';
|
$backup_path = WRITEPATH . '/backup/.env.bak';
|
||||||
if (! file_exists($backup_path)) {
|
if (!file_exists($backup_path)) {
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!unlink($backup_path)) {
|
@unlink($backup_path);
|
||||||
log_message('error', "Unable to remove $backup_path.");
|
log_message('info', "Removed $backup_path");
|
||||||
return;
|
|
||||||
}
|
|
||||||
log_message('info', "File $backup_path has been removed");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ use App\Models\Employee;
|
|||||||
use App\Models\Item_taxes;
|
use App\Models\Item_taxes;
|
||||||
use App\Models\Tax_category;
|
use App\Models\Tax_category;
|
||||||
use CodeIgniter\Database\ResultInterface;
|
use CodeIgniter\Database\ResultInterface;
|
||||||
|
use CodeIgniter\HTTP\IncomingRequest;
|
||||||
use CodeIgniter\Session\Session;
|
use CodeIgniter\Session\Session;
|
||||||
use Config\OSPOS;
|
use Config\OSPOS;
|
||||||
use Config\Services;
|
use Config\Services;
|
||||||
@@ -577,8 +578,8 @@ function item_kit_headers(): array
|
|||||||
['item_kit_number' => lang('Item_kits.item_kit_number')],
|
['item_kit_number' => lang('Item_kits.item_kit_number')],
|
||||||
['name' => lang('Item_kits.name')],
|
['name' => lang('Item_kits.name')],
|
||||||
['description' => lang('Item_kits.description')],
|
['description' => lang('Item_kits.description')],
|
||||||
['total_cost_price' => lang('Items.cost_price'), 'sortable' => FALSE],
|
['total_cost_price' => lang('Items.cost_price'), 'sortable' => false],
|
||||||
['total_unit_price' => lang('Items.unit_price'), 'sortable' => FALSE]
|
['total_unit_price' => lang('Items.unit_price'), 'sortable' => false]
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -654,7 +655,7 @@ function expand_attribute_values(array $definition_names, array $row): array
|
|||||||
foreach ($definition_names as $definition_id => $definitionInfo) {
|
foreach ($definition_names as $definition_id => $definitionInfo) {
|
||||||
if (isset($indexed_values[$definition_id])) {
|
if (isset($indexed_values[$definition_id])) {
|
||||||
$raw_value = $indexed_values[$definition_id];
|
$raw_value = $indexed_values[$definition_id];
|
||||||
|
|
||||||
// Format DECIMAL attributes according to locale
|
// Format DECIMAL attributes according to locale
|
||||||
if (is_array($definitionInfo) && isset($definitionInfo['type']) && $definitionInfo['type'] === DECIMAL) {
|
if (is_array($definitionInfo) && isset($definitionInfo['type']) && $definitionInfo['type'] === DECIMAL) {
|
||||||
$attribute_values["$definition_id"] = to_decimals($raw_value);
|
$attribute_values["$definition_id"] = to_decimals($raw_value);
|
||||||
@@ -742,7 +743,7 @@ function get_expense_category_manage_table_headers(): string
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Gets the html data row for the expenses category
|
* Gets the html data row for the expense category
|
||||||
*/
|
*/
|
||||||
function get_expense_category_data_row(object $expense_category): array
|
function get_expense_category_data_row(object $expense_category): array
|
||||||
{
|
{
|
||||||
@@ -841,7 +842,7 @@ function get_expenses_data_last_row(object $expense): array
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the expenses payments summary
|
* Get the expense payments summary
|
||||||
*/
|
*/
|
||||||
function get_expenses_manage_payments_summary(array $payments, ResultInterface $expenses): string // TODO: $expenses is passed but never used.
|
function get_expenses_manage_payments_summary(array $payments, ResultInterface $expenses): string // TODO: $expenses is passed but never used.
|
||||||
{
|
{
|
||||||
@@ -933,22 +934,22 @@ function get_controller(): string
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Restores filter values from URL query string.
|
* Restores filter values from the URL query string.
|
||||||
*
|
*
|
||||||
* @param CodeIgniter\HTTP\IncomingRequest $request The request object
|
* @param IncomingRequest $request The request object
|
||||||
* @return array Array with 'start_date', 'end_date', and 'selected_filters' keys
|
* @return array Array with 'start_date', 'end_date', and 'selected_filters' keys
|
||||||
*/
|
*/
|
||||||
function restoreTableFilters($request): array
|
function restoreTableFilters(IncomingRequest $request): array
|
||||||
{
|
{
|
||||||
$startDate = $request->getGet('start_date', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$startDate = $request->getGet('start_date', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
$endDate = $request->getGet('end_date', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$endDate = $request->getGet('end_date', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
$urlFilters = $request->getGet('filters', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
$urlFilters = $request->getGet('filters', FILTER_SANITIZE_FULL_SPECIAL_CHARS);
|
||||||
|
|
||||||
return array_filter([
|
return array_filter([
|
||||||
'start_date' => $startDate ?: null,
|
'start_date' => $startDate ?: null,
|
||||||
'end_date' => $endDate ?: null,
|
'end_date' => $endDate ?: null,
|
||||||
'selected_filters' => $urlFilters ?? []
|
'selected_filters' => $urlFilters ?? []
|
||||||
], function($value) {
|
], function ($value) {
|
||||||
return $value !== null && $value !== [];
|
return $value !== null && $value !== [];
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -143,8 +143,7 @@ function get_tax_rates_manage_table_headers(): string
|
|||||||
*/
|
*/
|
||||||
function get_tax_rates_data_row($tax_rates_row): array
|
function get_tax_rates_data_row($tax_rates_row): array
|
||||||
{
|
{
|
||||||
$router = service('router');
|
$controller_name = 'taxes';
|
||||||
$controller_name = strtolower($router->controllerName());
|
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'tax_rate_id' => $tax_rates_row->tax_rate_id,
|
'tax_rate_id' => $tax_rates_row->tax_rate_id,
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ if (!function_exists('base64url_encode')) {
|
|||||||
* @param string $data
|
* @param string $data
|
||||||
* @return string
|
* @return string
|
||||||
*/
|
*/
|
||||||
function base64url_encode($data)
|
function base64url_encode(string $data): string
|
||||||
{
|
{
|
||||||
return rtrim(strtr(base64_encode($data), '+/', '-_'), '=');
|
return rtrim(strtr(base64_encode($data), '+/', '-_'), '=');
|
||||||
}
|
}
|
||||||
@@ -20,7 +20,7 @@ if (!function_exists('base64url_decode')) {
|
|||||||
* @param string $data
|
* @param string $data
|
||||||
* @return string|false
|
* @return string|false
|
||||||
*/
|
*/
|
||||||
function base64url_decode($data)
|
function base64url_decode(string $data): false|string
|
||||||
{
|
{
|
||||||
$remainder = strlen($data) % 4;
|
$remainder = strlen($data) % 4;
|
||||||
if ($remainder) {
|
if ($remainder) {
|
||||||
@@ -28,4 +28,4 @@ if (!function_exists('base64url_decode')) {
|
|||||||
}
|
}
|
||||||
return base64_decode(strtr($data, '-_', '+/'));
|
return base64_decode(strtr($data, '-_', '+/'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
return [
|
return [
|
||||||
"all" => "الجميع",
|
'all' => "الجميع",
|
||||||
"columns" => "أعمدة",
|
'columns' => "أعمدة",
|
||||||
"hide_show_pagination" => "عرض/إخفاء أرقام الصفحات",
|
'hide_show_pagination' => "عرض/إخفاء أرقام الصفحات",
|
||||||
"loading" => "جارى التحميل، برجاء الإنتظار ...",
|
'loading' => "جارى التحميل، برجاء الإنتظار",
|
||||||
"page_from_to" => "عرض {0} إلى {1} من {2} صفوف",
|
'page_from_to' => "عرض {0} إلى {1} من {2} صفوف",
|
||||||
"refresh" => "إعادة تحميل",
|
'refresh' => "إعادة تحميل",
|
||||||
"rows_per_page" => "{0} صف بالصفحة",
|
'rows_per_page' => "{0} صف بالصفحة",
|
||||||
"toggle" => "تغيير",
|
'toggle' => "تغيير",
|
||||||
];
|
];
|
||||||
|
|||||||
49
app/Language/ar-EG/Calendar.php
Normal file
49
app/Language/ar-EG/Calendar.php
Normal file
@@ -0,0 +1,49 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
return [
|
||||||
|
"su" => "أحد",
|
||||||
|
"mo" => "اثنين",
|
||||||
|
"tu" => "ثلاثاء",
|
||||||
|
"we" => "أربعاء",
|
||||||
|
"th" => "خميس",
|
||||||
|
"fr" => "جمعة",
|
||||||
|
"sa" => "سبت",
|
||||||
|
"sun" => "الأحد",
|
||||||
|
"mon" => "الاثنين",
|
||||||
|
"tue" => "الثلاثاء",
|
||||||
|
"wed" => "الأربعاء",
|
||||||
|
"thu" => "الخميس",
|
||||||
|
"fri" => "الجمعة",
|
||||||
|
"sat" => "السبت",
|
||||||
|
"sunday" => "الأحد",
|
||||||
|
"monday" => "الاثنين",
|
||||||
|
"tuesday" => "الثلاثاء",
|
||||||
|
"wednesday" => "الأربعاء",
|
||||||
|
"thursday" => "الخميس",
|
||||||
|
"friday" => "الجمعة",
|
||||||
|
"saturday" => "السبت",
|
||||||
|
"jan" => "يناير",
|
||||||
|
"feb" => "فبراير",
|
||||||
|
"mar" => "مارس",
|
||||||
|
"apr" => "أبريل",
|
||||||
|
"may" => "مايو",
|
||||||
|
"jun" => "يونيو",
|
||||||
|
"jul" => "يوليو",
|
||||||
|
"aug" => "أغسطس",
|
||||||
|
"sep" => "سبتمبر",
|
||||||
|
"oct" => "أكتوبر",
|
||||||
|
"nov" => "نوفمبر",
|
||||||
|
"dec" => "ديسمبر",
|
||||||
|
"january" => "يناير",
|
||||||
|
"february" => "فبراير",
|
||||||
|
"march" => "مارس",
|
||||||
|
"april" => "أبريل",
|
||||||
|
"mayl" => "مايو",
|
||||||
|
"june" => "يونيو",
|
||||||
|
"july" => "يوليو",
|
||||||
|
"august" => "أغسطس",
|
||||||
|
"september" => "سبتمبر",
|
||||||
|
"october" => "أكتوبر",
|
||||||
|
"november" => "نوفمبر",
|
||||||
|
"december" => "ديسمبر",
|
||||||
|
];
|
||||||
@@ -282,6 +282,7 @@ return [
|
|||||||
"right" => "يمين",
|
"right" => "يمين",
|
||||||
"sales_invoice_format" => "شكل فاتورة البيع",
|
"sales_invoice_format" => "شكل فاتورة البيع",
|
||||||
"sales_quote_format" => "شكل فاتورة عرض الاسعار",
|
"sales_quote_format" => "شكل فاتورة عرض الاسعار",
|
||||||
|
"mailpath_invalid" => "",
|
||||||
"saved_successfully" => "تم حفظ التهيئة بنجاح.",
|
"saved_successfully" => "تم حفظ التهيئة بنجاح.",
|
||||||
"saved_unsuccessfully" => "لم يتم حفظ التهيئة بنجاح.",
|
"saved_unsuccessfully" => "لم يتم حفظ التهيئة بنجاح.",
|
||||||
"security_issue" => "تحذير من ثغرة أمنية",
|
"security_issue" => "تحذير من ثغرة أمنية",
|
||||||
|
|||||||
@@ -9,6 +9,15 @@ return [
|
|||||||
"login" => "دخول",
|
"login" => "دخول",
|
||||||
"logout" => "تسجيل خروج",
|
"logout" => "تسجيل خروج",
|
||||||
"migration_needed" => "سيبدأ ترحيل قاعدة البيانات إلى{0} بعد تسجيل الدخول.",
|
"migration_needed" => "سيبدأ ترحيل قاعدة البيانات إلى{0} بعد تسجيل الدخول.",
|
||||||
|
"migration_required" => "",
|
||||||
|
"migration_auth_message" => "",
|
||||||
|
"migration_initializing" => "",
|
||||||
|
"migration_running" => "",
|
||||||
|
"migration_complete" => "",
|
||||||
|
"migration_complete_login" => "",
|
||||||
|
"migration_failed" => "",
|
||||||
|
"migration_error_connection" => "",
|
||||||
|
"migration_complete_redirect" => "",
|
||||||
"password" => "كلمة السر",
|
"password" => "كلمة السر",
|
||||||
"required_username" => "",
|
"required_username" => "",
|
||||||
"username" => "اسم المستخدم",
|
"username" => "اسم المستخدم",
|
||||||
|
|||||||
@@ -73,6 +73,12 @@ return [
|
|||||||
"employee" => "الموظف",
|
"employee" => "الموظف",
|
||||||
"entry" => "ادخال",
|
"entry" => "ادخال",
|
||||||
"error_editing_item" => "خطاء فى تحرير الصنف",
|
"error_editing_item" => "خطاء فى تحرير الصنف",
|
||||||
|
"negative_price_invalid" => "",
|
||||||
|
"negative_quantity_invalid" => "",
|
||||||
|
"negative_discount_invalid" => "",
|
||||||
|
"discount_percent_exceeds_100" => "",
|
||||||
|
"discount_exceeds_item_total" => "",
|
||||||
|
"negative_total_invalid" => "",
|
||||||
"find_or_scan_item" => "بحث/مسح باركود صنف",
|
"find_or_scan_item" => "بحث/مسح باركود صنف",
|
||||||
"find_or_scan_item_or_receipt" => "بحث/مسح باركود صنف أو ايصال",
|
"find_or_scan_item_or_receipt" => "بحث/مسح باركود صنف أو ايصال",
|
||||||
"giftcard" => "بطاقة هدية",
|
"giftcard" => "بطاقة هدية",
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
return [
|
return [
|
||||||
"all" => "الكل",
|
'all' => "الكل",
|
||||||
"columns" => "أعمدة",
|
'columns' => "أعمدة",
|
||||||
"hide_show_pagination" => "عرض/إخفاء أرقام الصفحات",
|
'hide_show_pagination' => "عرض/إخفاء أرقام الصفحات",
|
||||||
"loading" => "جارى التحميل، برجاء الإنتظار ...",
|
'loading' => "جارى التحميل، برجاء الإنتظار",
|
||||||
"page_from_to" => "عرض {0} إلى {1} من {2} صفوف",
|
'page_from_to' => "عرض {0} إلى {1} من {2} صفوف",
|
||||||
"refresh" => "إعادة تحميل",
|
'refresh' => "إعادة تحميل",
|
||||||
"rows_per_page" => "{0} صف بالصفحة",
|
'rows_per_page' => "{0} صف بالصفحة",
|
||||||
"toggle" => "تغيير",
|
'toggle' => "تغيير",
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -282,6 +282,7 @@ return [
|
|||||||
"right" => "يمين",
|
"right" => "يمين",
|
||||||
"sales_invoice_format" => "شكل فاتورة البيع",
|
"sales_invoice_format" => "شكل فاتورة البيع",
|
||||||
"sales_quote_format" => "شكل فاتورة عرض الاسعار",
|
"sales_quote_format" => "شكل فاتورة عرض الاسعار",
|
||||||
|
"mailpath_invalid" => "",
|
||||||
"saved_successfully" => "تم حفظ التهيئة بنجاح.",
|
"saved_successfully" => "تم حفظ التهيئة بنجاح.",
|
||||||
"saved_unsuccessfully" => "لم يتم حفظ التهيئة بنجاح.",
|
"saved_unsuccessfully" => "لم يتم حفظ التهيئة بنجاح.",
|
||||||
"security_issue" => "تحذير من ثغرة أمنية",
|
"security_issue" => "تحذير من ثغرة أمنية",
|
||||||
|
|||||||
@@ -9,6 +9,15 @@ return [
|
|||||||
"login" => "دخول",
|
"login" => "دخول",
|
||||||
"logout" => "تسجيل خروج",
|
"logout" => "تسجيل خروج",
|
||||||
"migration_needed" => "سيبدأ ترحيل قاعدة البيانات إلى{0} بعد تسجيل الدخول.",
|
"migration_needed" => "سيبدأ ترحيل قاعدة البيانات إلى{0} بعد تسجيل الدخول.",
|
||||||
|
"migration_required" => "",
|
||||||
|
"migration_auth_message" => "",
|
||||||
|
"migration_initializing" => "",
|
||||||
|
"migration_running" => "",
|
||||||
|
"migration_complete" => "",
|
||||||
|
"migration_complete_login" => "",
|
||||||
|
"migration_failed" => "",
|
||||||
|
"migration_error_connection" => "",
|
||||||
|
"migration_complete_redirect" => "",
|
||||||
"password" => "كلمة السر",
|
"password" => "كلمة السر",
|
||||||
"required_username" => "خانة أسم المستخدم مطلوبة.",
|
"required_username" => "خانة أسم المستخدم مطلوبة.",
|
||||||
"username" => "اسم المستخدم",
|
"username" => "اسم المستخدم",
|
||||||
|
|||||||
@@ -73,6 +73,12 @@ return [
|
|||||||
"employee" => "الموظف",
|
"employee" => "الموظف",
|
||||||
"entry" => "ادخال",
|
"entry" => "ادخال",
|
||||||
"error_editing_item" => "خطاء فى تعديل المادة",
|
"error_editing_item" => "خطاء فى تعديل المادة",
|
||||||
|
"negative_price_invalid" => "",
|
||||||
|
"negative_quantity_invalid" => "",
|
||||||
|
"negative_discount_invalid" => "",
|
||||||
|
"discount_percent_exceeds_100" => "",
|
||||||
|
"discount_exceeds_item_total" => "",
|
||||||
|
"negative_total_invalid" => "",
|
||||||
"find_or_scan_item" => "بحث/مسح باركود المادة",
|
"find_or_scan_item" => "بحث/مسح باركود المادة",
|
||||||
"find_or_scan_item_or_receipt" => "بحث/مسح باركود المادة أو الايصال",
|
"find_or_scan_item_or_receipt" => "بحث/مسح باركود المادة أو الايصال",
|
||||||
"giftcard" => "بطاقة هدية",
|
"giftcard" => "بطاقة هدية",
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
return [
|
return [
|
||||||
"all" => "hamısı",
|
'all' => "hamısı",
|
||||||
"columns" => "Sütunlar",
|
'columns' => "Sütunlar",
|
||||||
"hide_show_pagination" => "Gizlət/Göstər səhifənin nömrələnməsin",
|
'hide_show_pagination' => "Gizlət/Göstər səhifənin nömrələnməsin",
|
||||||
"loading" => "Lütfən gözləyin, səhifə yüklənir...",
|
'loading' => "Lütfən gözləyin, səhifə yüklənir",
|
||||||
"page_from_to" => "Göstər {0} bundan {1} buna {2} kimi",
|
'page_from_to' => "Göstər {0} bundan {1} buna {2} kimi",
|
||||||
"refresh" => "Yenilə",
|
'refresh' => "Yenilə",
|
||||||
"rows_per_page" => "{0} yazı səhifədə",
|
'rows_per_page' => "{0} yazı səhifədə",
|
||||||
"toggle" => "Keçid",
|
'toggle' => "Keçid",
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -282,6 +282,7 @@ return [
|
|||||||
"right" => "Konfiqurasiya ugursuz oldu saxlanilmadi",
|
"right" => "Konfiqurasiya ugursuz oldu saxlanilmadi",
|
||||||
"sales_invoice_format" => "Satış Fatura Formatı",
|
"sales_invoice_format" => "Satış Fatura Formatı",
|
||||||
"sales_quote_format" => "Satış Sitat Formati",
|
"sales_quote_format" => "Satış Sitat Formati",
|
||||||
|
"mailpath_invalid" => "",
|
||||||
"saved_successfully" => "Konfiqurasiya uğurla saxlanıldı.",
|
"saved_successfully" => "Konfiqurasiya uğurla saxlanıldı.",
|
||||||
"saved_unsuccessfully" => "Konfiqurasiyanı saxlamq mümkün olmadı.",
|
"saved_unsuccessfully" => "Konfiqurasiyanı saxlamq mümkün olmadı.",
|
||||||
"security_issue" => "Təhlükəsizlik açığı xəbərdarlığı",
|
"security_issue" => "Təhlükəsizlik açığı xəbərdarlığı",
|
||||||
|
|||||||
@@ -9,6 +9,15 @@ return [
|
|||||||
"login" => "Giriş",
|
"login" => "Giriş",
|
||||||
"logout" => "Çıxış",
|
"logout" => "Çıxış",
|
||||||
"migration_needed" => "{0} -ə daxil olandan sonra verilənlər bazası miqrasiyası başlayacaq.",
|
"migration_needed" => "{0} -ə daxil olandan sonra verilənlər bazası miqrasiyası başlayacaq.",
|
||||||
|
"migration_required" => "",
|
||||||
|
"migration_auth_message" => "",
|
||||||
|
"migration_initializing" => "",
|
||||||
|
"migration_running" => "",
|
||||||
|
"migration_complete" => "",
|
||||||
|
"migration_complete_login" => "",
|
||||||
|
"migration_failed" => "",
|
||||||
|
"migration_error_connection" => "",
|
||||||
|
"migration_complete_redirect" => "",
|
||||||
"password" => "Şifrə",
|
"password" => "Şifrə",
|
||||||
"required_username" => "",
|
"required_username" => "",
|
||||||
"username" => "İstifadəçi",
|
"username" => "İstifadəçi",
|
||||||
|
|||||||
@@ -73,6 +73,12 @@ return [
|
|||||||
"employee" => "Əməkdaş",
|
"employee" => "Əməkdaş",
|
||||||
"entry" => "Daxil",
|
"entry" => "Daxil",
|
||||||
"error_editing_item" => "XƏTA Malın redaktəsində",
|
"error_editing_item" => "XƏTA Malın redaktəsində",
|
||||||
|
"negative_price_invalid" => "",
|
||||||
|
"negative_quantity_invalid" => "",
|
||||||
|
"negative_discount_invalid" => "",
|
||||||
|
"discount_percent_exceeds_100" => "",
|
||||||
|
"discount_exceeds_item_total" => "",
|
||||||
|
"negative_total_invalid" => "",
|
||||||
"find_or_scan_item" => "Malın axtarışı",
|
"find_or_scan_item" => "Malın axtarışı",
|
||||||
"find_or_scan_item_or_receipt" => "Tapmaq skan etmək və ya kvitansiya",
|
"find_or_scan_item_or_receipt" => "Tapmaq skan etmək və ya kvitansiya",
|
||||||
"giftcard" => "Hədiyyə Kartı",
|
"giftcard" => "Hədiyyə Kartı",
|
||||||
|
|||||||
@@ -1,12 +1,12 @@
|
|||||||
<?php
|
<?php
|
||||||
|
|
||||||
return [
|
return [
|
||||||
"all" => "Всичко/и",
|
'all' => "Всичко/и",
|
||||||
"columns" => "Колони",
|
'columns' => "Колони",
|
||||||
"hide_show_pagination" => "Скриване / Показване на страници",
|
'hide_show_pagination' => "Скриване / Показване на страници",
|
||||||
"loading" => "Зареждане, моля изчакайте...",
|
'loading' => "Зареждане, моля изчакайте",
|
||||||
"page_from_to" => "Показани са {0} до {1} от {2} реда",
|
'page_from_to' => "Показани са {0} до {1} от {2} реда",
|
||||||
"refresh" => "Опресняване",
|
'refresh' => "Опресняване",
|
||||||
"rows_per_page" => "{0} редове на страница",
|
'rows_per_page' => "{0} редове на страница",
|
||||||
"toggle" => "Щифт",
|
'toggle' => "Щифт",
|
||||||
];
|
];
|
||||||
|
|||||||
@@ -282,6 +282,7 @@ return [
|
|||||||
"right" => "Right",
|
"right" => "Right",
|
||||||
"sales_invoice_format" => "Sales Invoice Format",
|
"sales_invoice_format" => "Sales Invoice Format",
|
||||||
"sales_quote_format" => "Sales Quote Format",
|
"sales_quote_format" => "Sales Quote Format",
|
||||||
|
"mailpath_invalid" => "",
|
||||||
"saved_successfully" => "Configuration save successful.",
|
"saved_successfully" => "Configuration save successful.",
|
||||||
"saved_unsuccessfully" => "Configuration save failed.",
|
"saved_unsuccessfully" => "Configuration save failed.",
|
||||||
"security_issue" => "Security Vulnerability Warning",
|
"security_issue" => "Security Vulnerability Warning",
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user