Merge branch 'develop' into chore/migrate-select-helper
# Conflicts: # resources/views/settings/branding.blade.php # resources/views/settings/general.blade.php
This commit is contained in:
commit
2a156776a4
1109 changed files with 13883 additions and 6017 deletions
|
@ -3271,6 +3271,33 @@
|
||||||
"contributions": [
|
"contributions": [
|
||||||
"code"
|
"code"
|
||||||
]
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "addex12",
|
||||||
|
"name": "Adugna Gizaw",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/18550946?v=4",
|
||||||
|
"profile": "https://orbalia.pythonanywhere.com/",
|
||||||
|
"contributions": [
|
||||||
|
"translation"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "jostrander",
|
||||||
|
"name": "Jesse Ostrander",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/760989?v=4",
|
||||||
|
"profile": "https://github.com/jostrander",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"login": "azmcnutt",
|
||||||
|
"name": "James M",
|
||||||
|
"avatar_url": "https://avatars.githubusercontent.com/u/31522486?v=4",
|
||||||
|
"profile": "https://github.com/azmcnutt",
|
||||||
|
"contributions": [
|
||||||
|
"code"
|
||||||
|
]
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,7 +16,7 @@ APP_DEBUG=true
|
||||||
APP_KEY=base64:3ilviXqB9u6DX1NRcyWGJ+sjySF+H18CPDGb3+IVwMQ=
|
APP_KEY=base64:3ilviXqB9u6DX1NRcyWGJ+sjySF+H18CPDGb3+IVwMQ=
|
||||||
APP_URL=http://localhost:8000
|
APP_URL=http://localhost:8000
|
||||||
APP_TIMEZONE='UTC'
|
APP_TIMEZONE='UTC'
|
||||||
APP_LOCALE=en
|
APP_LOCALE=en-US
|
||||||
MAX_RESULTS=500
|
MAX_RESULTS=500
|
||||||
|
|
||||||
# --------------------------------------------
|
# --------------------------------------------
|
||||||
|
|
|
@ -99,7 +99,7 @@ PASSPORT_COOKIE_NAME='snipeit_passport_token'
|
||||||
COOKIE_DOMAIN=null
|
COOKIE_DOMAIN=null
|
||||||
SECURE_COOKIES=false
|
SECURE_COOKIES=false
|
||||||
API_TOKEN_EXPIRATION_YEARS=15
|
API_TOKEN_EXPIRATION_YEARS=15
|
||||||
BS_TABLE_STORAGE=cookieStorage
|
BS_TABLE_STORAGE=localStorage
|
||||||
BS_TABLE_DEEPLINK=true
|
BS_TABLE_DEEPLINK=true
|
||||||
|
|
||||||
# --------------------------------------------
|
# --------------------------------------------
|
||||||
|
|
4
.github/workflows/tests-mysql.yml
vendored
4
.github/workflows/tests-mysql.yml
vendored
|
@ -25,9 +25,9 @@ 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"
|
||||||
|
|
||||||
name: PHP ${{ matrix.php-version }}
|
name: PHP ${{ matrix.php-version }}
|
||||||
|
|
||||||
|
@ -67,7 +67,7 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
php artisan key:generate
|
php artisan key:generate
|
||||||
php artisan migrate --force
|
php artisan migrate --force
|
||||||
php artisan passport:install
|
php artisan passport:install --no-interaction
|
||||||
chmod -R 777 storage bootstrap/cache
|
chmod -R 777 storage bootstrap/cache
|
||||||
|
|
||||||
- name: Execute tests (Unit and Feature tests) via PHPUnit
|
- name: Execute tests (Unit and Feature tests) via PHPUnit
|
||||||
|
|
5
.github/workflows/tests-postgres.yml
vendored
5
.github/workflows/tests-postgres.yml
vendored
|
@ -21,9 +21,10 @@ 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"
|
||||||
|
|
||||||
|
|
||||||
name: PHP ${{ matrix.php-version }}
|
name: PHP ${{ matrix.php-version }}
|
||||||
|
|
||||||
|
@ -64,7 +65,7 @@ jobs:
|
||||||
run: |
|
run: |
|
||||||
php artisan key:generate
|
php artisan key:generate
|
||||||
php artisan migrate --force
|
php artisan migrate --force
|
||||||
php artisan passport:install
|
php artisan passport:install --no-interaction
|
||||||
chmod -R 777 storage bootstrap/cache
|
chmod -R 777 storage bootstrap/cache
|
||||||
|
|
||||||
- name: Execute tests (Unit and Feature tests) via PHPUnit
|
- name: Execute tests (Unit and Feature tests) via PHPUnit
|
||||||
|
|
2
.github/workflows/tests-sqlite.yml
vendored
2
.github/workflows/tests-sqlite.yml
vendored
|
@ -15,7 +15,7 @@ jobs:
|
||||||
fail-fast: false
|
fail-fast: false
|
||||||
matrix:
|
matrix:
|
||||||
php-version:
|
php-version:
|
||||||
- "8.1.1"
|
- "8.3"
|
||||||
|
|
||||||
name: PHP ${{ matrix.php-version }}
|
name: PHP ${{ matrix.php-version }}
|
||||||
|
|
||||||
|
|
|
@ -3,8 +3,8 @@
|
||||||
"DOC2": "In other words, what you see locally are the requirements for your _current_ install",
|
"DOC2": "In other words, what you see locally are the requirements for your _current_ install",
|
||||||
"DOC3": "Please don't rely on these versions for planning upgrades unless you've fetched the most recent version",
|
"DOC3": "Please don't rely on these versions for planning upgrades unless you've fetched the most recent version",
|
||||||
"DOC4": "You should really just ignore it and run upgrade.php. Really",
|
"DOC4": "You should really just ignore it and run upgrade.php. Really",
|
||||||
"php_min_version": "8.1.0",
|
"php_min_version": "8.2.0",
|
||||||
"php_max_major_minor": "8.3",
|
"php_max_major_minor": "8.4",
|
||||||
"php_max_wontwork": "8.4.0",
|
"php_max_wontwork": "8.5.0",
|
||||||
"current_snipeit_version": "7.0"
|
"current_snipeit_version": "8.0"
|
||||||
}
|
}
|
||||||
|
|
|
@ -53,7 +53,7 @@ Thanks goes to all of these wonderful people ([emoji key](https://github.com/ken
|
||||||
| [<img src="https://avatars.githubusercontent.com/u/55590532?v=4" width="110px;"/><br /><sub>squintfox</sub>](https://github.com/squintfox)<br />[💻](https://github.com/snipe/snipe-it/commits?author=squintfox "Code") | [<img src="https://avatars.githubusercontent.com/u/1380084?v=4" width="110px;"/><br /><sub>Jeff Clay</sub>](https://github.com/jeffclay)<br />[💻](https://github.com/snipe/snipe-it/commits?author=jeffclay "Code") | [<img src="https://avatars.githubusercontent.com/u/52716446?v=4" width="110px;"/><br /><sub>Phil J R</sub>](https://github.com/PP-JN-RL)<br />[💻](https://github.com/snipe/snipe-it/commits?author=PP-JN-RL "Code") | [<img src="https://avatars.githubusercontent.com/u/1496725?v=4" width="110px;"/><br /><sub>i_virus</sub>](https://www.corelight.com/)<br />[💻](https://github.com/snipe/snipe-it/commits?author=chandanchowdhury "Code") | [<img src="https://avatars.githubusercontent.com/u/1020541?v=4" width="110px;"/><br /><sub>Paul Grime</sub>](https://github.com/gitgrimbo)<br />[💻](https://github.com/snipe/snipe-it/commits?author=gitgrimbo "Code") | [<img src="https://avatars.githubusercontent.com/u/922815?v=4" width="110px;"/><br /><sub>Lee Porte</sub>](https://leeporte.co.uk)<br />[💻](https://github.com/snipe/snipe-it/commits?author=LeePorte "Code") | [<img src="https://avatars.githubusercontent.com/u/23613427?v=4" width="110px;"/><br /><sub>BRYAN </sub>](https://github.com/bryanlopezinc)<br />[💻](https://github.com/snipe/snipe-it/commits?author=bryanlopezinc "Code") [⚠️](https://github.com/snipe/snipe-it/commits?author=bryanlopezinc "Tests") |
|
| [<img src="https://avatars.githubusercontent.com/u/55590532?v=4" width="110px;"/><br /><sub>squintfox</sub>](https://github.com/squintfox)<br />[💻](https://github.com/snipe/snipe-it/commits?author=squintfox "Code") | [<img src="https://avatars.githubusercontent.com/u/1380084?v=4" width="110px;"/><br /><sub>Jeff Clay</sub>](https://github.com/jeffclay)<br />[💻](https://github.com/snipe/snipe-it/commits?author=jeffclay "Code") | [<img src="https://avatars.githubusercontent.com/u/52716446?v=4" width="110px;"/><br /><sub>Phil J R</sub>](https://github.com/PP-JN-RL)<br />[💻](https://github.com/snipe/snipe-it/commits?author=PP-JN-RL "Code") | [<img src="https://avatars.githubusercontent.com/u/1496725?v=4" width="110px;"/><br /><sub>i_virus</sub>](https://www.corelight.com/)<br />[💻](https://github.com/snipe/snipe-it/commits?author=chandanchowdhury "Code") | [<img src="https://avatars.githubusercontent.com/u/1020541?v=4" width="110px;"/><br /><sub>Paul Grime</sub>](https://github.com/gitgrimbo)<br />[💻](https://github.com/snipe/snipe-it/commits?author=gitgrimbo "Code") | [<img src="https://avatars.githubusercontent.com/u/922815?v=4" width="110px;"/><br /><sub>Lee Porte</sub>](https://leeporte.co.uk)<br />[💻](https://github.com/snipe/snipe-it/commits?author=LeePorte "Code") | [<img src="https://avatars.githubusercontent.com/u/23613427?v=4" width="110px;"/><br /><sub>BRYAN </sub>](https://github.com/bryanlopezinc)<br />[💻](https://github.com/snipe/snipe-it/commits?author=bryanlopezinc "Code") [⚠️](https://github.com/snipe/snipe-it/commits?author=bryanlopezinc "Tests") |
|
||||||
| [<img src="https://avatars.githubusercontent.com/u/64061710?v=4" width="110px;"/><br /><sub>U-H-T</sub>](https://github.com/U-H-T)<br />[💻](https://github.com/snipe/snipe-it/commits?author=U-H-T "Code") | [<img src="https://avatars.githubusercontent.com/u/5395363?v=4" width="110px;"/><br /><sub>Matt Tyree</sub>](https://github.com/Tyree)<br />[📖](https://github.com/snipe/snipe-it/commits?author=Tyree "Documentation") | [<img src="https://avatars.githubusercontent.com/u/292081?v=4" width="110px;"/><br /><sub>Florent Bervas</sub>](http://spoontux.net)<br />[💻](https://github.com/snipe/snipe-it/commits?author=FlorentDotMe "Code") | [<img src="https://avatars.githubusercontent.com/u/4498077?v=4" width="110px;"/><br /><sub>Daniel Albertsen</sub>](https://ditscheri.com)<br />[💻](https://github.com/snipe/snipe-it/commits?author=dbakan "Code") | [<img src="https://avatars.githubusercontent.com/u/100710244?v=4" width="110px;"/><br /><sub>r-xyz</sub>](https://github.com/r-xyz)<br />[💻](https://github.com/snipe/snipe-it/commits?author=r-xyz "Code") | [<img src="https://avatars.githubusercontent.com/u/47491036?v=4" width="110px;"/><br /><sub>Steven Mainor</sub>](https://github.com/DrekiDegga)<br />[💻](https://github.com/snipe/snipe-it/commits?author=DrekiDegga "Code") | [<img src="https://avatars.githubusercontent.com/u/65785975?v=4" width="110px;"/><br /><sub>arne-kroeger</sub>](https://github.com/arne-kroeger)<br />[💻](https://github.com/snipe/snipe-it/commits?author=arne-kroeger "Code") |
|
| [<img src="https://avatars.githubusercontent.com/u/64061710?v=4" width="110px;"/><br /><sub>U-H-T</sub>](https://github.com/U-H-T)<br />[💻](https://github.com/snipe/snipe-it/commits?author=U-H-T "Code") | [<img src="https://avatars.githubusercontent.com/u/5395363?v=4" width="110px;"/><br /><sub>Matt Tyree</sub>](https://github.com/Tyree)<br />[📖](https://github.com/snipe/snipe-it/commits?author=Tyree "Documentation") | [<img src="https://avatars.githubusercontent.com/u/292081?v=4" width="110px;"/><br /><sub>Florent Bervas</sub>](http://spoontux.net)<br />[💻](https://github.com/snipe/snipe-it/commits?author=FlorentDotMe "Code") | [<img src="https://avatars.githubusercontent.com/u/4498077?v=4" width="110px;"/><br /><sub>Daniel Albertsen</sub>](https://ditscheri.com)<br />[💻](https://github.com/snipe/snipe-it/commits?author=dbakan "Code") | [<img src="https://avatars.githubusercontent.com/u/100710244?v=4" width="110px;"/><br /><sub>r-xyz</sub>](https://github.com/r-xyz)<br />[💻](https://github.com/snipe/snipe-it/commits?author=r-xyz "Code") | [<img src="https://avatars.githubusercontent.com/u/47491036?v=4" width="110px;"/><br /><sub>Steven Mainor</sub>](https://github.com/DrekiDegga)<br />[💻](https://github.com/snipe/snipe-it/commits?author=DrekiDegga "Code") | [<img src="https://avatars.githubusercontent.com/u/65785975?v=4" width="110px;"/><br /><sub>arne-kroeger</sub>](https://github.com/arne-kroeger)<br />[💻](https://github.com/snipe/snipe-it/commits?author=arne-kroeger "Code") |
|
||||||
| [<img src="https://avatars.githubusercontent.com/u/167117705?v=4" width="110px;"/><br /><sub>Glukose1</sub>](https://github.com/Glukose1)<br />[💻](https://github.com/snipe/snipe-it/commits?author=Glukose1 "Code") | [<img src="https://avatars.githubusercontent.com/u/1197791?v=4" width="110px;"/><br /><sub>Scarzy</sub>](https://github.com/Scarzy)<br />[💻](https://github.com/snipe/snipe-it/commits?author=Scarzy "Code") | [<img src="https://avatars.githubusercontent.com/u/37372069?v=4" width="110px;"/><br /><sub>setpill</sub>](https://github.com/setpill)<br />[💻](https://github.com/snipe/snipe-it/commits?author=setpill "Code") | [<img src="https://avatars.githubusercontent.com/u/3755203?v=4" width="110px;"/><br /><sub>swift2512</sub>](https://github.com/swift2512)<br />[🐛](https://github.com/snipe/snipe-it/issues?q=author%3Aswift2512 "Bug reports") | [<img src="https://avatars.githubusercontent.com/u/6136439?v=4" width="110px;"/><br /><sub>Darren Rainey</sub>](https://darrenraineys.co.uk)<br />[💻](https://github.com/snipe/snipe-it/commits?author=DarrenRainey "Code") | [<img src="https://avatars.githubusercontent.com/u/133033121?v=4" width="110px;"/><br /><sub>maciej-poleszczyk</sub>](https://github.com/maciej-poleszczyk)<br />[💻](https://github.com/snipe/snipe-it/commits?author=maciej-poleszczyk "Code") | [<img src="https://avatars.githubusercontent.com/u/143394709?v=4" width="110px;"/><br /><sub>Sebastian Groß</sub>](https://github.com/sgross-emlix)<br />[💻](https://github.com/snipe/snipe-it/commits?author=sgross-emlix "Code") |
|
| [<img src="https://avatars.githubusercontent.com/u/167117705?v=4" width="110px;"/><br /><sub>Glukose1</sub>](https://github.com/Glukose1)<br />[💻](https://github.com/snipe/snipe-it/commits?author=Glukose1 "Code") | [<img src="https://avatars.githubusercontent.com/u/1197791?v=4" width="110px;"/><br /><sub>Scarzy</sub>](https://github.com/Scarzy)<br />[💻](https://github.com/snipe/snipe-it/commits?author=Scarzy "Code") | [<img src="https://avatars.githubusercontent.com/u/37372069?v=4" width="110px;"/><br /><sub>setpill</sub>](https://github.com/setpill)<br />[💻](https://github.com/snipe/snipe-it/commits?author=setpill "Code") | [<img src="https://avatars.githubusercontent.com/u/3755203?v=4" width="110px;"/><br /><sub>swift2512</sub>](https://github.com/swift2512)<br />[🐛](https://github.com/snipe/snipe-it/issues?q=author%3Aswift2512 "Bug reports") | [<img src="https://avatars.githubusercontent.com/u/6136439?v=4" width="110px;"/><br /><sub>Darren Rainey</sub>](https://darrenraineys.co.uk)<br />[💻](https://github.com/snipe/snipe-it/commits?author=DarrenRainey "Code") | [<img src="https://avatars.githubusercontent.com/u/133033121?v=4" width="110px;"/><br /><sub>maciej-poleszczyk</sub>](https://github.com/maciej-poleszczyk)<br />[💻](https://github.com/snipe/snipe-it/commits?author=maciej-poleszczyk "Code") | [<img src="https://avatars.githubusercontent.com/u/143394709?v=4" width="110px;"/><br /><sub>Sebastian Groß</sub>](https://github.com/sgross-emlix)<br />[💻](https://github.com/snipe/snipe-it/commits?author=sgross-emlix "Code") |
|
||||||
| [<img src="https://avatars.githubusercontent.com/u/41107778?v=4" width="110px;"/><br /><sub>Anouar Touati</sub>](https://github.com/AnouarTouati)<br />[💻](https://github.com/snipe/snipe-it/commits?author=AnouarTouati "Code") | [<img src="https://avatars.githubusercontent.com/u/25596663?v=4" width="110px;"/><br /><sub>aHVzY2g</sub>](https://github.com/aHVzY2g)<br />[💻](https://github.com/snipe/snipe-it/commits?author=aHVzY2g "Code") | [<img src="https://avatars.githubusercontent.com/u/13408130?v=4" width="110px;"/><br /><sub>林博仁 Buo-ren Lin</sub>](https://brlin.me)<br />[💻](https://github.com/snipe/snipe-it/commits?author=brlin-tw "Code") |
|
| [<img src="https://avatars.githubusercontent.com/u/41107778?v=4" width="110px;"/><br /><sub>Anouar Touati</sub>](https://github.com/AnouarTouati)<br />[💻](https://github.com/snipe/snipe-it/commits?author=AnouarTouati "Code") | [<img src="https://avatars.githubusercontent.com/u/25596663?v=4" width="110px;"/><br /><sub>aHVzY2g</sub>](https://github.com/aHVzY2g)<br />[💻](https://github.com/snipe/snipe-it/commits?author=aHVzY2g "Code") | [<img src="https://avatars.githubusercontent.com/u/13408130?v=4" width="110px;"/><br /><sub>林博仁 Buo-ren Lin</sub>](https://brlin.me)<br />[💻](https://github.com/snipe/snipe-it/commits?author=brlin-tw "Code") | [<img src="https://avatars.githubusercontent.com/u/18550946?v=4" width="110px;"/><br /><sub>Adugna Gizaw</sub>](https://orbalia.pythonanywhere.com/)<br />[🌍](#translation-addex12 "Translation") | [<img src="https://avatars.githubusercontent.com/u/760989?v=4" width="110px;"/><br /><sub>Jesse Ostrander</sub>](https://github.com/jostrander)<br />[💻](https://github.com/snipe/snipe-it/commits?author=jostrander "Code") | [<img src="https://avatars.githubusercontent.com/u/31522486?v=4" width="110px;"/><br /><sub>James M</sub>](https://github.com/azmcnutt)<br />[💻](https://github.com/snipe/snipe-it/commits?author=azmcnutt "Code") |
|
||||||
<!-- ALL-CONTRIBUTORS-LIST:END -->
|
<!-- ALL-CONTRIBUTORS-LIST:END -->
|
||||||
|
|
||||||
This project follows the [all-contributors](https://github.com/kentcdodds/all-contributors) specification. Contributions of any kind welcome!
|
This project follows the [all-contributors](https://github.com/kentcdodds/all-contributors) specification. Contributions of any kind welcome!
|
||||||
|
|
36
Dockerfile
36
Dockerfile
|
@ -1,4 +1,4 @@
|
||||||
FROM ubuntu:22.04
|
FROM ubuntu:24.04
|
||||||
LABEL maintainer="Brady Wetherington <bwetherington@grokability.com>"
|
LABEL maintainer="Brady Wetherington <bwetherington@grokability.com>"
|
||||||
|
|
||||||
# No need to add `apt-get clean` here, reference:
|
# No need to add `apt-get clean` here, reference:
|
||||||
|
@ -14,16 +14,16 @@ RUN export DEBIAN_FRONTEND=noninteractive; \
|
||||||
apt-utils \
|
apt-utils \
|
||||||
apache2 \
|
apache2 \
|
||||||
apache2-bin \
|
apache2-bin \
|
||||||
libapache2-mod-php8.1 \
|
libapache2-mod-php8.3 \
|
||||||
php8.1-curl \
|
php8.3-curl \
|
||||||
php8.1-ldap \
|
php8.3-ldap \
|
||||||
php8.1-mysql \
|
php8.3-mysql \
|
||||||
php8.1-gd \
|
php8.3-gd \
|
||||||
php8.1-xml \
|
php8.3-xml \
|
||||||
php8.1-mbstring \
|
php8.3-mbstring \
|
||||||
php8.1-zip \
|
php8.3-zip \
|
||||||
php8.1-bcmath \
|
php8.3-bcmath \
|
||||||
php8.1-redis \
|
php8.3-redis \
|
||||||
php-memcached \
|
php-memcached \
|
||||||
patch \
|
patch \
|
||||||
curl \
|
curl \
|
||||||
|
@ -40,8 +40,7 @@ autoconf \
|
||||||
libc-dev \
|
libc-dev \
|
||||||
libldap-common \
|
libldap-common \
|
||||||
pkg-config \
|
pkg-config \
|
||||||
libmcrypt-dev \
|
php8.3-dev \
|
||||||
php8.1-dev \
|
|
||||||
ca-certificates \
|
ca-certificates \
|
||||||
unzip \
|
unzip \
|
||||||
dnsutils \
|
dnsutils \
|
||||||
|
@ -51,18 +50,13 @@ dnsutils \
|
||||||
RUN curl -L -O https://github.com/pear/pearweb_phars/raw/master/go-pear.phar
|
RUN curl -L -O https://github.com/pear/pearweb_phars/raw/master/go-pear.phar
|
||||||
RUN php go-pear.phar
|
RUN php go-pear.phar
|
||||||
|
|
||||||
RUN pecl install mcrypt
|
|
||||||
|
|
||||||
RUN bash -c "echo extension=/usr/lib/php/20210902/mcrypt.so > /etc/php/8.1/mods-available/mcrypt.ini"
|
|
||||||
|
|
||||||
RUN phpenmod mcrypt
|
|
||||||
RUN phpenmod gd
|
RUN phpenmod gd
|
||||||
RUN phpenmod bcmath
|
RUN phpenmod bcmath
|
||||||
|
|
||||||
RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.1/apache2/php.ini
|
RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.3/apache2/php.ini
|
||||||
RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.1/cli/php.ini
|
RUN sed -i 's/variables_order = .*/variables_order = "EGPCS"/' /etc/php/8.3/cli/php.ini
|
||||||
|
|
||||||
RUN useradd -m --uid 1000 --gid 50 docker
|
RUN useradd -m --uid 10000 --gid 50 docker
|
||||||
|
|
||||||
RUN echo export APACHE_RUN_USER=docker >> /etc/apache2/envvars
|
RUN echo export APACHE_RUN_USER=docker >> /etc/apache2/envvars
|
||||||
RUN echo export APACHE_RUN_GROUP=staff >> /etc/apache2/envvars
|
RUN echo export APACHE_RUN_GROUP=staff >> /etc/apache2/envvars
|
||||||
|
|
|
@ -7,7 +7,7 @@
|
||||||
|
|
||||||
This is a FOSS project for asset management in IT Operations. Knowing who has which laptop, when it was purchased in order to depreciate it correctly, handling software licenses, etc.
|
This is a FOSS project for asset management in IT Operations. Knowing who has which laptop, when it was purchased in order to depreciate it correctly, handling software licenses, etc.
|
||||||
|
|
||||||
It is built on [Laravel 10](http://laravel.com).
|
It is built on [Laravel 11](http://laravel.com).
|
||||||
|
|
||||||
Snipe-IT is actively developed and we [release quite frequently](https://github.com/snipe/snipe-it/releases). ([Check out the live demo here](https://snipeitapp.com/demo/).)
|
Snipe-IT is actively developed and we [release quite frequently](https://github.com/snipe/snipe-it/releases). ([Check out the live demo here](https://snipeitapp.com/demo/).)
|
||||||
|
|
||||||
|
@ -94,6 +94,8 @@ Since the release of the JSON REST API, several third-party developers have been
|
||||||
- [SnipeAgent](https://github.com/ReticentRobot/SnipeAgent) by [@ReticentRobot](https://github.com/ReticentRobot) - Windows agent for Snipe-IT.
|
- [SnipeAgent](https://github.com/ReticentRobot/SnipeAgent) by [@ReticentRobot](https://github.com/ReticentRobot) - Windows agent for Snipe-IT.
|
||||||
- [Gate Pass Generator](https://github.com/cha7uraAE/snipe-it-gate-pass-system) by [@cha7uraAE](https://github.com/cha7uraAE) - A Streamlit application for generating gate passes based on hardware data from a Snipe-IT API.
|
- [Gate Pass Generator](https://github.com/cha7uraAE/snipe-it-gate-pass-system) by [@cha7uraAE](https://github.com/cha7uraAE) - A Streamlit application for generating gate passes based on hardware data from a Snipe-IT API.
|
||||||
|
|
||||||
|
We also have a handful of [Google Apps scripts](https://github.com/grokability/google-apps-scripts-for-snipe-it) to help with various tasks.
|
||||||
|
|
||||||
-----
|
-----
|
||||||
|
|
||||||
### Join the Community!
|
### Join the Community!
|
||||||
|
|
|
@ -125,6 +125,10 @@ class LdapSync extends Command
|
||||||
*/
|
*/
|
||||||
$attributes = array_values(array_filter($ldap_map));
|
$attributes = array_values(array_filter($ldap_map));
|
||||||
|
|
||||||
|
if (Setting::getSettings()->is_ad === 1 && is_null($ldap_map['active_flag'])) {
|
||||||
|
$attributes[] = 'useraccountcontrol';
|
||||||
|
}
|
||||||
|
|
||||||
$results = Ldap::findLdapUsers($search_base, -1, $filter, $attributes);
|
$results = Ldap::findLdapUsers($search_base, -1, $filter, $attributes);
|
||||||
|
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
|
@ -357,9 +361,15 @@ class LdapSync extends Command
|
||||||
// (Specifically, we don't handle a value of '0.0' correctly)
|
// (Specifically, we don't handle a value of '0.0' correctly)
|
||||||
$raw_value = @$results[$i][$ldap_map["active_flag"]][0];
|
$raw_value = @$results[$i][$ldap_map["active_flag"]][0];
|
||||||
$filter_var = filter_var($raw_value, FILTER_VALIDATE_BOOLEAN, FILTER_NULL_ON_FAILURE);
|
$filter_var = filter_var($raw_value, FILTER_VALIDATE_BOOLEAN, FILTER_NULL_ON_FAILURE);
|
||||||
|
|
||||||
$boolean_cast = (bool) $raw_value;
|
$boolean_cast = (bool) $raw_value;
|
||||||
|
|
||||||
|
if (Setting::getSettings()->ldap_invert_active_flag === 1) {
|
||||||
|
// Because ldap_active_flag is set, if filter_var is true or boolean_cast is true, then user is suspended
|
||||||
|
$user->activated = !($filter_var ?? $boolean_cast);
|
||||||
|
}else{
|
||||||
$user->activated = $filter_var ?? $boolean_cast; // if filter_var() was true or false, use that. If it's null, use the $boolean_cast
|
$user->activated = $filter_var ?? $boolean_cast; // if filter_var() was true or false, use that. If it's null, use the $boolean_cast
|
||||||
|
}
|
||||||
|
|
||||||
} elseif (array_key_exists('useraccountcontrol', $results[$i])) {
|
} elseif (array_key_exists('useraccountcontrol', $results[$i])) {
|
||||||
// ....otherwise, (ie if no 'active' LDAP flag is defined), IF the UAC setting exists,
|
// ....otherwise, (ie if no 'active' LDAP flag is defined), IF the UAC setting exists,
|
||||||
|
@ -424,8 +434,12 @@ class LdapSync extends Command
|
||||||
$item['note'] = $item['createorupdate'];
|
$item['note'] = $item['createorupdate'];
|
||||||
$item['status'] = 'success';
|
$item['status'] = 'success';
|
||||||
if ($item['createorupdate'] === 'created' && $ldap_default_group) {
|
if ($item['createorupdate'] === 'created' && $ldap_default_group) {
|
||||||
|
// Check if the relationship already exists
|
||||||
|
if (!$user->groups()->where('group_id', $ldap_default_group)->exists()) {
|
||||||
$user->groups()->attach($ldap_default_group);
|
$user->groups()->attach($ldap_default_group);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//updates assets location based on user's location
|
//updates assets location based on user's location
|
||||||
if ($user->wasChanged('location_id')) {
|
if ($user->wasChanged('location_id')) {
|
||||||
foreach ($user->assets as $asset) {
|
foreach ($user->assets as $asset) {
|
||||||
|
|
|
@ -1,157 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Console\Commands;
|
|
||||||
|
|
||||||
use App\LegacyEncrypter\McryptEncrypter;
|
|
||||||
use App\Models\Asset;
|
|
||||||
use App\Models\CustomField;
|
|
||||||
use App\Models\Setting;
|
|
||||||
use Illuminate\Console\Command;
|
|
||||||
use Illuminate\Support\Facades\Storage;
|
|
||||||
|
|
||||||
class RecryptFromMcrypt extends Command
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* The name and signature of the console command.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
protected $signature = 'snipeit:legacy-recrypt
|
|
||||||
{--force : Force a re-crypt of encrypted data from MCRYPT.}';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The console command description.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
protected $description = 'This command allows upgrading users to de-encrypt their deprecated mcrypt encrypted fields and re-encrypt them using the current OpenSSL encryption.';
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new command instance.
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function __construct()
|
|
||||||
{
|
|
||||||
parent::__construct();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Execute the console command.
|
|
||||||
*
|
|
||||||
* @return mixed
|
|
||||||
*/
|
|
||||||
public function handle()
|
|
||||||
{
|
|
||||||
|
|
||||||
// Check and see if they have a legacy app key listed in their .env
|
|
||||||
// If not, we can try to use the current APP_KEY if looks like it's old
|
|
||||||
$legacy_key = env('LEGACY_APP_KEY');
|
|
||||||
$key_parts = explode(':', $legacy_key);
|
|
||||||
$legacy_cipher = env('LEGACY_CIPHER', 'rijndael-256');
|
|
||||||
$errors = [];
|
|
||||||
|
|
||||||
if (! $legacy_key) {
|
|
||||||
$this->error('ERROR: You do not have a LEGACY_APP_KEY set in your .env file. Please locate your old APP_KEY and ADD a line to your .env file like: LEGACY_APP_KEY=YOUR_OLD_APP_KEY');
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Do some basic legacy app key length checks
|
|
||||||
if (strlen($legacy_key) == 32) {
|
|
||||||
$legacy_length_check = true;
|
|
||||||
} elseif (array_key_exists('1', $key_parts) && (strlen($key_parts[1]) == 44)) {
|
|
||||||
$legacy_key = base64_decode($key_parts[1], true);
|
|
||||||
$legacy_length_check = true;
|
|
||||||
} else {
|
|
||||||
$legacy_length_check = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check that the app key is 32 characters
|
|
||||||
if ($legacy_length_check === true) {
|
|
||||||
$this->comment('INFO: Your LEGACY_APP_KEY looks correct. Okay to continue.');
|
|
||||||
} else {
|
|
||||||
$this->error('ERROR: Your LEGACY_APP_KEY is not the correct length (32 characters or base64 followed by 44 characters for later versions). Please locate your old APP_KEY and use that as your LEGACY_APP_KEY in your .env file to continue.');
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->error('================================!!!! WARNING !!!!================================');
|
|
||||||
$this->error('================================!!!! WARNING !!!!================================');
|
|
||||||
$this->comment("This tool will attempt to decrypt your old Snipe-IT (mcrypt, now deprecated) encrypted data and re-encrypt it using OpenSSL. \n\nYou should only continue if you have backed up any and all old APP_KEYs and have backed up your data.");
|
|
||||||
|
|
||||||
$force = ($this->option('force')) ? true : false;
|
|
||||||
|
|
||||||
if ($force || ($this->confirm('Are you SURE you wish to continue?'))) {
|
|
||||||
$backup_file = 'backups/env-backups/'.'app_key-'.date('Y-m-d-gis');
|
|
||||||
|
|
||||||
try {
|
|
||||||
Storage::disk('local')->put($backup_file, 'APP_KEY: '.config('app.key'));
|
|
||||||
Storage::disk('local')->append($backup_file, 'LEGACY_APP_KEY: '.$legacy_key);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
$this->info('WARNING: Could not backup app keys');
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($legacy_cipher) {
|
|
||||||
$mcrypter = new McryptEncrypter($legacy_key, $legacy_cipher);
|
|
||||||
} else {
|
|
||||||
$mcrypter = new McryptEncrypter($legacy_key);
|
|
||||||
}
|
|
||||||
$settings = Setting::getSettings();
|
|
||||||
|
|
||||||
if ($settings->ldap_pword == '') {
|
|
||||||
$this->comment('INFO: No LDAP password found. Skipping... ');
|
|
||||||
} else {
|
|
||||||
$decrypted_ldap_pword = $mcrypter->decrypt($settings->ldap_pword);
|
|
||||||
$settings->ldap_pword = Crypt::encrypt($decrypted_ldap_pword);
|
|
||||||
$settings->save();
|
|
||||||
}
|
|
||||||
/** @var CustomField[] $custom_fields */
|
|
||||||
$custom_fields = CustomField::where('field_encrypted', '=', 1)->get();
|
|
||||||
$this->comment('INFO: Retrieving encrypted custom fields...');
|
|
||||||
|
|
||||||
$query = Asset::withTrashed();
|
|
||||||
|
|
||||||
foreach ($custom_fields as $custom_field) {
|
|
||||||
$this->comment('FIELD TO RECRYPT: '.$custom_field->name.' ('.$custom_field->db_column.')');
|
|
||||||
$query->orWhereNotNull($custom_field->db_column);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get all assets with a value in any of the fields that were encrypted
|
|
||||||
/** @var Asset[] $assets */
|
|
||||||
$assets = $query->get();
|
|
||||||
|
|
||||||
$bar = $this->output->createProgressBar(count($assets));
|
|
||||||
|
|
||||||
foreach ($assets as $asset) {
|
|
||||||
foreach ($custom_fields as $encrypted_field) {
|
|
||||||
$columnName = $encrypted_field->db_column;
|
|
||||||
|
|
||||||
// Make sure the value isn't null
|
|
||||||
if ($asset->{$columnName} != '') {
|
|
||||||
// Try to decrypt the payload using the legacy app key
|
|
||||||
try {
|
|
||||||
$decrypted_field = $mcrypter->decrypt($asset->{$columnName});
|
|
||||||
$asset->{$columnName} = Crypt::encrypt($decrypted_field);
|
|
||||||
$this->comment($decrypted_field);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
$errors[] = ' - ERROR: Could not decrypt field ['.$encrypted_field->name.']: '.$e->getMessage();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$asset->save();
|
|
||||||
$bar->advance();
|
|
||||||
}
|
|
||||||
|
|
||||||
$bar->finish();
|
|
||||||
|
|
||||||
if (count($errors) > 0) {
|
|
||||||
$this->comment("\n\n");
|
|
||||||
$this->error("The decrypter encountered some errors: \n");
|
|
||||||
foreach ($errors as $error) {
|
|
||||||
$this->error($error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -64,28 +64,43 @@ class SendAcceptanceReminder extends Command
|
||||||
->groupBy(function($item) {
|
->groupBy(function($item) {
|
||||||
return $item['acceptance']->assignedTo ? $item['acceptance']->assignedTo->id : '';
|
return $item['acceptance']->assignedTo ? $item['acceptance']->assignedTo->id : '';
|
||||||
});
|
});
|
||||||
|
$no_email_list= [];
|
||||||
|
|
||||||
foreach($unacceptedAssetGroups as $unacceptedAssetGroup) {
|
foreach($unacceptedAssetGroups as $unacceptedAssetGroup) {
|
||||||
// The [0] is weird, but it allows for the item_count to work and grabs the appropriate info for each user.
|
// The [0] is weird, but it allows for the item_count to work and grabs the appropriate info for each user.
|
||||||
// Collapsing and flattening the collection doesn't work above.
|
// Collapsing and flattening the collection doesn't work above.
|
||||||
$acceptance = $unacceptedAssetGroup[0]['acceptance'];
|
$acceptance = $unacceptedAssetGroup[0]['acceptance'];
|
||||||
|
|
||||||
$locale = $acceptance->assignedTo?->locale;
|
$locale = $acceptance->assignedTo?->locale;
|
||||||
$email = $acceptance->assignedTo?->email;
|
$email = $acceptance->assignedTo?->email;
|
||||||
|
|
||||||
if(!$email){
|
if(!$email){
|
||||||
$this->info($acceptance->assignedTo?->present()->fullName().' has no email address.');
|
$no_email_list[] = [
|
||||||
|
'id' => $acceptance->assignedTo?->id,
|
||||||
|
'name' => $acceptance->assignedTo?->present()->fullName(),
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
$count++;
|
||||||
}
|
}
|
||||||
$item_count = $unacceptedAssetGroup->count();
|
$item_count = $unacceptedAssetGroup->count();
|
||||||
|
|
||||||
if ($locale && $email) {
|
if ($locale && $email) {
|
||||||
Mail::to($email)->send((new UnacceptedAssetReminderMail($acceptance, $item_count))->locale($locale));
|
Mail::to($email)->send((new UnacceptedAssetReminderMail($acceptance, $item_count))->locale($locale));
|
||||||
|
|
||||||
} elseif ($email) {
|
} elseif ($email) {
|
||||||
Mail::to($email)->send((new UnacceptedAssetReminderMail($acceptance, $item_count)));
|
Mail::to($email)->send((new UnacceptedAssetReminderMail($acceptance, $item_count)));
|
||||||
}
|
}
|
||||||
$count++;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
$this->info($count.' users notified.');
|
$this->info($count.' users notified.');
|
||||||
|
$headers = ['ID', 'Name'];
|
||||||
|
$rows = [];
|
||||||
|
|
||||||
|
foreach ($no_email_list as $user) {
|
||||||
|
$rows[] = [$user['id'], $user['name']];
|
||||||
|
}
|
||||||
|
$this->info("The following users do not have an email address:");
|
||||||
|
$this->table($headers, $rows);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,13 +2,13 @@
|
||||||
|
|
||||||
namespace App\Console\Commands;
|
namespace App\Console\Commands;
|
||||||
|
|
||||||
|
use App\Mail\ExpiringAssetsMail;
|
||||||
|
use App\Mail\ExpiringLicenseMail;
|
||||||
use App\Models\Asset;
|
use App\Models\Asset;
|
||||||
use App\Models\License;
|
use App\Models\License;
|
||||||
use App\Models\Recipients\AlertRecipient;
|
|
||||||
use App\Models\Setting;
|
use App\Models\Setting;
|
||||||
use App\Notifications\ExpiringAssetsNotification;
|
|
||||||
use App\Notifications\ExpiringLicenseNotification;
|
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Support\Facades\Mail;
|
||||||
|
|
||||||
class SendExpirationAlerts extends Command
|
class SendExpirationAlerts extends Command
|
||||||
{
|
{
|
||||||
|
@ -47,22 +47,22 @@ class SendExpirationAlerts extends Command
|
||||||
if (($settings->alert_email != '') && ($settings->alerts_enabled == 1)) {
|
if (($settings->alert_email != '') && ($settings->alerts_enabled == 1)) {
|
||||||
|
|
||||||
// Send a rollup to the admin, if settings dictate
|
// Send a rollup to the admin, if settings dictate
|
||||||
$recipients = collect(explode(',', $settings->alert_email))->map(function ($item, $key) {
|
$recipients = collect(explode(',', $settings->alert_email))
|
||||||
return new AlertRecipient($item);
|
->map(fn($item) => trim($item)) // Trim each email
|
||||||
});
|
->all();
|
||||||
|
|
||||||
// Expiring Assets
|
// Expiring Assets
|
||||||
$assets = Asset::getExpiringWarrantee($threshold);
|
$assets = Asset::getExpiringWarrantee($threshold);
|
||||||
|
|
||||||
if ($assets->count() > 0) {
|
if ($assets->count() > 0) {
|
||||||
$this->info(trans_choice('mail.assets_warrantee_alert', $assets->count(), ['count' => $assets->count(), 'threshold' => $threshold]));
|
$this->info(trans_choice('mail.assets_warrantee_alert', $assets->count(), ['count' => $assets->count(), 'threshold' => $threshold]));
|
||||||
\Notification::send($recipients, new ExpiringAssetsNotification($assets, $threshold));
|
Mail::to($recipients)->send(new ExpiringAssetsMail($assets, $threshold));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Expiring licenses
|
// Expiring licenses
|
||||||
$licenses = License::getExpiringLicenses($threshold);
|
$licenses = License::getExpiringLicenses($threshold);
|
||||||
if ($licenses->count() > 0) {
|
if ($licenses->count() > 0) {
|
||||||
$this->info(trans_choice('mail.license_expiring_alert', $licenses->count(), ['count' => $licenses->count(), 'threshold' => $threshold]));
|
$this->info(trans_choice('mail.license_expiring_alert', $licenses->count(), ['count' => $licenses->count(), 'threshold' => $threshold]));
|
||||||
\Notification::send($recipients, new ExpiringLicenseNotification($licenses, $threshold));
|
Mail::to($recipients)->send(new ExpiringLicenseMail($licenses, $threshold));
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if ($settings->alert_email == '') {
|
if ($settings->alert_email == '') {
|
||||||
|
|
|
@ -2,13 +2,12 @@
|
||||||
|
|
||||||
namespace App\Console\Commands;
|
namespace App\Console\Commands;
|
||||||
|
|
||||||
|
use App\Mail\SendUpcomingAuditMail;
|
||||||
use App\Models\Asset;
|
use App\Models\Asset;
|
||||||
use App\Models\Recipients\AlertRecipient;
|
|
||||||
use App\Models\Setting;
|
use App\Models\Setting;
|
||||||
use App\Notifications\SendUpcomingAuditNotification;
|
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Illuminate\Support\Facades\DB;
|
|
||||||
use Illuminate\Console\Command;
|
use Illuminate\Console\Command;
|
||||||
|
use Illuminate\Support\Facades\Mail;
|
||||||
|
|
||||||
class SendUpcomingAuditReport extends Command
|
class SendUpcomingAuditReport extends Command
|
||||||
{
|
{
|
||||||
|
@ -48,19 +47,19 @@ class SendUpcomingAuditReport extends Command
|
||||||
$today = Carbon::now();
|
$today = Carbon::now();
|
||||||
$interval_date = $today->copy()->addDays($interval);
|
$interval_date = $today->copy()->addDays($interval);
|
||||||
|
|
||||||
$assets = Asset::whereNull('deleted_at')->DueOrOverdueForAudit($settings)->orderBy('assets.next_audit_date', 'desc')->get();
|
$assets = Asset::whereNull('deleted_at')->dueOrOverdueForAudit($settings)->orderBy('assets.next_audit_date', 'desc')->get();
|
||||||
$this->info($assets->count().' assets must be audited in on or before '.$interval_date.' is deadline');
|
$this->info($assets->count() . ' assets must be audited in on or before ' . $interval_date . ' is deadline');
|
||||||
|
|
||||||
|
|
||||||
if (($assets) && ($assets->count() > 0) && ($settings->alert_email != '')) {
|
if ((count($assets) !== 0) && ($assets->count() > 0) && ($settings->alert_email != '')) {
|
||||||
// Send a rollup to the admin, if settings dictate
|
// Send a rollup to the admin, if settings dictate
|
||||||
$recipients = collect(explode(',', $settings->alert_email))->map(function ($item) {
|
$recipients = collect(explode(',', $settings->alert_email))
|
||||||
return new AlertRecipient($item);
|
->map(fn($item) => trim($item))
|
||||||
});
|
->all();
|
||||||
|
|
||||||
$this->info('Sending Admin SendUpcomingAuditNotification to: '.$settings->alert_email);
|
|
||||||
\Notification::send($recipients, new SendUpcomingAuditNotification($assets, $settings->audit_warning_days));
|
|
||||||
|
|
||||||
|
$this->info('Sending Admin SendUpcomingAuditNotification to: ' . $settings->alert_email);
|
||||||
|
Mail::to($recipients)->send(new SendUpcomingAuditMail($assets, $settings->audit_warning_days));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ namespace App\Console;
|
||||||
use App\Console\Commands\ImportLocations;
|
use App\Console\Commands\ImportLocations;
|
||||||
use App\Console\Commands\ReEncodeCustomFieldNames;
|
use App\Console\Commands\ReEncodeCustomFieldNames;
|
||||||
use App\Console\Commands\RestoreDeletedUsers;
|
use App\Console\Commands\RestoreDeletedUsers;
|
||||||
|
use App\Models\Setting;
|
||||||
use Illuminate\Console\Scheduling\Schedule;
|
use Illuminate\Console\Scheduling\Schedule;
|
||||||
use Illuminate\Foundation\Console\Kernel as ConsoleKernel;
|
use Illuminate\Foundation\Console\Kernel as ConsoleKernel;
|
||||||
|
|
||||||
|
@ -18,12 +19,14 @@ class Kernel extends ConsoleKernel
|
||||||
*/
|
*/
|
||||||
protected function schedule(Schedule $schedule)
|
protected function schedule(Schedule $schedule)
|
||||||
{
|
{
|
||||||
|
if(Setting::getSettings()->alerts_enabled === 1) {
|
||||||
$schedule->command('snipeit:inventory-alerts')->daily();
|
$schedule->command('snipeit:inventory-alerts')->daily();
|
||||||
$schedule->command('snipeit:expiring-alerts')->daily();
|
$schedule->command('snipeit:expiring-alerts')->daily();
|
||||||
$schedule->command('snipeit:expected-checkin')->daily();
|
$schedule->command('snipeit:expected-checkin')->daily();
|
||||||
|
$schedule->command('snipeit:upcoming-audits')->daily();
|
||||||
|
}
|
||||||
$schedule->command('snipeit:backup')->weekly();
|
$schedule->command('snipeit:backup')->weekly();
|
||||||
$schedule->command('backup:clean')->daily();
|
$schedule->command('backup:clean')->daily();
|
||||||
$schedule->command('snipeit:upcoming-audits')->daily();
|
|
||||||
$schedule->command('auth:clear-resets')->everyFifteenMinutes();
|
$schedule->command('auth:clear-resets')->everyFifteenMinutes();
|
||||||
$schedule->command('saml:clear_expired_nonces')->weekly();
|
$schedule->command('saml:clear_expired_nonces')->weekly();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,28 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Events;
|
|
||||||
|
|
||||||
use App\Models\User;
|
|
||||||
use Illuminate\Foundation\Events\Dispatchable;
|
|
||||||
use Illuminate\Queue\SerializesModels;
|
|
||||||
|
|
||||||
class NoteAdded
|
|
||||||
{
|
|
||||||
use Dispatchable, SerializesModels;
|
|
||||||
public $itemNoteAddedOn;
|
|
||||||
public $note;
|
|
||||||
public $noteAddedBy;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new event instance.
|
|
||||||
*
|
|
||||||
* @return void
|
|
||||||
*/
|
|
||||||
public function __construct($itemNoteAddedOn, User $noteAddedBy, $note)
|
|
||||||
{
|
|
||||||
$this->itemNoteAddedOn = $itemNoteAddedOn;
|
|
||||||
$this->note = $note;
|
|
||||||
$this->noteAddedBy = $noteAddedBy;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -122,6 +122,29 @@ class Handler extends ExceptionHandler
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// This is traaaaash but it handles models that are not found while using route model binding :(
|
||||||
|
// The only alternative is to set that at *each* route, which is crazypants
|
||||||
|
if ($e instanceof \Illuminate\Database\Eloquent\ModelNotFoundException) {
|
||||||
|
$model_name = last(explode('\\', $e->getModel()));
|
||||||
|
$route = str_plural(strtolower(last(explode('\\', $e->getModel())))).'.index';
|
||||||
|
|
||||||
|
// Sigh.
|
||||||
|
if ($route == 'assets.index') {
|
||||||
|
$route = 'hardware.index';
|
||||||
|
} elseif ($route == 'reporttemplates.index') {
|
||||||
|
$route = 'reports/custom';
|
||||||
|
} elseif ($route == 'assetmodels.index') {
|
||||||
|
$route = 'models.index';
|
||||||
|
} elseif ($route == 'predefinedkits.index') {
|
||||||
|
$route = 'kits.index';
|
||||||
|
} elseif ($route == 'assetmaintenances.index') {
|
||||||
|
$route = 'maintenances.index';
|
||||||
|
}
|
||||||
|
|
||||||
|
return redirect()
|
||||||
|
->route($route)
|
||||||
|
->withError(trans('general.generic_model_not_found', ['model' => $model_name]));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
if ($this->isHttpException($e) && (isset($statusCode)) && ($statusCode == '404' )) {
|
if ($this->isHttpException($e) && (isset($statusCode)) && ($statusCode == '404' )) {
|
||||||
|
|
|
@ -1520,11 +1520,11 @@ class Helper
|
||||||
if ($redirect_option == 'target') {
|
if ($redirect_option == 'target') {
|
||||||
switch ($checkout_to_type) {
|
switch ($checkout_to_type) {
|
||||||
case 'user':
|
case 'user':
|
||||||
return route('users.show', ['user' => $request->assigned_user]);
|
return route('users.show', $request->assigned_user);
|
||||||
case 'location':
|
case 'location':
|
||||||
return route('locations.show', ['location' => $request->assigned_location]);
|
return route('locations.show', $request->assigned_location);
|
||||||
case 'asset':
|
case 'asset':
|
||||||
return route('hardware.show', ['hardware' => $request->assigned_asset]);
|
return route('hardware.show', $request->assigned_asset);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'));
|
return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'));
|
||||||
|
|
|
@ -59,6 +59,8 @@ class IconHelper
|
||||||
return 'fas fa-cog';
|
return 'fas fa-cog';
|
||||||
case 'angle-left':
|
case 'angle-left':
|
||||||
return 'fas fa-angle-left';
|
return 'fas fa-angle-left';
|
||||||
|
case 'angle-right':
|
||||||
|
return 'fas fa-angle-right';
|
||||||
case 'warning':
|
case 'warning':
|
||||||
return 'fas fa-exclamation-triangle';
|
return 'fas fa-exclamation-triangle';
|
||||||
case 'kits':
|
case 'kits':
|
||||||
|
@ -184,6 +186,8 @@ class IconHelper
|
||||||
return 'fa-regular fa-id-card';
|
return 'fa-regular fa-id-card';
|
||||||
case 'department' :
|
case 'department' :
|
||||||
return 'fa-solid fa-building-user';
|
return 'fa-solid fa-building-user';
|
||||||
|
case 'home' :
|
||||||
|
return 'fa-solid fa-house';
|
||||||
case 'note':
|
case 'note':
|
||||||
case 'notes':
|
case 'notes':
|
||||||
return 'fas fa-sticky-note';
|
return 'fas fa-sticky-note';
|
||||||
|
|
|
@ -95,16 +95,10 @@ class AccessoriesController extends Controller
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
* @param int $accessoryId
|
* @param int $accessoryId
|
||||||
*/
|
*/
|
||||||
public function edit($accessoryId = null) : View | RedirectResponse
|
public function edit(Accessory $accessory) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
|
$this->authorize('update', Accessory::class);
|
||||||
if ($item = Accessory::find($accessoryId)) {
|
return view('accessories.edit')->with('item', $accessory)->with('category_type', 'accessory');
|
||||||
$this->authorize($item);
|
|
||||||
return view('accessories.edit', compact('item'))->with('category_type', 'accessory');
|
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist'));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -114,19 +108,12 @@ class AccessoriesController extends Controller
|
||||||
* @param int $accessoryId
|
* @param int $accessoryId
|
||||||
* @since [v6.0]
|
* @since [v6.0]
|
||||||
*/
|
*/
|
||||||
public function getClone($accessoryId = null) : View | RedirectResponse
|
public function getClone(Accessory $accessory) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
|
|
||||||
$this->authorize('create', Accessory::class);
|
$this->authorize('create', Accessory::class);
|
||||||
|
|
||||||
// Check if the asset exists
|
$accessory = clone $accessory;
|
||||||
if (is_null($accessory_to_clone = Accessory::find($accessoryId))) {
|
|
||||||
// Redirect to the asset management page
|
|
||||||
return redirect()->route('accessories.index')
|
|
||||||
->with('error', trans('admin/accessories/message.does_not_exist', ['id' => $accessoryId]));
|
|
||||||
}
|
|
||||||
|
|
||||||
$accessory = clone $accessory_to_clone;
|
|
||||||
$accessory->id = null;
|
$accessory->id = null;
|
||||||
$accessory->location_id = null;
|
$accessory->location_id = null;
|
||||||
|
|
||||||
|
@ -142,9 +129,9 @@ class AccessoriesController extends Controller
|
||||||
* @param ImageUploadRequest $request
|
* @param ImageUploadRequest $request
|
||||||
* @param int $accessoryId
|
* @param int $accessoryId
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $accessoryId = null) : RedirectResponse
|
public function update(ImageUploadRequest $request, Accessory $accessory) : RedirectResponse
|
||||||
{
|
{
|
||||||
if ($accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessoryId)) {
|
if ($accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessory->id)) {
|
||||||
|
|
||||||
$this->authorize($accessory);
|
$this->authorize($accessory);
|
||||||
|
|
||||||
|
@ -231,14 +218,10 @@ class AccessoriesController extends Controller
|
||||||
* @see AccessoriesController::getDataView() method that generates the JSON response
|
* @see AccessoriesController::getDataView() method that generates the JSON response
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function show($accessoryID = null) : View | RedirectResponse
|
public function show(Accessory $accessory) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessoryID);
|
$accessory = Accessory::withCount('checkouts as checkouts_count')->find($accessory->id);
|
||||||
$this->authorize('view', $accessory);
|
$this->authorize('view', $accessory);
|
||||||
if (isset($accessory->id)) {
|
|
||||||
return view('accessories.view', compact('accessory'));
|
return view('accessories.view', compact('accessory'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('accessories.index')->with('error', trans('admin/accessories/message.does_not_exist', ['id' => $accessoryID]));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -77,6 +77,18 @@ class AssetModelsController extends Controller
|
||||||
$assetmodels->onlyTrashed();
|
$assetmodels->onlyTrashed();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($request->filled('name')) {
|
||||||
|
$assetmodels = $assetmodels->where('models.name', '=', $request->input('name'));
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($request->filled('model_number')) {
|
||||||
|
$assetmodels = $assetmodels->where('models.model_number', '=', $request->input('model_number'));
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($request->filled('notes')) {
|
||||||
|
$assetmodels = $assetmodels->where('models.notes', '=', $request->input('notes'));
|
||||||
|
}
|
||||||
|
|
||||||
if ($request->filled('category_id')) {
|
if ($request->filled('category_id')) {
|
||||||
$assetmodels = $assetmodels->where('models.category_id', '=', $request->input('category_id'));
|
$assetmodels = $assetmodels->where('models.category_id', '=', $request->input('category_id'));
|
||||||
}
|
}
|
||||||
|
|
|
@ -765,9 +765,13 @@ class AssetsController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($problems_updating_encrypted_custom_fields) {
|
if ($problems_updating_encrypted_custom_fields) {
|
||||||
return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.encrypted_warning')));
|
return response()->json(Helper::formatStandardApiResponse('success', $asset, trans('admin/hardware/message.update.encrypted_warning')));
|
||||||
|
// Below is the *correct* return since it uses the transformer, but we have to use the old, flat return for now until we can update Jamf2Snipe and Kanji2Snipe
|
||||||
|
// return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.encrypted_warning')));
|
||||||
} else {
|
} else {
|
||||||
return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.success')));
|
return response()->json(Helper::formatStandardApiResponse('success', $asset, trans('admin/hardware/message.update.success')));
|
||||||
|
// Below is the *correct* return since it uses the transformer, but we have to use the old, flat return for now until we can update Jamf2Snipe and Kanji2Snipe
|
||||||
|
/// return response()->json(Helper::formatStandardApiResponse('success', (new AssetsTransformer)->transformAsset($asset), trans('admin/hardware/message.update.success')));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return response()->json(Helper::formatStandardApiResponse('error', null, $asset->getErrors()), 200);
|
return response()->json(Helper::formatStandardApiResponse('error', null, $asset->getErrors()), 200);
|
||||||
|
@ -1226,7 +1230,10 @@ class AssetsController extends Controller
|
||||||
{
|
{
|
||||||
$this->authorize('view', Asset::class);
|
$this->authorize('view', Asset::class);
|
||||||
$this->authorize('view', $asset);
|
$this->authorize('view', $asset);
|
||||||
$accessory_checkouts = AccessoryCheckout::AssetsAssigned()->with('adminuser')->with('accessories');
|
$accessory_checkouts = AccessoryCheckout::AssetsAssigned()
|
||||||
|
->where('assigned_to', $asset->id)
|
||||||
|
->with('adminuser')
|
||||||
|
->with('accessories');
|
||||||
|
|
||||||
$offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value');
|
$offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value');
|
||||||
$limit = app('api_limit_value');
|
$limit = app('api_limit_value');
|
||||||
|
@ -1235,6 +1242,8 @@ class AssetsController extends Controller
|
||||||
$accessory_checkouts = $accessory_checkouts->skip($offset)->take($limit)->get();
|
$accessory_checkouts = $accessory_checkouts->skip($offset)->take($limit)->get();
|
||||||
return (new AssetsTransformer)->transformCheckedoutAccessories($accessory_checkouts, $total);
|
return (new AssetsTransformer)->transformCheckedoutAccessories($accessory_checkouts, $total);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generate asset labels by tag
|
* Generate asset labels by tag
|
||||||
*
|
*
|
||||||
|
|
|
@ -39,6 +39,7 @@ class CategoriesController extends Controller
|
||||||
'components_count',
|
'components_count',
|
||||||
'licenses_count',
|
'licenses_count',
|
||||||
'image',
|
'image',
|
||||||
|
'notes',
|
||||||
];
|
];
|
||||||
|
|
||||||
$categories = Category::select([
|
$categories = Category::select([
|
||||||
|
@ -52,6 +53,7 @@ class CategoriesController extends Controller
|
||||||
'require_acceptance',
|
'require_acceptance',
|
||||||
'checkin_email',
|
'checkin_email',
|
||||||
'image',
|
'image',
|
||||||
|
'notes',
|
||||||
])
|
])
|
||||||
->with('adminuser')
|
->with('adminuser')
|
||||||
->withCount('accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'licenses as licenses_count');
|
->withCount('accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'licenses as licenses_count');
|
||||||
|
|
|
@ -38,11 +38,12 @@ class CompaniesController extends Controller
|
||||||
'accessories_count',
|
'accessories_count',
|
||||||
'consumables_count',
|
'consumables_count',
|
||||||
'components_count',
|
'components_count',
|
||||||
|
'notes',
|
||||||
];
|
];
|
||||||
|
|
||||||
$companies = Company::withCount(['assets as assets_count' => function ($query) {
|
$companies = Company::withCount(['assets as assets_count' => function ($query) {
|
||||||
$query->AssetsForShow();
|
$query->AssetsForShow();
|
||||||
}])->withCount('assets as assets_count', 'licenses as licenses_count', 'accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'users as users_count');
|
}])->withCount('licenses as licenses_count', 'accessories as accessories_count', 'consumables as consumables_count', 'components as components_count', 'users as users_count');
|
||||||
|
|
||||||
if ($request->filled('search')) {
|
if ($request->filled('search')) {
|
||||||
$companies->TextSearch($request->input('search'));
|
$companies->TextSearch($request->input('search'));
|
||||||
|
|
|
@ -48,7 +48,8 @@ class ComponentsController extends Controller
|
||||||
];
|
];
|
||||||
|
|
||||||
$components = Component::select('components.*')
|
$components = Component::select('components.*')
|
||||||
->with('company', 'location', 'category', 'assets', 'supplier', 'adminuser', 'manufacturer');
|
->with('company', 'location', 'category', 'assets', 'supplier', 'adminuser', 'manufacturer', 'uncontrainedAssets')
|
||||||
|
->withSum('uncontrainedAssets', 'components_assets.assigned_qty');
|
||||||
|
|
||||||
if ($request->filled('search')) {
|
if ($request->filled('search')) {
|
||||||
$components = $components->TextSearch($request->input('search'));
|
$components = $components->TextSearch($request->input('search'));
|
||||||
|
@ -197,6 +198,11 @@ class ComponentsController extends Controller
|
||||||
$this->authorize('delete', Component::class);
|
$this->authorize('delete', Component::class);
|
||||||
$component = Component::findOrFail($id);
|
$component = Component::findOrFail($id);
|
||||||
$this->authorize('delete', $component);
|
$this->authorize('delete', $component);
|
||||||
|
|
||||||
|
if ($component->numCheckedOut() > 0) {
|
||||||
|
return response()->json(Helper::formatStandardApiResponse('error', null, trans('admin/components/message.delete.error_qty')));
|
||||||
|
}
|
||||||
|
|
||||||
$component->delete();
|
$component->delete();
|
||||||
|
|
||||||
return response()->json(Helper::formatStandardApiResponse('success', null, trans('admin/components/message.delete.success')));
|
return response()->json(Helper::formatStandardApiResponse('success', null, trans('admin/components/message.delete.success')));
|
||||||
|
|
|
@ -23,7 +23,7 @@ class DepartmentsController extends Controller
|
||||||
public function index(Request $request) : JsonResponse | array
|
public function index(Request $request) : JsonResponse | array
|
||||||
{
|
{
|
||||||
$this->authorize('view', Department::class);
|
$this->authorize('view', Department::class);
|
||||||
$allowed_columns = ['id', 'name', 'image', 'users_count'];
|
$allowed_columns = ['id', 'name', 'image', 'users_count', 'notes'];
|
||||||
|
|
||||||
$departments = Department::select(
|
$departments = Department::select(
|
||||||
'departments.id',
|
'departments.id',
|
||||||
|
@ -35,7 +35,8 @@ class DepartmentsController extends Controller
|
||||||
'departments.manager_id',
|
'departments.manager_id',
|
||||||
'departments.created_at',
|
'departments.created_at',
|
||||||
'departments.updated_at',
|
'departments.updated_at',
|
||||||
'departments.image'
|
'departments.image',
|
||||||
|
'departments.notes',
|
||||||
)->with('users')->with('location')->with('manager')->with('company')->withCount('users as users_count');
|
)->with('users')->with('location')->with('manager')->with('company')->withCount('users as users_count');
|
||||||
|
|
||||||
if ($request->filled('search')) {
|
if ($request->filled('search')) {
|
||||||
|
|
|
@ -24,7 +24,7 @@ class GroupsController extends Controller
|
||||||
|
|
||||||
$this->authorize('view', Group::class);
|
$this->authorize('view', Group::class);
|
||||||
|
|
||||||
$groups = Group::select('id', 'name', 'permissions', 'created_at', 'updated_at', 'created_by')->with('adminuser')->withCount('users as users_count');
|
$groups = Group::select('id', 'name', 'permissions', 'notes', 'created_at', 'updated_at', 'created_by')->with('adminuser')->withCount('users as users_count');
|
||||||
|
|
||||||
if ($request->filled('search')) {
|
if ($request->filled('search')) {
|
||||||
$groups = $groups->TextSearch($request->input('search'));
|
$groups = $groups->TextSearch($request->input('search'));
|
||||||
|
@ -81,6 +81,7 @@ class GroupsController extends Controller
|
||||||
|
|
||||||
$group->name = $request->input('name');
|
$group->name = $request->input('name');
|
||||||
$group->created_by = auth()->id();
|
$group->created_by = auth()->id();
|
||||||
|
$group->notes = $request->input('notes');
|
||||||
$group->permissions = json_encode($request->input('permissions', $groupPermissions));
|
$group->permissions = json_encode($request->input('permissions', $groupPermissions));
|
||||||
|
|
||||||
if ($group->save()) {
|
if ($group->save()) {
|
||||||
|
@ -118,6 +119,7 @@ class GroupsController extends Controller
|
||||||
$group = Group::findOrFail($id);
|
$group = Group::findOrFail($id);
|
||||||
|
|
||||||
$group->name = $request->input('name');
|
$group->name = $request->input('name');
|
||||||
|
$group->notes = $request->input('notes');
|
||||||
$group->permissions = $request->input('permissions'); // Todo - some JSON validation stuff here
|
$group->permissions = $request->input('permissions'); // Todo - some JSON validation stuff here
|
||||||
|
|
||||||
if ($group->save()) {
|
if ($group->save()) {
|
||||||
|
|
|
@ -66,16 +66,31 @@ class ImportController extends Controller
|
||||||
if (! ini_get('auto_detect_line_endings')) {
|
if (! ini_get('auto_detect_line_endings')) {
|
||||||
ini_set('auto_detect_line_endings', '1');
|
ini_set('auto_detect_line_endings', '1');
|
||||||
}
|
}
|
||||||
|
if (function_exists('iconv')) {
|
||||||
$file_contents = $file->getContent(); //TODO - this *does* load the whole file in RAM, but we need that to be able to 'iconv' it?
|
$file_contents = $file->getContent(); //TODO - this *does* load the whole file in RAM, but we need that to be able to 'iconv' it?
|
||||||
$encoding = $detector->getEncoding($file_contents);
|
$encoding = $detector->getEncoding($file_contents);
|
||||||
|
\Log::warning("Discovered encoding: $encoding in uploaded CSV");
|
||||||
$reader = null;
|
$reader = null;
|
||||||
if (strcasecmp($encoding, 'UTF-8') != 0) {
|
if (strcasecmp($encoding, 'UTF-8') != 0) {
|
||||||
$transliterated = iconv($encoding, 'UTF-8', $file_contents);
|
$transliterated = false;
|
||||||
|
try {
|
||||||
|
$transliterated = iconv(strtoupper($encoding), 'UTF-8', $file_contents);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
$transliterated = false; //blank out the partially-decoded string
|
||||||
|
return response()->json(
|
||||||
|
Helper::formatStandardApiResponse(
|
||||||
|
'error',
|
||||||
|
null,
|
||||||
|
trans('admin/hardware/message.import.transliterate_failure', ["encoding" => $encoding])
|
||||||
|
),
|
||||||
|
422
|
||||||
|
);
|
||||||
|
}
|
||||||
if ($transliterated !== false) {
|
if ($transliterated !== false) {
|
||||||
$tmpname = tempnam(sys_get_temp_dir(), '');
|
$tmpname = tempnam(sys_get_temp_dir(), '');
|
||||||
$tmpresults = file_put_contents($tmpname, $transliterated);
|
$tmpresults = file_put_contents($tmpname, $transliterated);
|
||||||
if ($tmpresults !== false) {
|
|
||||||
$transliterated = null; //save on memory?
|
$transliterated = null; //save on memory?
|
||||||
|
if ($tmpresults !== false) {
|
||||||
$newfile = new UploadedFile($tmpname, $file->getClientOriginalName(), null, null, true); //WARNING: this is enabling 'test mode' - which is gross, but otherwise the file won't be treated as 'uploaded'
|
$newfile = new UploadedFile($tmpname, $file->getClientOriginalName(), null, null, true); //WARNING: this is enabling 'test mode' - which is gross, but otherwise the file won't be treated as 'uploaded'
|
||||||
if ($newfile->isValid()) {
|
if ($newfile->isValid()) {
|
||||||
$file = $newfile;
|
$file = $newfile;
|
||||||
|
@ -83,8 +98,9 @@ class ImportController extends Controller
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$reader = Reader::createFromFileObject($file->openFile('r')); //file pointer leak?
|
|
||||||
$file_contents = null; //try to save on memory, I guess?
|
$file_contents = null; //try to save on memory, I guess?
|
||||||
|
}
|
||||||
|
$reader = Reader::createFromFileObject($file->openFile('r')); //file pointer leak?
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$import->header_row = $reader->fetchOne(0);
|
$import->header_row = $reader->fetchOne(0);
|
||||||
|
|
|
@ -53,6 +53,7 @@ class LocationsController extends Controller
|
||||||
'updated_at',
|
'updated_at',
|
||||||
'users_count',
|
'users_count',
|
||||||
'zip',
|
'zip',
|
||||||
|
'notes',
|
||||||
];
|
];
|
||||||
|
|
||||||
$locations = Location::with('parent', 'manager', 'children')->select([
|
$locations = Location::with('parent', 'manager', 'children')->select([
|
||||||
|
@ -73,6 +74,7 @@ class LocationsController extends Controller
|
||||||
'locations.image',
|
'locations.image',
|
||||||
'locations.ldap_ou',
|
'locations.ldap_ou',
|
||||||
'locations.currency',
|
'locations.currency',
|
||||||
|
'locations.notes',
|
||||||
])
|
])
|
||||||
->withCount('assignedAssets as assigned_assets_count')
|
->withCount('assignedAssets as assigned_assets_count')
|
||||||
->withCount('assets as assets_count')
|
->withCount('assets as assets_count')
|
||||||
|
@ -190,6 +192,7 @@ class LocationsController extends Controller
|
||||||
'locations.updated_at',
|
'locations.updated_at',
|
||||||
'locations.image',
|
'locations.image',
|
||||||
'locations.currency',
|
'locations.currency',
|
||||||
|
'locations.notes',
|
||||||
])
|
])
|
||||||
->withCount('assignedAssets as assigned_assets_count')
|
->withCount('assignedAssets as assigned_assets_count')
|
||||||
->withCount('assets as assets_count')
|
->withCount('assets as assets_count')
|
||||||
|
@ -255,7 +258,7 @@ class LocationsController extends Controller
|
||||||
{
|
{
|
||||||
$this->authorize('view', Accessory::class);
|
$this->authorize('view', Accessory::class);
|
||||||
$this->authorize('view', $location);
|
$this->authorize('view', $location);
|
||||||
$accessory_checkouts = AccessoryCheckout::LocationAssigned()->with('adminuser')->with('accessories');
|
$accessory_checkouts = AccessoryCheckout::LocationAssigned()->where('assigned_to', $location->id)->with('adminuser')->with('accessories');
|
||||||
|
|
||||||
$offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value');
|
$offset = ($request->input('offset') > $accessory_checkouts->count()) ? $accessory_checkouts->count() : app('api_offset_value');
|
||||||
$limit = app('api_limit_value');
|
$limit = app('api_limit_value');
|
||||||
|
|
|
@ -39,7 +39,8 @@ class ManufacturersController extends Controller
|
||||||
'assets_count',
|
'assets_count',
|
||||||
'consumables_count',
|
'consumables_count',
|
||||||
'components_count',
|
'components_count',
|
||||||
'licenses_count'
|
'licenses_count',
|
||||||
|
'notes',
|
||||||
];
|
];
|
||||||
|
|
||||||
$manufacturers = Manufacturer::select([
|
$manufacturers = Manufacturer::select([
|
||||||
|
@ -55,6 +56,7 @@ class ManufacturersController extends Controller
|
||||||
'updated_at',
|
'updated_at',
|
||||||
'image',
|
'image',
|
||||||
'deleted_at',
|
'deleted_at',
|
||||||
|
'notes',
|
||||||
])
|
])
|
||||||
->with('adminuser')
|
->with('adminuser')
|
||||||
->withCount('assets as assets_count')
|
->withCount('assets as assets_count')
|
||||||
|
|
|
@ -1,43 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace App\Http\Controllers\Api;
|
|
||||||
|
|
||||||
use App\Events\NoteAdded;
|
|
||||||
use App\Helpers\Helper;
|
|
||||||
use App\Http\Controllers\Controller;
|
|
||||||
use App\Models\Asset;
|
|
||||||
use Illuminate\Http\Request;
|
|
||||||
use Illuminate\Support\Facades\Auth;
|
|
||||||
use Illuminate\Validation\Rule;
|
|
||||||
|
|
||||||
class NotesController extends Controller
|
|
||||||
{
|
|
||||||
public function store(Request $request)
|
|
||||||
{
|
|
||||||
$validated = $request->validate([
|
|
||||||
'note' => 'required|string|max:500',
|
|
||||||
'type' => [
|
|
||||||
'required',
|
|
||||||
Rule::in(['asset']),
|
|
||||||
],
|
|
||||||
]);
|
|
||||||
|
|
||||||
// This can be made dynamic by using $request->input('type') to determine which model type to add the note to.
|
|
||||||
// For now, we are only placing this on Assets
|
|
||||||
$item = Asset::findOrFail($request->input("id"));
|
|
||||||
$this->authorize('update', $item);
|
|
||||||
|
|
||||||
event(new NoteAdded($item, Auth::user(), $validated['note']));
|
|
||||||
|
|
||||||
return response()->json(Helper::formatStandardApiResponse('success'));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function update(Request $request)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
public function destroy(Request $request)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -290,10 +290,12 @@ class StatuslabelsController extends Controller
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a boolean response based on whether the status label
|
* Returns a boolean response based on whether the status label
|
||||||
* is one that is deployable.
|
* is one that is deployable or pending.
|
||||||
*
|
*
|
||||||
* This is used by the hardware create/edit view to determine whether
|
* This is used by the hardware create/edit view to determine whether
|
||||||
* we should provide a dropdown of users for them to check the asset out to.
|
* we should provide a dropdown of users for them to check the asset out to,
|
||||||
|
* and whether we show a warning that the asset will be checked in if it's already
|
||||||
|
* assigned but the status is changed to one that isn't pending or deployable
|
||||||
*
|
*
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
* @since [v4.0]
|
* @since [v4.0]
|
||||||
|
@ -301,7 +303,7 @@ class StatuslabelsController extends Controller
|
||||||
public function checkIfDeployable($id) : string
|
public function checkIfDeployable($id) : string
|
||||||
{
|
{
|
||||||
$statuslabel = Statuslabel::findOrFail($id);
|
$statuslabel = Statuslabel::findOrFail($id);
|
||||||
if ($statuslabel->getStatuslabelType() == 'deployable') {
|
if (($statuslabel->getStatuslabelType() == 'pending') || ($statuslabel->getStatuslabelType() == 'deployable')) {
|
||||||
return '1';
|
return '1';
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -139,19 +139,12 @@ class AssetMaintenancesController extends Controller
|
||||||
* @version v1.0
|
* @version v1.0
|
||||||
* @since [v1.8]
|
* @since [v1.8]
|
||||||
*/
|
*/
|
||||||
public function edit($assetMaintenanceId = null) : View | RedirectResponse
|
public function edit(AssetMaintenance $maintenance) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Asset::class);
|
$this->authorize('update', Asset::class);
|
||||||
// Check if the asset maintenance exists
|
if ((!$maintenance->asset) || ($maintenance->asset->deleted_at!='')) {
|
||||||
$this->authorize('update', Asset::class);
|
|
||||||
// Check if the asset maintenance exists
|
|
||||||
if (is_null($assetMaintenance = AssetMaintenance::find($assetMaintenanceId))) {
|
|
||||||
// Redirect to the asset maintenance management page
|
|
||||||
return redirect()->route('maintenances.index')->with('error', trans('admin/asset_maintenances/message.not_found'));
|
|
||||||
} elseif ((!$assetMaintenance->asset) || ($assetMaintenance->asset->deleted_at!='')) {
|
|
||||||
// Redirect to the asset maintenance management page
|
|
||||||
return redirect()->route('maintenances.index')->with('error', 'asset does not exist');
|
return redirect()->route('maintenances.index')->with('error', 'asset does not exist');
|
||||||
} elseif (! Company::isCurrentUserHasAccess($assetMaintenance->asset)) {
|
} elseif (! Company::isCurrentUserHasAccess($maintenance->asset)) {
|
||||||
return static::getInsufficientPermissionsRedirect();
|
return static::getInsufficientPermissionsRedirect();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -161,7 +154,7 @@ class AssetMaintenancesController extends Controller
|
||||||
return view('asset_maintenances/edit')
|
return view('asset_maintenances/edit')
|
||||||
->with('selectedAsset', null)
|
->with('selectedAsset', null)
|
||||||
->with('assetMaintenanceType', $assetMaintenanceType)
|
->with('assetMaintenanceType', $assetMaintenanceType)
|
||||||
->with('item', $assetMaintenance);
|
->with('item', $maintenance);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -174,24 +167,20 @@ class AssetMaintenancesController extends Controller
|
||||||
* @version v1.0
|
* @version v1.0
|
||||||
* @since [v1.8]
|
* @since [v1.8]
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, $assetMaintenanceId = null) : View | RedirectResponse
|
public function update(Request $request, AssetMaintenance $maintenance) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Asset::class);
|
$this->authorize('update', Asset::class);
|
||||||
// Check if the asset maintenance exists
|
|
||||||
if (is_null($assetMaintenance = AssetMaintenance::find($assetMaintenanceId))) {
|
if ((!$maintenance->asset) || ($maintenance->asset->deleted_at!='')) {
|
||||||
// Redirect to the asset maintenance management page
|
|
||||||
return redirect()->route('maintenances.index')->with('error', trans('admin/asset_maintenances/message.not_found'));
|
|
||||||
} elseif ((!$assetMaintenance->asset) || ($assetMaintenance->asset->deleted_at!='')) {
|
|
||||||
// Redirect to the asset maintenance management page
|
|
||||||
return redirect()->route('maintenances.index')->with('error', 'asset does not exist');
|
return redirect()->route('maintenances.index')->with('error', 'asset does not exist');
|
||||||
} elseif (! Company::isCurrentUserHasAccess($assetMaintenance->asset)) {
|
} elseif (! Company::isCurrentUserHasAccess($maintenance->asset)) {
|
||||||
return static::getInsufficientPermissionsRedirect();
|
return static::getInsufficientPermissionsRedirect();
|
||||||
}
|
}
|
||||||
|
|
||||||
$assetMaintenance->supplier_id = $request->input('supplier_id');
|
$maintenance->supplier_id = $request->input('supplier_id');
|
||||||
$assetMaintenance->is_warranty = $request->input('is_warranty');
|
$maintenance->is_warranty = $request->input('is_warranty');
|
||||||
$assetMaintenance->cost = $request->input('cost');
|
$maintenance->cost = $request->input('cost');
|
||||||
$assetMaintenance->notes = $request->input('notes');
|
$maintenance->notes = $request->input('notes');
|
||||||
|
|
||||||
$asset = Asset::find(request('asset_id'));
|
$asset = Asset::find(request('asset_id'));
|
||||||
|
|
||||||
|
@ -200,39 +189,39 @@ class AssetMaintenancesController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
// Save the asset maintenance data
|
// Save the asset maintenance data
|
||||||
$assetMaintenance->asset_id = $request->input('asset_id');
|
$maintenance->asset_id = $request->input('asset_id');
|
||||||
$assetMaintenance->asset_maintenance_type = $request->input('asset_maintenance_type');
|
$maintenance->asset_maintenance_type = $request->input('asset_maintenance_type');
|
||||||
$assetMaintenance->title = $request->input('title');
|
$maintenance->title = $request->input('title');
|
||||||
$assetMaintenance->start_date = $request->input('start_date');
|
$maintenance->start_date = $request->input('start_date');
|
||||||
$assetMaintenance->completion_date = $request->input('completion_date');
|
$maintenance->completion_date = $request->input('completion_date');
|
||||||
|
|
||||||
if (($assetMaintenance->completion_date == null)
|
if (($maintenance->completion_date == null)
|
||||||
) {
|
) {
|
||||||
if (($assetMaintenance->asset_maintenance_time !== 0)
|
if (($maintenance->asset_maintenance_time !== 0)
|
||||||
|| (! is_null($assetMaintenance->asset_maintenance_time))
|
|| (! is_null($maintenance->asset_maintenance_time))
|
||||||
) {
|
) {
|
||||||
$assetMaintenance->asset_maintenance_time = null;
|
$maintenance->asset_maintenance_time = null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (($assetMaintenance->completion_date !== null)
|
if (($maintenance->completion_date !== null)
|
||||||
&& ($assetMaintenance->start_date !== '')
|
&& ($maintenance->start_date !== '')
|
||||||
&& ($assetMaintenance->start_date !== '0000-00-00')
|
&& ($maintenance->start_date !== '0000-00-00')
|
||||||
) {
|
) {
|
||||||
$startDate = Carbon::parse($assetMaintenance->start_date);
|
$startDate = Carbon::parse($maintenance->start_date);
|
||||||
$completionDate = Carbon::parse($assetMaintenance->completion_date);
|
$completionDate = Carbon::parse($maintenance->completion_date);
|
||||||
$assetMaintenance->asset_maintenance_time = $completionDate->diffInDays($startDate);
|
$maintenance->asset_maintenance_time = $completionDate->diffInDays($startDate);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Was the asset maintenance created?
|
// Was the asset maintenance created?
|
||||||
if ($assetMaintenance->save()) {
|
if ($maintenance->save()) {
|
||||||
|
|
||||||
// Redirect to the new asset maintenance page
|
// Redirect to the new asset maintenance page
|
||||||
return redirect()->route('maintenances.index')
|
return redirect()->route('maintenances.index')
|
||||||
->with('success', trans('admin/asset_maintenances/message.edit.success'));
|
->with('success', trans('admin/asset_maintenances/message.edit.success'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->back()->withInput()->withErrors($assetMaintenance->getErrors());
|
return redirect()->back()->withInput()->withErrors($maintenance->getErrors());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -271,19 +260,13 @@ class AssetMaintenancesController extends Controller
|
||||||
* @version v1.0
|
* @version v1.0
|
||||||
* @since [v1.8]
|
* @since [v1.8]
|
||||||
*/
|
*/
|
||||||
public function show($assetMaintenanceId) : View | RedirectResponse
|
public function show(AssetMaintenance $maintenance) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', Asset::class);
|
$this->authorize('view', Asset::class);
|
||||||
|
if (! Company::isCurrentUserHasAccess($maintenance->asset)) {
|
||||||
// Check if the asset maintenance exists
|
|
||||||
if (is_null($assetMaintenance = AssetMaintenance::find($assetMaintenanceId))) {
|
|
||||||
// Redirect to the asset maintenance management page
|
|
||||||
return redirect()->route('maintenances.index')
|
|
||||||
->with('error', trans('admin/asset_maintenances/message.not_found'));
|
|
||||||
} elseif (! Company::isCurrentUserHasAccess($assetMaintenance->asset)) {
|
|
||||||
return static::getInsufficientPermissionsRedirect();
|
return static::getInsufficientPermissionsRedirect();
|
||||||
}
|
}
|
||||||
|
|
||||||
return view('asset_maintenances/view')->with('assetMaintenance', $assetMaintenance);
|
return view('asset_maintenances/view')->with('assetMaintenance', $maintenance);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -109,16 +109,11 @@ class AssetModelsController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @param int $modelId
|
* @param int $modelId
|
||||||
*/
|
*/
|
||||||
public function edit($modelId = null) : View | RedirectResponse
|
public function edit(AssetModel $model) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', AssetModel::class);
|
$this->authorize('update', AssetModel::class);
|
||||||
if ($item = AssetModel::find($modelId)) {
|
|
||||||
$category_type = 'asset';
|
$category_type = 'asset';
|
||||||
return view('models/edit', compact('item', 'category_type'))->with('depreciation_list', Helper::depreciationList());
|
return view('models/edit', compact('category_type'))->with('item', $model)->with('depreciation_list', Helper::depreciationList());
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -133,16 +128,11 @@ class AssetModelsController extends Controller
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function update(StoreAssetModelRequest $request, $modelId) : RedirectResponse
|
public function update(StoreAssetModelRequest $request, AssetModel $model) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', AssetModel::class);
|
$this->authorize('update', AssetModel::class);
|
||||||
|
|
||||||
if (is_null($model = AssetModel::find($modelId))) {
|
|
||||||
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$model = $request->handleImages($model);
|
$model = $request->handleImages($model);
|
||||||
|
|
||||||
$model->depreciation_id = $request->input('depreciation_id');
|
$model->depreciation_id = $request->input('depreciation_id');
|
||||||
$model->eol = $request->input('eol');
|
$model->eol = $request->input('eol');
|
||||||
$model->name = $request->input('name');
|
$model->name = $request->input('name');
|
||||||
|
@ -188,28 +178,16 @@ class AssetModelsController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @param int $modelId
|
* @param int $modelId
|
||||||
*/
|
*/
|
||||||
public function destroy($modelId) : RedirectResponse
|
public function destroy(AssetModel $model) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('delete', AssetModel::class);
|
$this->authorize('delete', AssetModel::class);
|
||||||
// Check if the model exists
|
|
||||||
if (is_null($model = AssetModel::find($modelId))) {
|
|
||||||
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($model->assets()->count() > 0) {
|
if ($model->assets()->count() > 0) {
|
||||||
// Throw an error that this model is associated with assets
|
// Throw an error that this model is associated with assets
|
||||||
return redirect()->route('models.index')->with('error', trans('admin/models/message.assoc_users'));
|
return redirect()->route('models.index')->with('error', trans('admin/models/message.assoc_users'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($model->image) {
|
|
||||||
try {
|
|
||||||
Storage::disk('public')->delete('models/'.$model->image);
|
|
||||||
$model->update(['image' => null]);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
Log::info($e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Delete the model
|
// Delete the model
|
||||||
$model->delete();
|
$model->delete();
|
||||||
|
|
||||||
|
@ -267,18 +245,12 @@ class AssetModelsController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @param int $modelId
|
* @param int $modelId
|
||||||
*/
|
*/
|
||||||
public function show($modelId = null) : View | RedirectResponse
|
public function show(AssetModel $model) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', AssetModel::class);
|
$this->authorize('view', AssetModel::class);
|
||||||
$model = AssetModel::withTrashed()->find($modelId);
|
|
||||||
|
|
||||||
if (isset($model->id)) {
|
|
||||||
return view('models/view', compact('model'));
|
return view('models/view', compact('model'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the clone page to clone a model
|
* Get the clone page to clone a model
|
||||||
*
|
*
|
||||||
|
@ -286,23 +258,20 @@ class AssetModelsController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @param int $modelId
|
* @param int $modelId
|
||||||
*/
|
*/
|
||||||
public function getClone($modelId = null) : View | RedirectResponse
|
public function getClone(AssetModel $model) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('create', AssetModel::class);
|
$this->authorize('create', AssetModel::class);
|
||||||
// Check if the model exists
|
|
||||||
if (is_null($model_to_clone = AssetModel::find($modelId))) {
|
|
||||||
return redirect()->route('models.index')->with('error', trans('admin/models/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$model = clone $model_to_clone;
|
$cloned_model = clone $model;
|
||||||
$model->id = null;
|
$model->id = null;
|
||||||
|
$model->deleted_at = null;
|
||||||
|
|
||||||
// Show the page
|
// Show the page
|
||||||
return view('models/edit')
|
return view('models/edit')
|
||||||
->with('depreciation_list', Helper::depreciationList())
|
->with('depreciation_list', Helper::depreciationList())
|
||||||
->with('item', $model)
|
->with('item', $model)
|
||||||
->with('model_id', $model_to_clone->id)
|
->with('model_id', $model->id)
|
||||||
->with('clone_model', $model_to_clone);
|
->with('clone_model', $cloned_model);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -321,7 +290,7 @@ class AssetModelsController extends Controller
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a view that allows the user to bulk edit model attrbutes
|
* Returns a view that allows the user to bulk edit model attributes
|
||||||
*
|
*
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
* @since [v1.7]
|
* @since [v1.7]
|
||||||
|
|
|
@ -58,11 +58,9 @@ class AssetModelsFilesController extends Controller
|
||||||
* @param int $fileId
|
* @param int $fileId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function show($modelId = null, $fileId = null) : StreamedResponse | Response | RedirectResponse | BinaryFileResponse
|
public function show(AssetModel $model, $fileId = null) : StreamedResponse | Response | RedirectResponse | BinaryFileResponse
|
||||||
{
|
{
|
||||||
$model = AssetModel::find($modelId);
|
|
||||||
// the asset is valid
|
|
||||||
if (isset($model->id)) {
|
|
||||||
$this->authorize('view', $model);
|
$this->authorize('view', $model);
|
||||||
|
|
||||||
if (! $log = Actionlog::find($fileId)) {
|
if (! $log = Actionlog::find($fileId)) {
|
||||||
|
@ -88,12 +86,6 @@ class AssetModelsFilesController extends Controller
|
||||||
|
|
||||||
return StorageHelper::downloader($file);
|
return StorageHelper::downloader($file);
|
||||||
}
|
}
|
||||||
// Prepare the error message
|
|
||||||
$error = trans('admin/hardware/message.does_not_exist', ['id' => $fileId]);
|
|
||||||
|
|
||||||
// Redirect to the hardware management page
|
|
||||||
return redirect()->route('hardware.index')->with('error', $error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Delete the associated file
|
* Delete the associated file
|
||||||
|
@ -103,14 +95,9 @@ class AssetModelsFilesController extends Controller
|
||||||
* @param int $fileId
|
* @param int $fileId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function destroy($modelId = null, $fileId = null) : RedirectResponse
|
public function destroy(AssetModel $model, $fileId = null) : RedirectResponse
|
||||||
{
|
{
|
||||||
$model = AssetModel::find($modelId);
|
|
||||||
$this->authorize('update', $model);
|
|
||||||
$rel_path = 'private_uploads/assetmodels';
|
$rel_path = 'private_uploads/assetmodels';
|
||||||
|
|
||||||
// the asset is valid
|
|
||||||
if (isset($model->id)) {
|
|
||||||
$this->authorize('update', $model);
|
$this->authorize('update', $model);
|
||||||
$log = Actionlog::find($fileId);
|
$log = Actionlog::find($fileId);
|
||||||
if ($log) {
|
if ($log) {
|
||||||
|
@ -123,9 +110,6 @@ class AssetModelsFilesController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
|
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
|
||||||
}
|
|
||||||
|
|
||||||
// Redirect to the hardware management page
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -27,18 +27,12 @@ class AssetCheckinController extends Controller
|
||||||
* @param string $backto
|
* @param string $backto
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function create($assetId, $backto = null) : View | RedirectResponse
|
public function create(Asset $asset, $backto = null) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
// Check if the asset exists
|
|
||||||
if (is_null($asset = Asset::find($assetId))) {
|
|
||||||
// Redirect to the asset management page with error
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('checkin', $asset);
|
$this->authorize('checkin', $asset);
|
||||||
|
|
||||||
// This asset is already checked in, redirect
|
// This asset is already checked in, redirect
|
||||||
|
|
||||||
if (is_null($asset->assignedTo)) {
|
if (is_null($asset->assignedTo)) {
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.checkin.already_checked_in'));
|
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.checkin.already_checked_in'));
|
||||||
}
|
}
|
||||||
|
@ -47,7 +41,11 @@ class AssetCheckinController extends Controller
|
||||||
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix'));
|
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return view('hardware/checkin', compact('asset'))->with('statusLabel_list', Helper::statusLabelList())->with('backto', $backto)->with('table_name', 'Assets');
|
return view('hardware/checkin', compact('asset'))
|
||||||
|
->with('item', $asset)
|
||||||
|
->with('statusLabel_list', Helper::statusLabelList())
|
||||||
|
->with('backto', $backto)
|
||||||
|
->with('table_name', 'Assets');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -91,6 +89,9 @@ class AssetCheckinController extends Controller
|
||||||
$asset->status_id = e($request->get('status_id'));
|
$asset->status_id = e($request->get('status_id'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Add any custom fields that should be included in the checkout
|
||||||
|
$asset->customFieldsForCheckinCheckout('display_checkin');
|
||||||
|
|
||||||
$this->migrateLegacyLocations($asset);
|
$this->migrateLegacyLocations($asset);
|
||||||
|
|
||||||
$asset->location_id = $asset->rtd_location_id;
|
$asset->location_id = $asset->rtd_location_id;
|
||||||
|
@ -128,6 +129,9 @@ class AssetCheckinController extends Controller
|
||||||
|
|
||||||
session()->put('redirect_option', $request->get('redirect_option'));
|
session()->put('redirect_option', $request->get('redirect_option'));
|
||||||
|
|
||||||
|
// Add any custom fields that should be included in the checkout
|
||||||
|
$asset->customFieldsForCheckinCheckout('display_checkin');
|
||||||
|
|
||||||
if ($asset->save()) {
|
if ($asset->save()) {
|
||||||
|
|
||||||
event(new CheckoutableCheckedIn($asset, $target, auth()->user(), $request->input('note'), $checkin_at, $originalValues));
|
event(new CheckoutableCheckedIn($asset, $target, auth()->user(), $request->input('note'), $checkin_at, $originalValues));
|
||||||
|
|
|
@ -26,27 +26,25 @@ class AssetCheckoutController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
*/
|
*/
|
||||||
public function create($assetId) : View | RedirectResponse
|
public function create(Asset $asset) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
// Check if the asset exists
|
|
||||||
if (is_null($asset = Asset::with('company')->find(e($assetId)))) {
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('checkout', $asset);
|
$this->authorize('checkout', $asset);
|
||||||
|
|
||||||
if (!$asset->model) {
|
if (!$asset->model) {
|
||||||
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix'));
|
return redirect()->route('hardware.show', $asset)
|
||||||
|
->with('error', trans('admin/hardware/general.model_invalid_fix'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($asset->availableForCheckout()) {
|
if ($asset->availableForCheckout()) {
|
||||||
return view('hardware/checkout', compact('asset'))
|
return view('hardware/checkout', compact('asset'))
|
||||||
->with('statusLabel_list', Helper::deployableStatusLabelList())
|
->with('statusLabel_list', Helper::deployableStatusLabelList())
|
||||||
->with('table_name', 'Assets');
|
->with('table_name', 'Assets')
|
||||||
|
->with('item', $asset);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return redirect()->route('hardware.index')
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.checkout.not_available'));
|
->with('error', trans('admin/hardware/message.checkout.not_available'));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -68,7 +66,7 @@ class AssetCheckoutController extends Controller
|
||||||
$this->authorize('checkout', $asset);
|
$this->authorize('checkout', $asset);
|
||||||
|
|
||||||
if (!$asset->model) {
|
if (!$asset->model) {
|
||||||
return redirect()->route('hardware.show', $asset->id)->with('error', trans('admin/hardware/general.model_invalid_fix'));
|
return redirect()->route('hardware.show', $asset)->with('error', trans('admin/hardware/general.model_invalid_fix'));
|
||||||
}
|
}
|
||||||
|
|
||||||
$admin = auth()->user();
|
$admin = auth()->user();
|
||||||
|
@ -91,6 +89,7 @@ class AssetCheckoutController extends Controller
|
||||||
$asset->status_id = $request->get('status_id');
|
$asset->status_id = $request->get('status_id');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if(!empty($asset->licenseseats->all())){
|
if(!empty($asset->licenseseats->all())){
|
||||||
if(request('checkout_to_type') == 'user') {
|
if(request('checkout_to_type') == 'user') {
|
||||||
foreach ($asset->licenseseats as $seat){
|
foreach ($asset->licenseseats as $seat){
|
||||||
|
@ -100,12 +99,15 @@ class AssetCheckoutController extends Controller
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Add any custom fields that should be included in the checkout
|
||||||
|
$asset->customFieldsForCheckinCheckout('display_checkout');
|
||||||
|
|
||||||
$settings = \App\Models\Setting::getSettings();
|
$settings = \App\Models\Setting::getSettings();
|
||||||
|
|
||||||
// We have to check whether $target->company_id is null here since locations don't have a company yet
|
// We have to check whether $target->company_id is null here since locations don't have a company yet
|
||||||
if (($settings->full_multiple_companies_support) && ((!is_null($target->company_id)) && (!is_null($asset->company_id)))) {
|
if (($settings->full_multiple_companies_support) && ((!is_null($target->company_id)) && (!is_null($asset->company_id)))) {
|
||||||
if ($target->company_id != $asset->company_id){
|
if ($target->company_id != $asset->company_id){
|
||||||
return redirect()->to("hardware/$assetId/checkout")->with('error', trans('general.error_user_company'));
|
return redirect()->route('hardware.checkout.create', $asset)->with('error', trans('general.error_user_company'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -116,7 +118,7 @@ class AssetCheckoutController extends Controller
|
||||||
->with('success', trans('admin/hardware/message.checkout.success'));
|
->with('success', trans('admin/hardware/message.checkout.success'));
|
||||||
}
|
}
|
||||||
// Redirect to the asset management page with error
|
// Redirect to the asset management page with error
|
||||||
return redirect()->to("hardware/$assetId/checkout")->with('error', trans('admin/hardware/message.checkout.error').$asset->getErrors());
|
return redirect()->route("hardware.checkout.create", $asset)->with('error', trans('admin/hardware/message.checkout.error').$asset->getErrors());
|
||||||
} catch (ModelNotFoundException $e) {
|
} catch (ModelNotFoundException $e) {
|
||||||
return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'))->withErrors($asset->getErrors());
|
return redirect()->back()->with('error', trans('admin/hardware/message.checkout.error'))->withErrors($asset->getErrors());
|
||||||
} catch (CheckoutNotAllowed $e) {
|
} catch (CheckoutNotAllowed $e) {
|
||||||
|
|
|
@ -26,11 +26,8 @@ class AssetFilesController extends Controller
|
||||||
*@since [v1.0]
|
*@since [v1.0]
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
*/
|
*/
|
||||||
public function store(UploadFileRequest $request, $assetId = null) : RedirectResponse
|
public function store(UploadFileRequest $request, Asset $asset) : RedirectResponse
|
||||||
{
|
{
|
||||||
if (! $asset = Asset::find($assetId)) {
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $asset);
|
$this->authorize('update', $asset);
|
||||||
|
|
||||||
|
@ -59,9 +56,8 @@ class AssetFilesController extends Controller
|
||||||
* @param int $fileId
|
* @param int $fileId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function show($assetId = null, $fileId = null) : View | RedirectResponse | Response | StreamedResponse | BinaryFileResponse
|
public function show(Asset $asset, $fileId = null) : View | RedirectResponse | Response | StreamedResponse | BinaryFileResponse
|
||||||
{
|
{
|
||||||
if ($asset = Asset::find($assetId)) {
|
|
||||||
|
|
||||||
$this->authorize('view', $asset);
|
$this->authorize('view', $asset);
|
||||||
|
|
||||||
|
@ -75,15 +71,13 @@ class AssetFilesController extends Controller
|
||||||
try {
|
try {
|
||||||
return StorageHelper::showOrDownloadFile($file, $log->filename);
|
return StorageHelper::showOrDownloadFile($file, $log->filename);
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.file_not_found'));
|
return redirect()->route('hardware.show', $asset)->with('error', trans('general.file_not_found'));
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.log_record_not_found'));
|
return redirect()->route('hardware.show', $asset)->with('error', trans('general.log_record_not_found'));
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -95,9 +89,8 @@ class AssetFilesController extends Controller
|
||||||
* @param int $fileId
|
* @param int $fileId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function destroy($assetId = null, $fileId = null) : RedirectResponse
|
public function destroy(Asset $asset, $fileId = null) : RedirectResponse
|
||||||
{
|
{
|
||||||
if ($asset = Asset::find($assetId)) {
|
|
||||||
$this->authorize('update', $asset);
|
$this->authorize('update', $asset);
|
||||||
$rel_path = 'private_uploads/assets';
|
$rel_path = 'private_uploads/assets';
|
||||||
|
|
||||||
|
@ -109,9 +102,7 @@ class AssetFilesController extends Controller
|
||||||
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
|
return redirect()->back()->withFragment('files')->with('success', trans('admin/hardware/message.deletefile.success'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('hardware.show', ['hardware' => $asset])->with('error', trans('general.log_record_not_found'));
|
return redirect()->route('hardware.show', $asset)->with('error', trans('general.log_record_not_found'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -30,6 +30,7 @@ use Illuminate\Http\Response;
|
||||||
use Illuminate\Contracts\View\View;
|
use Illuminate\Contracts\View\View;
|
||||||
use Illuminate\Http\RedirectResponse;
|
use Illuminate\Http\RedirectResponse;
|
||||||
use Symfony\Component\HttpFoundation\BinaryFileResponse;
|
use Symfony\Component\HttpFoundation\BinaryFileResponse;
|
||||||
|
use TypeError;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class controls all actions related to assets for
|
* This class controls all actions related to assets for
|
||||||
|
@ -201,7 +202,7 @@ class AssetsController extends Controller
|
||||||
$asset->checkOut($target, auth()->user(), date('Y-m-d H:i:s'), $request->input('expected_checkin', null), 'Checked out on asset creation', $request->get('name'), $location);
|
$asset->checkOut($target, auth()->user(), date('Y-m-d H:i:s'), $request->input('expected_checkin', null), 'Checked out on asset creation', $request->get('name'), $location);
|
||||||
}
|
}
|
||||||
|
|
||||||
$successes[] = "<a href='" . route('hardware.show', ['hardware' => $asset->id]) . "' style='color: white;'>" . e($asset->asset_tag) . "</a>";
|
$successes[] = "<a href='" . route('hardware.show', $asset) . "' style='color: white;'>" . e($asset->asset_tag) . "</a>";
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
$failures[] = join(",", $asset->getErrors()->all());
|
$failures[] = join(",", $asset->getErrors()->all());
|
||||||
|
@ -222,7 +223,7 @@ class AssetsController extends Controller
|
||||||
//the most common case, keeping it so we don't have to make every use of that translation string be trans_choice'ed
|
//the most common case, keeping it so we don't have to make every use of that translation string be trans_choice'ed
|
||||||
//and re-translated
|
//and re-translated
|
||||||
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
|
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
|
||||||
->with('success-unescaped', trans('admin/hardware/message.create.success_linked', ['link' => route('hardware.show', ['hardware' => $asset->id]), 'id', 'tag' => e($asset->asset_tag)]));
|
->with('success-unescaped', trans('admin/hardware/message.create.success_linked', ['link' => route('hardware.show', $asset), 'id', 'tag' => e($asset->asset_tag)]));
|
||||||
} else {
|
} else {
|
||||||
//multi-success
|
//multi-success
|
||||||
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
|
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
|
||||||
|
@ -240,20 +241,14 @@ class AssetsController extends Controller
|
||||||
* Returns a view that presents a form to edit an existing asset.
|
* Returns a view that presents a form to edit an existing asset.
|
||||||
*
|
*
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
* @param int $assetId
|
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
*/
|
*/
|
||||||
public function edit($assetId = null) : View | RedirectResponse
|
public function edit(Asset $asset) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
if (! $item = Asset::find($assetId)) {
|
$this->authorize($asset);
|
||||||
// Redirect to the asset management page with error
|
return view('hardware/edit')
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
->with('item', $asset)
|
||||||
}
|
|
||||||
//Handles company checks and permissions.
|
|
||||||
$this->authorize($item);
|
|
||||||
|
|
||||||
return view('hardware/edit', compact('item'))
|
|
||||||
->with('statuslabel_list', Helper::statusLabelList())
|
->with('statuslabel_list', Helper::statusLabelList())
|
||||||
->with('statuslabel_types', Helper::statusTypeList());
|
->with('statuslabel_types', Helper::statusTypeList());
|
||||||
}
|
}
|
||||||
|
@ -267,15 +262,14 @@ class AssetsController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
*/
|
*/
|
||||||
public function show($assetId = null) : View | RedirectResponse
|
public function show(Asset $asset) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$asset = Asset::withTrashed()->find($assetId);
|
|
||||||
$this->authorize('view', $asset);
|
$this->authorize('view', $asset);
|
||||||
$settings = Setting::getSettings();
|
$settings = Setting::getSettings();
|
||||||
|
|
||||||
if (isset($asset)) {
|
if (isset($asset)) {
|
||||||
$audit_log = Actionlog::where('action_type', '=', 'audit')
|
$audit_log = Actionlog::where('action_type', '=', 'audit')
|
||||||
->where('item_id', '=', $assetId)
|
->where('item_id', '=', $asset->id)
|
||||||
->where('item_type', '=', Asset::class)
|
->where('item_type', '=', Asset::class)
|
||||||
->orderBy('created_at', 'DESC')->first();
|
->orderBy('created_at', 'DESC')->first();
|
||||||
|
|
||||||
|
@ -291,7 +285,7 @@ class AssetsController extends Controller
|
||||||
|
|
||||||
$qr_code = (object) [
|
$qr_code = (object) [
|
||||||
'display' => $settings->qr_code == '1',
|
'display' => $settings->qr_code == '1',
|
||||||
'url' => route('qr_code/hardware', $asset->id),
|
'url' => route('qr_code/hardware', $asset),
|
||||||
];
|
];
|
||||||
|
|
||||||
return view('hardware/view', compact('asset', 'qr_code', 'settings'))
|
return view('hardware/view', compact('asset', 'qr_code', 'settings'))
|
||||||
|
@ -308,14 +302,9 @@ class AssetsController extends Controller
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $assetId = null) : RedirectResponse
|
public function update(ImageUploadRequest $request, Asset $asset) : RedirectResponse
|
||||||
{
|
{
|
||||||
|
|
||||||
// Check if the asset exists
|
|
||||||
if (! $asset = Asset::find($assetId)) {
|
|
||||||
// Redirect to the asset management page with error
|
|
||||||
return redirect()->route('hardware.index')->with('error', trans('admin/hardware/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
$this->authorize($asset);
|
$this->authorize($asset);
|
||||||
|
|
||||||
$asset->status_id = $request->input('status_id', null);
|
$asset->status_id = $request->input('status_id', null);
|
||||||
|
@ -351,15 +340,14 @@ class AssetsController extends Controller
|
||||||
|
|
||||||
$status = Statuslabel::find($request->input('status_id'));
|
$status = Statuslabel::find($request->input('status_id'));
|
||||||
|
|
||||||
// This is a non-deployable status label - we should check the asset back in.
|
// This is an archived or undeployable - we should check the asset back in.
|
||||||
if (($status && $status->getStatuslabelType() != 'deployable') && ($target = $asset->assignedTo)) {
|
// Pending is allowed here
|
||||||
|
if (($status) && (($status->getStatuslabelType() != 'pending') && ($status->getStatuslabelType() != 'deployable')) && ($target = $asset->assignedTo)) {
|
||||||
$originalValues = $asset->getRawOriginal();
|
$originalValues = $asset->getRawOriginal();
|
||||||
$asset->assigned_to = null;
|
$asset->assigned_to = null;
|
||||||
$asset->assigned_type = null;
|
$asset->assigned_type = null;
|
||||||
$asset->accepted = null;
|
$asset->accepted = null;
|
||||||
|
event(new CheckoutableCheckedIn($asset, $target, auth()->user(), 'Checkin on asset update with '.$status->getStatuslabelType().' status', date('Y-m-d H:i:s'), $originalValues));
|
||||||
event(new CheckoutableCheckedIn($asset, $target, auth()->user(), 'Checkin on asset update', date('Y-m-d H:i:s'), $originalValues));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($asset->assigned_to == '') {
|
if ($asset->assigned_to == '') {
|
||||||
|
@ -430,7 +418,7 @@ class AssetsController extends Controller
|
||||||
session()->put(['redirect_option' => $request->get('redirect_option'), 'checkout_to_type' => $request->get('checkout_to_type')]);
|
session()->put(['redirect_option' => $request->get('redirect_option'), 'checkout_to_type' => $request->get('checkout_to_type')]);
|
||||||
|
|
||||||
if ($asset->save()) {
|
if ($asset->save()) {
|
||||||
return redirect()->to(Helper::getRedirectOption($request, $assetId, 'Assets'))
|
return redirect()->to(Helper::getRedirectOption($request, $asset->id, 'Assets'))
|
||||||
->with('success', trans('admin/hardware/message.update.success'));
|
->with('success', trans('admin/hardware/message.update.success'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -531,12 +519,12 @@ class AssetsController extends Controller
|
||||||
* @param int $assetId
|
* @param int $assetId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function getQrCode($assetId = null) : Response | BinaryFileResponse | string | bool
|
public function getQrCode(Asset $asset) : Response | BinaryFileResponse | string | bool
|
||||||
{
|
{
|
||||||
$settings = Setting::getSettings();
|
$settings = Setting::getSettings();
|
||||||
|
|
||||||
if (($settings->qr_code === '1') && ($settings->label2_2d_type !== 'none')) {
|
if (($settings->qr_code == '1') && ($settings->label2_2d_type !== 'none')) {
|
||||||
$asset = Asset::withTrashed()->find($assetId);
|
|
||||||
if ($asset) {
|
if ($asset) {
|
||||||
$size = Helper::barcodeDimensions($settings->label2_2d_type);
|
$size = Helper::barcodeDimensions($settings->label2_2d_type);
|
||||||
$qr_file = public_path().'/uploads/barcodes/qr-'.str_slug($asset->asset_tag).'-'.str_slug($asset->id).'.png';
|
$qr_file = public_path().'/uploads/barcodes/qr-'.str_slug($asset->asset_tag).'-'.str_slug($asset->id).'.png';
|
||||||
|
@ -590,7 +578,7 @@ class AssetsController extends Controller
|
||||||
file_put_contents($barcode_file, $barcode_obj->getPngData());
|
file_put_contents($barcode_file, $barcode_obj->getPngData());
|
||||||
|
|
||||||
return response($barcode_obj->getPngData())->header('Content-type', 'image/png');
|
return response($barcode_obj->getPngData())->header('Content-type', 'image/png');
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception|TypeError $e) {
|
||||||
Log::debug('The barcode format is invalid.');
|
Log::debug('The barcode format is invalid.');
|
||||||
|
|
||||||
return response(file_get_contents(public_path('uploads/barcodes/invalid_barcode.gif')))->header('Content-type', 'image/gif');
|
return response(file_get_contents(public_path('uploads/barcodes/invalid_barcode.gif')))->header('Content-type', 'image/gif');
|
||||||
|
@ -877,12 +865,11 @@ class AssetsController extends Controller
|
||||||
return view('hardware/quickscan-checkin')->with('statusLabel_list', Helper::statusLabelList());
|
return view('hardware/quickscan-checkin')->with('statusLabel_list', Helper::statusLabelList());
|
||||||
}
|
}
|
||||||
|
|
||||||
public function audit($id)
|
public function audit(Asset $asset)
|
||||||
{
|
{
|
||||||
$settings = Setting::getSettings();
|
$settings = Setting::getSettings();
|
||||||
$this->authorize('audit', Asset::class);
|
$this->authorize('audit', Asset::class);
|
||||||
$dt = Carbon::now()->addMonths($settings->audit_interval)->toDateString();
|
$dt = Carbon::now()->addMonths($settings->audit_interval)->toDateString();
|
||||||
$asset = Asset::findOrFail($id);
|
|
||||||
return view('hardware/audit')->with('asset', $asset)->with('next_audit_date', $dt)->with('locations_list');
|
return view('hardware/audit')->with('asset', $asset)->with('next_audit_date', $dt)->with('locations_list');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -901,7 +888,7 @@ class AssetsController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
public function auditStore(UploadFileRequest $request, $id)
|
public function auditStore(UploadFileRequest $request, Asset $asset)
|
||||||
{
|
{
|
||||||
$this->authorize('audit', Asset::class);
|
$this->authorize('audit', Asset::class);
|
||||||
|
|
||||||
|
@ -916,8 +903,6 @@ class AssetsController extends Controller
|
||||||
return response()->json(Helper::formatStandardApiResponse('error', null, $validator->errors()->all()));
|
return response()->json(Helper::formatStandardApiResponse('error', null, $validator->errors()->all()));
|
||||||
}
|
}
|
||||||
|
|
||||||
$asset = Asset::findOrFail($id);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Even though we do a save() further down, we don't want to log this as a "normal" asset update,
|
* Even though we do a save() further down, we don't want to log this as a "normal" asset update,
|
||||||
* which would trigger the Asset Observer and would log an asset *update* log entry (because the
|
* which would trigger the Asset Observer and would log an asset *update* log entry (because the
|
||||||
|
|
|
@ -358,7 +358,11 @@ class BulkAssetsController extends Controller
|
||||||
* to someone/something.
|
* to someone/something.
|
||||||
*/
|
*/
|
||||||
if ($request->filled('status_id')) {
|
if ($request->filled('status_id')) {
|
||||||
$updated_status = Statuslabel::find($request->input('status_id'));
|
try {
|
||||||
|
$updated_status = Statuslabel::findOrFail($request->input('status_id'));
|
||||||
|
} catch (ModelNotFoundException $e) {
|
||||||
|
return redirect($bulk_back_url)->with('error', trans('admin/statuslabels/message.does_not_exist'));
|
||||||
|
}
|
||||||
|
|
||||||
// We cannot assign a non-deployable status type if the asset is already assigned.
|
// We cannot assign a non-deployable status type if the asset is already assigned.
|
||||||
// This could probably be added to a form request.
|
// This could probably be added to a form request.
|
||||||
|
@ -366,7 +370,7 @@ class BulkAssetsController extends Controller
|
||||||
// Otherwise we need to make sure the status type is still a deployable one.
|
// Otherwise we need to make sure the status type is still a deployable one.
|
||||||
if (
|
if (
|
||||||
($asset->assigned_to == '')
|
($asset->assigned_to == '')
|
||||||
|| ($updated_status->deployable == '1') && ($asset->assetstatus->deployable == '1')
|
|| ($updated_status->deployable == '1') && ($asset->assetstatus?->deployable == '1')
|
||||||
) {
|
) {
|
||||||
$this->update_array['status_id'] = $updated_status->id;
|
$this->update_array['status_id'] = $updated_status->id;
|
||||||
}
|
}
|
||||||
|
@ -525,21 +529,31 @@ class BulkAssetsController extends Controller
|
||||||
$this->authorize('delete', Asset::class);
|
$this->authorize('delete', Asset::class);
|
||||||
|
|
||||||
$bulk_back_url = route('hardware.index');
|
$bulk_back_url = route('hardware.index');
|
||||||
|
|
||||||
if ($request->session()->has('bulk_back_url')) {
|
if ($request->session()->has('bulk_back_url')) {
|
||||||
$bulk_back_url = $request->session()->pull('bulk_back_url');
|
$bulk_back_url = $request->session()->pull('bulk_back_url');
|
||||||
}
|
}
|
||||||
|
$assetIds = $request->get('ids');
|
||||||
|
|
||||||
if ($request->filled('ids')) {
|
if(empty($assetIds)) {
|
||||||
$assets = Asset::find($request->get('ids'));
|
return redirect($bulk_back_url)->with('error', trans('admin/hardware/message.delete.nothing_updated'));
|
||||||
foreach ($assets as $asset) {
|
}
|
||||||
|
|
||||||
|
$assignedAssets = Asset::whereIn('id', $assetIds)->whereNotNull('assigned_to')->get();
|
||||||
|
if($assignedAssets->isNotEmpty()) {
|
||||||
|
|
||||||
|
//if assets are checked out, return a list of asset tags that would need to be checked in first.
|
||||||
|
$assetTags = $assignedAssets->pluck('asset_tag')->implode(', ');
|
||||||
|
return redirect($bulk_back_url)->with('error', trans_choice('admin/hardware/message.delete.assigned_to_error', $assignedAssets->count(), ['asset_tag' => $assetTags] ));
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach (Asset::wherein('id', $assetIds)->get() as $asset) {
|
||||||
$asset->delete();
|
$asset->delete();
|
||||||
} // endforeach
|
}
|
||||||
|
|
||||||
return redirect($bulk_back_url)->with('success', trans('admin/hardware/message.delete.success'));
|
return redirect($bulk_back_url)->with('success', trans('admin/hardware/message.delete.success'));
|
||||||
// no values given, nothing to update
|
// no values given, nothing to update
|
||||||
}
|
|
||||||
|
|
||||||
return redirect($bulk_back_url)->with('error', trans('admin/hardware/message.delete.nothing_updated'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -206,6 +206,7 @@ class LoginController extends Controller
|
||||||
$user->password = bcrypt($request->input('password'));
|
$user->password = bcrypt($request->input('password'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
$user->last_login = \Carbon::now();
|
||||||
$user->email = $ldap_attr['email'];
|
$user->email = $ldap_attr['email'];
|
||||||
$user->first_name = $ldap_attr['firstname'];
|
$user->first_name = $ldap_attr['firstname'];
|
||||||
$user->last_name = $ldap_attr['lastname']; //FIXME (or TODO?) - do we need to map additional fields that we now support? E.g. country, phone, etc.
|
$user->last_name = $ldap_attr['lastname']; //FIXME (or TODO?) - do we need to map additional fields that we now support? E.g. country, phone, etc.
|
||||||
|
@ -432,6 +433,7 @@ class LoginController extends Controller
|
||||||
|
|
||||||
if (Google2FA::verifyKey($user->two_factor_secret, $secret)) {
|
if (Google2FA::verifyKey($user->two_factor_secret, $secret)) {
|
||||||
$user->two_factor_enrolled = 1;
|
$user->two_factor_enrolled = 1;
|
||||||
|
$user->last_login = \Carbon::now();
|
||||||
$user->saveQuietly();
|
$user->saveQuietly();
|
||||||
$request->session()->put('2fa_authed', $user->id);
|
$request->session()->put('2fa_authed', $user->id);
|
||||||
|
|
||||||
|
|
|
@ -69,6 +69,7 @@ class CategoriesController extends Controller
|
||||||
$category->use_default_eula = $request->input('use_default_eula', '0');
|
$category->use_default_eula = $request->input('use_default_eula', '0');
|
||||||
$category->require_acceptance = $request->input('require_acceptance', '0');
|
$category->require_acceptance = $request->input('require_acceptance', '0');
|
||||||
$category->checkin_email = $request->input('checkin_email', '0');
|
$category->checkin_email = $request->input('checkin_email', '0');
|
||||||
|
$category->notes = $request->input('notes');
|
||||||
$category->created_by = auth()->id();
|
$category->created_by = auth()->id();
|
||||||
|
|
||||||
$category = $request->handleImages($category);
|
$category = $request->handleImages($category);
|
||||||
|
@ -87,14 +88,10 @@ class CategoriesController extends Controller
|
||||||
* @param int $categoryId
|
* @param int $categoryId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($categoryId = null) : RedirectResponse | View
|
public function edit(Category $category) : RedirectResponse | View
|
||||||
{
|
{
|
||||||
$this->authorize('update', Category::class);
|
$this->authorize('update', Category::class);
|
||||||
if (is_null($item = Category::find($categoryId))) {
|
return view('categories/edit')->with('item', $category)
|
||||||
return redirect()->route('categories.index')->with('error', trans('admin/categories/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
return view('categories/edit', compact('item'))
|
|
||||||
->with('category_types', Helper::categoryTypeList());
|
->with('category_types', Helper::categoryTypeList());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,19 +104,10 @@ class CategoriesController extends Controller
|
||||||
* @param int $categoryId
|
* @param int $categoryId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $categoryId = null) : RedirectResponse
|
public function update(ImageUploadRequest $request, Category $category) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Category::class);
|
$this->authorize('update', Category::class);
|
||||||
if (is_null($category = Category::find($categoryId))) {
|
|
||||||
// Redirect to the categories management page
|
|
||||||
return redirect()->route('categories.index')->with('error', trans('admin/categories/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the category data
|
|
||||||
$category->name = $request->input('name');
|
$category->name = $request->input('name');
|
||||||
// If the item count is > 0, we disable the category type in the edit. Disabled items
|
|
||||||
// don't POST, so if the category_type is blank we just set it to the default.
|
|
||||||
|
|
||||||
|
|
||||||
// Don't allow the user to change the category_type once it's been created
|
// Don't allow the user to change the category_type once it's been created
|
||||||
if (($request->filled('category_type') && ($category->itemCount() > 0))) {
|
if (($request->filled('category_type') && ($category->itemCount() > 0))) {
|
||||||
|
@ -134,6 +122,7 @@ class CategoriesController extends Controller
|
||||||
$category->use_default_eula = $request->input('use_default_eula', '0');
|
$category->use_default_eula = $request->input('use_default_eula', '0');
|
||||||
$category->require_acceptance = $request->input('require_acceptance', '0');
|
$category->require_acceptance = $request->input('require_acceptance', '0');
|
||||||
$category->checkin_email = $request->input('checkin_email', '0');
|
$category->checkin_email = $request->input('checkin_email', '0');
|
||||||
|
$category->notes = $request->input('notes');
|
||||||
|
|
||||||
$category = $request->handleImages($category);
|
$category = $request->handleImages($category);
|
||||||
|
|
||||||
|
@ -179,10 +168,10 @@ class CategoriesController extends Controller
|
||||||
* @param $id
|
* @param $id
|
||||||
* @since [v1.8]
|
* @since [v1.8]
|
||||||
*/
|
*/
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(Category $category) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', Category::class);
|
$this->authorize('view', Category::class);
|
||||||
if ($category = Category::find($id)) {
|
|
||||||
if ($category->category_type == 'asset') {
|
if ($category->category_type == 'asset') {
|
||||||
$category_type = 'hardware';
|
$category_type = 'hardware';
|
||||||
$category_type_route = 'assets';
|
$category_type_route = 'assets';
|
||||||
|
@ -198,7 +187,4 @@ class CategoriesController extends Controller
|
||||||
->with('category_type', $category_type)
|
->with('category_type', $category_type)
|
||||||
->with('category_type_route', $category_type_route);
|
->with('category_type_route', $category_type_route);
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('categories.index')->with('error', trans('admin/categories/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -60,6 +60,7 @@ final class CompaniesController extends Controller
|
||||||
$company->phone = $request->input('phone');
|
$company->phone = $request->input('phone');
|
||||||
$company->fax = $request->input('fax');
|
$company->fax = $request->input('fax');
|
||||||
$company->email = $request->input('email');
|
$company->email = $request->input('email');
|
||||||
|
$company->notes = $request->input('notes');
|
||||||
$company->created_by = auth()->id();
|
$company->created_by = auth()->id();
|
||||||
|
|
||||||
$company = $request->handleImages($company);
|
$company = $request->handleImages($company);
|
||||||
|
@ -79,16 +80,10 @@ final class CompaniesController extends Controller
|
||||||
* @since [v1.8]
|
* @since [v1.8]
|
||||||
* @param int $companyId
|
* @param int $companyId
|
||||||
*/
|
*/
|
||||||
public function edit($companyId) : View | RedirectResponse
|
public function edit(Company $company) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
if (is_null($item = Company::find($companyId))) {
|
$this->authorize('update', $company);
|
||||||
return redirect()->route('companies.index')
|
return view('companies/edit')->with('item', $company);
|
||||||
->with('error', trans('admin/companies/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $item);
|
|
||||||
|
|
||||||
return view('companies/edit')->with('item', $item);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -99,18 +94,15 @@ final class CompaniesController extends Controller
|
||||||
* @param ImageUploadRequest $request
|
* @param ImageUploadRequest $request
|
||||||
* @param int $companyId
|
* @param int $companyId
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $companyId) : RedirectResponse
|
public function update(ImageUploadRequest $request, Company $company) : RedirectResponse
|
||||||
{
|
{
|
||||||
if (is_null($company = Company::find($companyId))) {
|
|
||||||
return redirect()->route('companies.index')->with('error', trans('admin/companies/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $company);
|
$this->authorize('update', $company);
|
||||||
|
|
||||||
$company->name = $request->input('name');
|
$company->name = $request->input('name');
|
||||||
$company->phone = $request->input('phone');
|
$company->phone = $request->input('phone');
|
||||||
$company->fax = $request->input('fax');
|
$company->fax = $request->input('fax');
|
||||||
$company->email = $request->input('email');
|
$company->email = $request->input('email');
|
||||||
|
$company->notes = $request->input('notes');
|
||||||
|
|
||||||
$company = $request->handleImages($company);
|
$company = $request->handleImages($company);
|
||||||
|
|
||||||
|
@ -156,15 +148,9 @@ final class CompaniesController extends Controller
|
||||||
->with('success', trans('admin/companies/message.delete.success'));
|
->with('success', trans('admin/companies/message.delete.success'));
|
||||||
}
|
}
|
||||||
|
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(Company $company) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', Company::class);
|
$this->authorize('view', Company::class);
|
||||||
|
|
||||||
if (is_null($company = Company::find($id))) {
|
|
||||||
return redirect()->route('companies.index')
|
|
||||||
->with('error', trans('admin/companies/message.not_found'));
|
|
||||||
}
|
|
||||||
|
|
||||||
return view('companies/view')->with('company', $company);
|
return view('companies/view')->with('company', $company);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -107,15 +107,13 @@ class ComponentsController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function edit($componentId = null)
|
public function edit(Component $component)
|
||||||
{
|
{
|
||||||
if ($item = Component::find($componentId)) {
|
|
||||||
$this->authorize('update', $item);
|
|
||||||
|
|
||||||
return view('components/edit', compact('item'))->with('category_type', 'component');
|
$this->authorize('update', $component);
|
||||||
}
|
return view('components/edit')
|
||||||
|
->with('item', $component)
|
||||||
return redirect()->route('components.index')->with('error', trans('admin/components/message.does_not_exist'));
|
->with('category_type', 'component');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -130,11 +128,8 @@ class ComponentsController extends Controller
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
* @since [v3.0]
|
* @since [v3.0]
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $componentId = null)
|
public function update(ImageUploadRequest $request, Component $component)
|
||||||
{
|
{
|
||||||
if (is_null($component = Component::find($componentId))) {
|
|
||||||
return redirect()->route('components.index')->with('error', trans('admin/components/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
$min = $component->numCheckedOut();
|
$min = $component->numCheckedOut();
|
||||||
$validator = Validator::make($request->all(), [
|
$validator = Validator::make($request->all(), [
|
||||||
'qty' => "required|numeric|min:$min",
|
'qty' => "required|numeric|min:$min",
|
||||||
|
@ -201,6 +196,10 @@ class ComponentsController extends Controller
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($component->numCheckedOut() > 0) {
|
||||||
|
return redirect()->route('components.index')->with('error', trans('admin/components/message.delete.error_qty'));
|
||||||
|
}
|
||||||
|
|
||||||
$component->delete();
|
$component->delete();
|
||||||
|
|
||||||
return redirect()->route('components.index')->with('success', trans('admin/components/message.delete.success'));
|
return redirect()->route('components.index')->with('success', trans('admin/components/message.delete.success'));
|
||||||
|
@ -216,17 +215,9 @@ class ComponentsController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function show($componentId = null)
|
public function show(Component $component)
|
||||||
{
|
{
|
||||||
$component = Component::find($componentId);
|
|
||||||
|
|
||||||
if (isset($component->id)) {
|
|
||||||
$this->authorize('view', $component);
|
$this->authorize('view', $component);
|
||||||
|
|
||||||
return view('components/view', compact('component'));
|
return view('components/view', compact('component'));
|
||||||
}
|
}
|
||||||
// Redirect to the user management page
|
|
||||||
return redirect()->route('components.index')
|
|
||||||
->with('error', trans('admin/components/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -104,15 +104,13 @@ class ConsumablesController extends Controller
|
||||||
* @see ConsumablesController::postEdit() method that stores the form data.
|
* @see ConsumablesController::postEdit() method that stores the form data.
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($consumableId = null) : View | RedirectResponse
|
public function edit(Consumable $consumable) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
if ($item = Consumable::find($consumableId)) {
|
$this->authorize($consumable);
|
||||||
$this->authorize($item);
|
return view('consumables/edit')
|
||||||
|
->with('item', $consumable)
|
||||||
|
->with('category_type', 'consumable');
|
||||||
|
|
||||||
return view('consumables/edit', compact('item'))->with('category_type', 'consumable');
|
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route('consumables.index')->with('error', trans('admin/consumables/message.does_not_exist'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -126,11 +124,8 @@ class ConsumablesController extends Controller
|
||||||
* @see ConsumablesController::getEdit() method that stores the form data.
|
* @see ConsumablesController::getEdit() method that stores the form data.
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(StoreConsumableRequest $request, $consumableId = null)
|
public function update(StoreConsumableRequest $request, Consumable $consumable)
|
||||||
{
|
{
|
||||||
if (is_null($consumable = Consumable::find($consumableId))) {
|
|
||||||
return redirect()->route('consumables.index')->with('error', trans('admin/consumables/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$min = $consumable->numCheckedOut();
|
$min = $consumable->numCheckedOut();
|
||||||
$validator = Validator::make($request->all(), [
|
$validator = Validator::make($request->all(), [
|
||||||
|
@ -202,18 +197,13 @@ class ConsumablesController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function show($consumableId = null)
|
public function show(Consumable $consumable)
|
||||||
{
|
{
|
||||||
$consumable = Consumable::withCount('users as users_consumables')->find($consumableId);
|
$consumable = Consumable::withCount('users as users_consumables')->find($consumable->id);
|
||||||
$this->authorize($consumable);
|
$this->authorize($consumable);
|
||||||
if (isset($consumable->id)) {
|
|
||||||
return view('consumables/view', compact('consumable'));
|
return view('consumables/view', compact('consumable'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('consumables.index')
|
|
||||||
->with('error', trans('admin/consumables/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
public function clone(Consumable $consumable) : View
|
public function clone(Consumable $consumable) : View
|
||||||
{
|
{
|
||||||
$this->authorize('create', $consumable);
|
$this->authorize('create', $consumable);
|
||||||
|
|
|
@ -104,6 +104,8 @@ class CustomFieldsController extends Controller
|
||||||
"auto_add_to_fieldsets" => $request->get("auto_add_to_fieldsets", 0),
|
"auto_add_to_fieldsets" => $request->get("auto_add_to_fieldsets", 0),
|
||||||
"show_in_listview" => $request->get("show_in_listview", 0),
|
"show_in_listview" => $request->get("show_in_listview", 0),
|
||||||
"show_in_requestable_list" => $request->get("show_in_requestable_list", 0),
|
"show_in_requestable_list" => $request->get("show_in_requestable_list", 0),
|
||||||
|
"display_checkin" => $request->get("display_checkin", 0),
|
||||||
|
"display_checkout" => $request->get("display_checkout", 0),
|
||||||
"created_by" => auth()->id()
|
"created_by" => auth()->id()
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
@ -193,10 +195,8 @@ class CustomFieldsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v4.0]
|
* @since [v4.0]
|
||||||
*/
|
*/
|
||||||
public function edit(Request $request, $id) : View | RedirectResponse
|
public function edit(Request $request, CustomField $field) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
if ($field = CustomField::find($id)) {
|
|
||||||
|
|
||||||
$this->authorize('update', $field);
|
$this->authorize('update', $field);
|
||||||
$fieldsets = CustomFieldset::get();
|
$fieldsets = CustomFieldset::get();
|
||||||
$customFormat = '';
|
$customFormat = '';
|
||||||
|
@ -210,10 +210,6 @@ class CustomFieldsController extends Controller
|
||||||
'fieldsets' => $fieldsets,
|
'fieldsets' => $fieldsets,
|
||||||
'predefinedFormats' => Helper::predefined_formats(),
|
'predefinedFormats' => Helper::predefined_formats(),
|
||||||
]);
|
]);
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route("fields.index")
|
|
||||||
->with("error", trans('admin/custom_fields/message.field.invalid'));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -229,13 +225,9 @@ class CustomFieldsController extends Controller
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function update(CustomFieldRequest $request, $id) : RedirectResponse
|
public function update(CustomFieldRequest $request, CustomField $field) : RedirectResponse
|
||||||
{
|
{
|
||||||
$field = CustomField::find($id);
|
|
||||||
|
|
||||||
$this->authorize('update', $field);
|
$this->authorize('update', $field);
|
||||||
|
|
||||||
|
|
||||||
$show_in_email = $request->get("show_in_email", 0);
|
$show_in_email = $request->get("show_in_email", 0);
|
||||||
$display_in_user_view = $request->get("display_in_user_view", 0);
|
$display_in_user_view = $request->get("display_in_user_view", 0);
|
||||||
|
|
||||||
|
@ -256,6 +248,8 @@ class CustomFieldsController extends Controller
|
||||||
$field->auto_add_to_fieldsets = $request->get("auto_add_to_fieldsets", 0);
|
$field->auto_add_to_fieldsets = $request->get("auto_add_to_fieldsets", 0);
|
||||||
$field->show_in_listview = $request->get("show_in_listview", 0);
|
$field->show_in_listview = $request->get("show_in_listview", 0);
|
||||||
$field->show_in_requestable_list = $request->get("show_in_requestable_list", 0);
|
$field->show_in_requestable_list = $request->get("show_in_requestable_list", 0);
|
||||||
|
$field->display_checkin = $request->get("display_checkin", 0);
|
||||||
|
$field->display_checkout = $request->get("display_checkout", 0);
|
||||||
|
|
||||||
if ($request->get('format') == 'CUSTOM REGEX') {
|
if ($request->get('format') == 'CUSTOM REGEX') {
|
||||||
$field->format = e($request->get('custom_format'));
|
$field->format = e($request->get('custom_format'));
|
||||||
|
|
|
@ -35,10 +35,12 @@ class CustomFieldsetsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v1.8]
|
* @since [v1.8]
|
||||||
*/
|
*/
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(CustomFieldset $fieldset) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$cfset = CustomFieldset::with('fields')
|
$cfset = CustomFieldset::with('fields')
|
||||||
->where('id', '=', $id)->orderBy('id', 'ASC')->first();
|
->where('id', '=', $fieldset->id)
|
||||||
|
->orderBy('id', 'ASC')
|
||||||
|
->first();
|
||||||
|
|
||||||
$this->authorize('view', $cfset);
|
$this->authorize('view', $cfset);
|
||||||
|
|
||||||
|
@ -122,18 +124,12 @@ class CustomFieldsetsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v6.0.14]
|
* @since [v6.0.14]
|
||||||
*/
|
*/
|
||||||
public function edit($id) : View | RedirectResponse
|
public function edit(CustomFieldset $fieldset) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('create', CustomField::class);
|
$this->authorize('create', CustomField::class);
|
||||||
|
|
||||||
if ($fieldset = CustomFieldset::find($id)) {
|
|
||||||
return view('custom_fields.fieldsets.edit')->with('item', $fieldset);
|
return view('custom_fields.fieldsets.edit')->with('item', $fieldset);
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('fields.index')->with('error', trans('admin/custom_fields/general.fieldset_does_not_exist', ['id' => $id]));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Saves updated fieldset data
|
* Saves updated fieldset data
|
||||||
*
|
*
|
||||||
|
@ -141,12 +137,10 @@ class CustomFieldsetsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v6.0.14]
|
* @since [v6.0.14]
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, $id) : RedirectResponse
|
public function update(Request $request, CustomFieldset $fieldset) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('create', CustomField::class);
|
$this->authorize('create', CustomField::class);
|
||||||
|
|
||||||
if ($fieldset = CustomFieldset::find($id)) {
|
|
||||||
|
|
||||||
$fieldset->name = $request->input('name');
|
$fieldset->name = $request->input('name');
|
||||||
|
|
||||||
if ($fieldset->save()) {
|
if ($fieldset->save()) {
|
||||||
|
@ -157,9 +151,6 @@ class CustomFieldsetsController extends Controller
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('fields.index')->with('error', trans('admin/custom_fields/general.fieldset_does_not_exist', ['id' => $id]));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates a custom fieldset and then deletes if it has no models associated.
|
* Validates a custom fieldset and then deletes if it has no models associated.
|
||||||
*
|
*
|
||||||
|
|
|
@ -40,7 +40,7 @@ class DashboardController extends Controller
|
||||||
|
|
||||||
if ((! file_exists(storage_path().'/oauth-private.key')) || (! file_exists(storage_path().'/oauth-public.key'))) {
|
if ((! file_exists(storage_path().'/oauth-private.key')) || (! file_exists(storage_path().'/oauth-public.key'))) {
|
||||||
Artisan::call('migrate', ['--force' => true]);
|
Artisan::call('migrate', ['--force' => true]);
|
||||||
\Artisan::call('passport:install');
|
Artisan::call('passport:install', ['--no-interaction' => true]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return view('dashboard')->with('asset_stats', $asset_stats)->with('counts', $counts);
|
return view('dashboard')->with('asset_stats', $asset_stats)->with('counts', $counts);
|
||||||
|
|
|
@ -55,6 +55,7 @@ class DepartmentsController extends Controller
|
||||||
$department->manager_id = ($request->filled('manager_id') ? $request->input('manager_id') : null);
|
$department->manager_id = ($request->filled('manager_id') ? $request->input('manager_id') : null);
|
||||||
$department->location_id = ($request->filled('location_id') ? $request->input('location_id') : null);
|
$department->location_id = ($request->filled('location_id') ? $request->input('location_id') : null);
|
||||||
$department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null);
|
$department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null);
|
||||||
|
$department->notes = $request->input('notes');
|
||||||
$department = $request->handleImages($department);
|
$department = $request->handleImages($department);
|
||||||
|
|
||||||
if ($department->save()) {
|
if ($department->save()) {
|
||||||
|
@ -72,19 +73,12 @@ class DepartmentsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v4.0]
|
* @since [v4.0]
|
||||||
*/
|
*/
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(Department $department) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$department = Department::find($id);
|
|
||||||
|
|
||||||
$this->authorize('view', $department);
|
$this->authorize('view', $department);
|
||||||
|
|
||||||
if (isset($department->id)) {
|
|
||||||
return view('departments/view', compact('department'));
|
return view('departments/view', compact('department'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('departments.index')->with('error', trans('admin/departments/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns a form view used to create a new department.
|
* Returns a form view used to create a new department.
|
||||||
*
|
*
|
||||||
|
@ -138,15 +132,10 @@ class DepartmentsController extends Controller
|
||||||
* @param int $departmentId
|
* @param int $departmentId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($departmentId = null) : View | RedirectResponse
|
public function edit(Department $department) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
if (is_null($item = Department::find($departmentId))) {
|
$this->authorize('update', $department);
|
||||||
return redirect()->back()->with('error', trans('admin/locations/message.does_not_exist'));
|
return view('departments/edit')->with('item', $department);
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $item);
|
|
||||||
|
|
||||||
return view('departments/edit', compact('item'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -157,11 +146,8 @@ class DepartmentsController extends Controller
|
||||||
* @param int $departmentId
|
* @param int $departmentId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $id) : RedirectResponse
|
public function update(ImageUploadRequest $request, Department $department) : RedirectResponse
|
||||||
{
|
{
|
||||||
if (is_null($department = Department::find($id))) {
|
|
||||||
return redirect()->route('departments.index')->with('error', trans('admin/departments/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $department);
|
$this->authorize('update', $department);
|
||||||
|
|
||||||
|
@ -171,7 +157,7 @@ class DepartmentsController extends Controller
|
||||||
$department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null);
|
$department->company_id = ($request->filled('company_id') ? $request->input('company_id') : null);
|
||||||
$department->phone = $request->input('phone');
|
$department->phone = $request->input('phone');
|
||||||
$department->fax = $request->input('fax');
|
$department->fax = $request->input('fax');
|
||||||
|
$department->notes = $request->input('notes');
|
||||||
$department = $request->handleImages($department);
|
$department = $request->handleImages($department);
|
||||||
|
|
||||||
if ($department->save()) {
|
if ($department->save()) {
|
||||||
|
|
|
@ -95,17 +95,11 @@ class DepreciationsController extends Controller
|
||||||
* @param int $depreciationId
|
* @param int $depreciationId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($depreciationId = null) : RedirectResponse | View
|
public function edit(Depreciation $depreciation) : RedirectResponse | View
|
||||||
{
|
{
|
||||||
// Check if the depreciation exists
|
|
||||||
if (is_null($item = Depreciation::find($depreciationId))) {
|
|
||||||
// Redirect to the blogs management page
|
|
||||||
return redirect()->route('depreciations.index')->with('error', trans('admin/depreciations/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $item);
|
$this->authorize('update', $depreciation);
|
||||||
|
return view('depreciations/edit')->with('item', $depreciation);
|
||||||
return view('depreciations/edit', compact('item'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -117,17 +111,10 @@ class DepreciationsController extends Controller
|
||||||
* @param int $depreciationId
|
* @param int $depreciationId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, $depreciationId = null) : RedirectResponse
|
public function update(Request $request, Depreciation $depreciation) : RedirectResponse
|
||||||
{
|
{
|
||||||
// Check if the depreciation exists
|
|
||||||
if (is_null($depreciation = Depreciation::find($depreciationId))) {
|
|
||||||
// Redirect to the blogs management page
|
|
||||||
return redirect()->route('depreciations.index')->with('error', trans('admin/depreciations/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $depreciation);
|
$this->authorize('update', $depreciation);
|
||||||
|
|
||||||
// Depreciation data
|
|
||||||
$depreciation->name = $request->input('name');
|
$depreciation->name = $request->input('name');
|
||||||
$depreciation->months = $request->input('months');
|
$depreciation->months = $request->input('months');
|
||||||
|
|
||||||
|
@ -191,12 +178,12 @@ class DepreciationsController extends Controller
|
||||||
* @param int $depreciationId
|
* @param int $depreciationId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(Depreciation $depreciation) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$depreciation = Depreciation::withCount('assets as assets_count')
|
$depreciation = Depreciation::withCount('assets as assets_count')
|
||||||
->withCount('models as models_count')
|
->withCount('models as models_count')
|
||||||
->withCount('licenses as licenses_count')
|
->withCount('licenses as licenses_count')
|
||||||
->find($id);
|
->find($depreciation->id);
|
||||||
|
|
||||||
$this->authorize('view', $depreciation);
|
$this->authorize('view', $depreciation);
|
||||||
|
|
||||||
|
|
|
@ -62,6 +62,7 @@ class GroupsController extends Controller
|
||||||
$group->name = $request->input('name');
|
$group->name = $request->input('name');
|
||||||
$group->permissions = json_encode($request->input('permission'));
|
$group->permissions = json_encode($request->input('permission'));
|
||||||
$group->created_by = auth()->id();
|
$group->created_by = auth()->id();
|
||||||
|
$group->notes = $request->input('notes');
|
||||||
|
|
||||||
if ($group->save()) {
|
if ($group->save()) {
|
||||||
return redirect()->route('groups.index')->with('success', trans('admin/groups/message.success.create'));
|
return redirect()->route('groups.index')->with('success', trans('admin/groups/message.success.create'));
|
||||||
|
@ -78,21 +79,14 @@ class GroupsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($id) : View | RedirectResponse
|
public function edit(Group $group) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$group = Group::find($id);
|
|
||||||
|
|
||||||
if ($group) {
|
|
||||||
$permissions = config('permissions');
|
$permissions = config('permissions');
|
||||||
$groupPermissions = $group->decodePermissions();
|
$groupPermissions = $group->decodePermissions();
|
||||||
$selected_array = Helper::selectedPermissionsArray($permissions, $groupPermissions);
|
$selected_array = Helper::selectedPermissionsArray($permissions, $groupPermissions);
|
||||||
|
|
||||||
return view('groups.edit', compact('group', 'permissions', 'selected_array', 'groupPermissions'));
|
return view('groups.edit', compact('group', 'permissions', 'selected_array', 'groupPermissions'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('groups.index')->with('error', trans('admin/groups/message.group_not_found', ['id' => $id]));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates and stores the updated User Group data.
|
* Validates and stores the updated User Group data.
|
||||||
*
|
*
|
||||||
|
@ -101,13 +95,11 @@ class GroupsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, $id = null) : RedirectResponse
|
public function update(Request $request, Group $group) : RedirectResponse
|
||||||
{
|
{
|
||||||
if (! $group = Group::find($id)) {
|
|
||||||
return redirect()->route('groups.index')->with('error', trans('admin/groups/message.group_not_found', ['id' => $id]));
|
|
||||||
}
|
|
||||||
$group->name = $request->input('name');
|
$group->name = $request->input('name');
|
||||||
$group->permissions = json_encode($request->input('permission'));
|
$group->permissions = json_encode($request->input('permission'));
|
||||||
|
$group->notes = $request->input('notes');
|
||||||
|
|
||||||
if (! config('app.lock_passwords')) {
|
if (! config('app.lock_passwords')) {
|
||||||
if ($group->save()) {
|
if ($group->save()) {
|
||||||
|
@ -149,14 +141,8 @@ class GroupsController extends Controller
|
||||||
* @param $id
|
* @param $id
|
||||||
* @since [v4.0.11]
|
* @since [v4.0.11]
|
||||||
*/
|
*/
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(Group $group) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$group = Group::find($id);
|
|
||||||
|
|
||||||
if ($group) {
|
|
||||||
return view('groups/view', compact('group'));
|
return view('groups/view', compact('group'));
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('groups.index')->with('error', trans('admin/groups/message.group_not_found', ['id' => $id]));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,10 +4,8 @@ namespace App\Http\Controllers\Kits;
|
||||||
|
|
||||||
use App\Http\Controllers\CheckInOutRequest;
|
use App\Http\Controllers\CheckInOutRequest;
|
||||||
use App\Http\Controllers\Controller;
|
use App\Http\Controllers\Controller;
|
||||||
use App\Models\PredefinedKit;
|
|
||||||
use App\Models\Asset;
|
use App\Models\Asset;
|
||||||
use App\Models\PredefinedLicence;
|
use App\Models\PredefinedKit;
|
||||||
use App\Models\PredefinedModel;
|
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Services\PredefinedKitCheckoutService;
|
use App\Services\PredefinedKitCheckoutService;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
|
@ -35,12 +33,9 @@ class CheckoutKitController extends Controller
|
||||||
* @author [D. Minaev.] [<dmitriy.minaev.v@gmail.com>]
|
* @author [D. Minaev.] [<dmitriy.minaev.v@gmail.com>]
|
||||||
* @return \Illuminate\Contracts\View\View View to checkout
|
* @return \Illuminate\Contracts\View\View View to checkout
|
||||||
*/
|
*/
|
||||||
public function showCheckout($kit_id)
|
public function showCheckout(PredefinedKit $kit)
|
||||||
{
|
{
|
||||||
$this->authorize('checkout', Asset::class);
|
$this->authorize('checkout', Asset::class);
|
||||||
|
|
||||||
$kit = PredefinedKit::findOrFail($kit_id);
|
|
||||||
|
|
||||||
return view('kits/checkout')->with('kit', $kit);
|
return view('kits/checkout')->with('kit', $kit);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -76,17 +76,15 @@ class PredefinedKitsController extends Controller
|
||||||
* @param int $kit_id
|
* @param int $kit_id
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
*/
|
*/
|
||||||
public function edit($kit_id = null)
|
public function edit(PredefinedKit $kit)
|
||||||
{
|
{
|
||||||
$this->authorize('update', PredefinedKit::class);
|
$this->authorize('update', PredefinedKit::class);
|
||||||
if ($kit = PredefinedKit::find($kit_id)) {
|
|
||||||
return view('kits/edit')
|
return view('kits/edit')
|
||||||
->with('item', $kit)
|
->with('item', $kit)
|
||||||
->with('models', $kit->models)
|
->with('models', $kit->models)
|
||||||
->with('licenses', $kit->licenses);
|
->with('licenses', $kit->licenses);
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route('kits.index')->with('error', trans('admin/kits/general.kit_none'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -98,15 +96,9 @@ class PredefinedKitsController extends Controller
|
||||||
* @param int $kit_id
|
* @param int $kit_id
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $kit_id = null)
|
public function update(ImageUploadRequest $request, PredefinedKit $kit)
|
||||||
{
|
{
|
||||||
$this->authorize('update', PredefinedKit::class);
|
$this->authorize('update', PredefinedKit::class);
|
||||||
// Check if the kit exists
|
|
||||||
if (is_null($kit = PredefinedKit::find($kit_id))) {
|
|
||||||
// Redirect to the kits management page
|
|
||||||
return redirect()->route('kits.index')->with('error', trans('admin/kits/general.kit_none'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$kit->name = $request->input('name');
|
$kit->name = $request->input('name');
|
||||||
|
|
||||||
if ($kit->save()) {
|
if ($kit->save()) {
|
||||||
|
@ -153,9 +145,9 @@ class PredefinedKitsController extends Controller
|
||||||
* @param int $modelId
|
* @param int $modelId
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
*/
|
*/
|
||||||
public function show($kit_id = null)
|
public function show(PredefinedKit $kit)
|
||||||
{
|
{
|
||||||
return $this->edit($kit_id);
|
return $this->edit($kit);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -28,16 +28,11 @@ class LicenseCheckinController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function create($seatId = null, $backTo = null)
|
public function create(LicenseSeat $licenseSeat, $backTo = null)
|
||||||
{
|
{
|
||||||
// Check if the asset exists
|
// Check if the asset exists
|
||||||
if (is_null($licenseSeat = LicenseSeat::find($seatId)) || is_null($license = License::find($licenseSeat->license_id))) {
|
$license = License::find($licenseSeat->license_id);
|
||||||
// Redirect to the asset management page with error
|
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('checkout', $license);
|
$this->authorize('checkout', $license);
|
||||||
|
|
||||||
return view('licenses/checkin', compact('licenseSeat'))->with('backto', $backTo);
|
return view('licenses/checkin', compact('licenseSeat'))->with('backto', $backTo);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -28,17 +28,14 @@ class LicenseCheckoutController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function create($id)
|
public function create(License $license)
|
||||||
{
|
{
|
||||||
|
|
||||||
if ($license = License::find($id)) {
|
|
||||||
|
|
||||||
$this->authorize('checkout', $license);
|
$this->authorize('checkout', $license);
|
||||||
|
|
||||||
if ($license->category) {
|
if ($license->category) {
|
||||||
|
|
||||||
// Make sure there is at least one available to checkout
|
// Make sure there is at least one available to checkout
|
||||||
if ($license->availCount()->count() < 1){
|
if ($license->availCount()->count() < 1) {
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.checkout.not_enough_seats'));
|
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.checkout.not_enough_seats'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -52,12 +49,6 @@ class LicenseCheckoutController extends Controller
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Not found
|
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found'));
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Validates and stores the license checkout action.
|
* Validates and stores the license checkout action.
|
||||||
*
|
*
|
||||||
|
|
|
@ -121,13 +121,10 @@ class LicensesController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function edit($licenseId = null)
|
public function edit(License $license)
|
||||||
{
|
{
|
||||||
if (is_null($item = License::find($licenseId))) {
|
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $item);
|
$this->authorize('update', $license);
|
||||||
|
|
||||||
$maintained_list = [
|
$maintained_list = [
|
||||||
'' => 'Maintained',
|
'' => 'Maintained',
|
||||||
|
@ -135,7 +132,8 @@ class LicensesController extends Controller
|
||||||
'0' => 'No',
|
'0' => 'No',
|
||||||
];
|
];
|
||||||
|
|
||||||
return view('licenses/edit', compact('item'))
|
return view('licenses/edit')
|
||||||
|
->with('item', $license)
|
||||||
->with('depreciation_list', Helper::depreciationList())
|
->with('depreciation_list', Helper::depreciationList())
|
||||||
->with('maintained_list', $maintained_list);
|
->with('maintained_list', $maintained_list);
|
||||||
}
|
}
|
||||||
|
@ -153,11 +151,9 @@ class LicensesController extends Controller
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, $licenseId = null)
|
public function update(Request $request, License $license)
|
||||||
{
|
{
|
||||||
if (is_null($license = License::find($licenseId))) {
|
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->authorize('update', $license);
|
$this->authorize('update', $license);
|
||||||
|
|
||||||
|
@ -201,10 +197,10 @@ class LicensesController extends Controller
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function destroy($licenseId)
|
public function destroy(License $license)
|
||||||
{
|
{
|
||||||
// Check if the license exists
|
// Check if the license exists
|
||||||
if (is_null($license = License::find($licenseId))) {
|
if (is_null($license = License::find($license->id))) {
|
||||||
// Redirect to the license management page
|
// Redirect to the license management page
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found'));
|
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.not_found'));
|
||||||
}
|
}
|
||||||
|
@ -238,14 +234,9 @@ class LicensesController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function show($licenseId = null)
|
public function show(License $license)
|
||||||
{
|
{
|
||||||
$license = License::with('assignedusers')->find($licenseId);
|
$license = License::with('assignedusers')->find($license->id);
|
||||||
|
|
||||||
if (!$license) {
|
|
||||||
return redirect()->route('licenses.index')
|
|
||||||
->with('error', trans('admin/licenses/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$users_count = User::where('autoassign_licenses', '1')->count();
|
$users_count = User::where('autoassign_licenses', '1')->count();
|
||||||
$total_seats_count = $license->totalSeatsByLicenseID();
|
$total_seats_count = $license->totalSeatsByLicenseID();
|
||||||
|
@ -267,10 +258,10 @@ class LicensesController extends Controller
|
||||||
*
|
*
|
||||||
* @author [A. Gianotto] [<snipe@snipe.net>]
|
* @author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
* @param int $licenseId
|
* @param int $licenseId
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse | \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function getClone($licenseId = null)
|
public function getClone($licenseId = null) : \Illuminate\Contracts\View\View | \Illuminate\Http\RedirectResponse
|
||||||
{
|
{
|
||||||
if (is_null($license_to_clone = License::find($licenseId))) {
|
if (is_null($license_to_clone = License::find($licenseId))) {
|
||||||
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));
|
return redirect()->route('licenses.index')->with('error', trans('admin/licenses/message.does_not_exist'));
|
||||||
|
|
|
@ -78,6 +78,7 @@ class LocationsController extends Controller
|
||||||
$location->created_by = auth()->id();
|
$location->created_by = auth()->id();
|
||||||
$location->phone = request('phone');
|
$location->phone = request('phone');
|
||||||
$location->fax = request('fax');
|
$location->fax = request('fax');
|
||||||
|
$location->notes = $request->input('notes');
|
||||||
|
|
||||||
$location = $request->handleImages($location);
|
$location = $request->handleImages($location);
|
||||||
|
|
||||||
|
@ -96,15 +97,10 @@ class LocationsController extends Controller
|
||||||
* @param int $locationId
|
* @param int $locationId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($locationId = null) : View | RedirectResponse
|
public function edit(Location $location) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Location::class);
|
$this->authorize('update', Location::class);
|
||||||
// Check if the location exists
|
return view('locations/edit')->with('item', $location);
|
||||||
if (is_null($item = Location::find($locationId))) {
|
|
||||||
return redirect()->route('locations.index')->with('error', trans('admin/locations/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
return view('locations/edit', compact('item'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -116,15 +112,10 @@ class LocationsController extends Controller
|
||||||
* @param int $locationId
|
* @param int $locationId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $locationId = null) : RedirectResponse
|
public function update(ImageUploadRequest $request, Location $location) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Location::class);
|
$this->authorize('update', Location::class);
|
||||||
// Check if the location exists
|
|
||||||
if (is_null($location = Location::find($locationId))) {
|
|
||||||
return redirect()->route('locations.index')->with('error', trans('admin/locations/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the location data
|
|
||||||
$location->name = $request->input('name');
|
$location->name = $request->input('name');
|
||||||
$location->parent_id = $request->input('parent_id', null);
|
$location->parent_id = $request->input('parent_id', null);
|
||||||
$location->currency = $request->input('currency', '$');
|
$location->currency = $request->input('currency', '$');
|
||||||
|
@ -138,6 +129,7 @@ class LocationsController extends Controller
|
||||||
$location->fax = request('fax');
|
$location->fax = request('fax');
|
||||||
$location->ldap_ou = $request->input('ldap_ou');
|
$location->ldap_ou = $request->input('ldap_ou');
|
||||||
$location->manager_id = $request->input('manager_id');
|
$location->manager_id = $request->input('manager_id');
|
||||||
|
$location->notes = $request->input('notes');
|
||||||
|
|
||||||
$location = $request->handleImages($location);
|
$location = $request->handleImages($location);
|
||||||
|
|
||||||
|
@ -192,7 +184,7 @@ class LocationsController extends Controller
|
||||||
* @param int $id
|
* @param int $id
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function show($id = null) : View | RedirectResponse
|
public function show(Location $location) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$location = Location::withCount('assignedAssets as assigned_assets_count')
|
$location = Location::withCount('assignedAssets as assigned_assets_count')
|
||||||
->withCount('assets as assets_count')
|
->withCount('assets as assets_count')
|
||||||
|
@ -200,7 +192,7 @@ class LocationsController extends Controller
|
||||||
->withCount('children as children_count')
|
->withCount('children as children_count')
|
||||||
->withCount('users as users_count')
|
->withCount('users as users_count')
|
||||||
->withTrashed()
|
->withTrashed()
|
||||||
->find($id);
|
->find($location->id);
|
||||||
|
|
||||||
if (isset($location->id)) {
|
if (isset($location->id)) {
|
||||||
return view('locations/view', compact('location'));
|
return view('locations/view', compact('location'));
|
||||||
|
|
|
@ -67,6 +67,7 @@ class ManufacturersController extends Controller
|
||||||
$manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url');
|
$manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url');
|
||||||
$manufacturer->support_phone = $request->input('support_phone');
|
$manufacturer->support_phone = $request->input('support_phone');
|
||||||
$manufacturer->support_email = $request->input('support_email');
|
$manufacturer->support_email = $request->input('support_email');
|
||||||
|
$manufacturer->notes = $request->input('notes');
|
||||||
$manufacturer = $request->handleImages($manufacturer);
|
$manufacturer = $request->handleImages($manufacturer);
|
||||||
|
|
||||||
if ($manufacturer->save()) {
|
if ($manufacturer->save()) {
|
||||||
|
@ -84,18 +85,10 @@ class ManufacturersController extends Controller
|
||||||
* @param int $manufacturerId
|
* @param int $manufacturerId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function edit($manufacturerId = null) : View | RedirectResponse
|
public function edit(Manufacturer $manufacturer) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
// Handles manufacturer checks and permissions.
|
|
||||||
$this->authorize('update', Manufacturer::class);
|
$this->authorize('update', Manufacturer::class);
|
||||||
|
return view('manufacturers/edit')->with('item', $manufacturer);
|
||||||
// Check if the manufacturer exists
|
|
||||||
if (! $item = Manufacturer::find($manufacturerId)) {
|
|
||||||
return redirect()->route('manufacturers.index')->with('error', trans('admin/manufacturers/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show the page
|
|
||||||
return view('manufacturers/edit', compact('item'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -107,22 +100,17 @@ class ManufacturersController extends Controller
|
||||||
* @param int $manufacturerId
|
* @param int $manufacturerId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function update(ImageUploadRequest $request, $manufacturerId = null) : RedirectResponse
|
public function update(ImageUploadRequest $request, Manufacturer $manufacturer) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Manufacturer::class);
|
$this->authorize('update', Manufacturer::class);
|
||||||
// Check if the manufacturer exists
|
|
||||||
if (is_null($manufacturer = Manufacturer::find($manufacturerId))) {
|
|
||||||
// Redirect to the manufacturer page
|
|
||||||
return redirect()->route('manufacturers.index')->with('error', trans('admin/manufacturers/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save the data
|
|
||||||
$manufacturer->name = $request->input('name');
|
$manufacturer->name = $request->input('name');
|
||||||
$manufacturer->url = $request->input('url');
|
$manufacturer->url = $request->input('url');
|
||||||
$manufacturer->support_url = $request->input('support_url');
|
$manufacturer->support_url = $request->input('support_url');
|
||||||
$manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url');
|
$manufacturer->warranty_lookup_url = $request->input('warranty_lookup_url');
|
||||||
$manufacturer->support_phone = $request->input('support_phone');
|
$manufacturer->support_phone = $request->input('support_phone');
|
||||||
$manufacturer->support_email = $request->input('support_email');
|
$manufacturer->support_email = $request->input('support_email');
|
||||||
|
$manufacturer->notes = $request->input('notes');
|
||||||
|
|
||||||
// Set the model's image property to null if the image is being deleted
|
// Set the model's image property to null if the image is being deleted
|
||||||
if ($request->input('image_delete') == 1) {
|
if ($request->input('image_delete') == 1) {
|
||||||
|
@ -183,20 +171,12 @@ class ManufacturersController extends Controller
|
||||||
* @param int $manufacturerId
|
* @param int $manufacturerId
|
||||||
* @since [v1.0]
|
* @since [v1.0]
|
||||||
*/
|
*/
|
||||||
public function show($manufacturerId = null) : View | RedirectResponse
|
public function show(Manufacturer $manufacturer) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', Manufacturer::class);
|
$this->authorize('view', Manufacturer::class);
|
||||||
$manufacturer = Manufacturer::find($manufacturerId);
|
|
||||||
|
|
||||||
if (isset($manufacturer->id)) {
|
|
||||||
return view('manufacturers/view', compact('manufacturer'));
|
return view('manufacturers/view', compact('manufacturer'));
|
||||||
}
|
}
|
||||||
|
|
||||||
$error = trans('admin/manufacturers/message.does_not_exist');
|
|
||||||
// Redirect to the user management page
|
|
||||||
return redirect()->route('manufacturers.index')->with('error', $error);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Restore a given Manufacturer (mark as un-deleted)
|
* Restore a given Manufacturer (mark as un-deleted)
|
||||||
*
|
*
|
||||||
|
|
42
app/Http/Controllers/NotesController.php
Normal file
42
app/Http/Controllers/NotesController.php
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Http\Controllers;
|
||||||
|
|
||||||
|
use App\Models\Actionlog;
|
||||||
|
use App\Models\Asset;
|
||||||
|
use Illuminate\Http\Request;
|
||||||
|
use Illuminate\Support\Facades\Auth;
|
||||||
|
use Illuminate\Validation\Rule;
|
||||||
|
|
||||||
|
class NotesController extends Controller
|
||||||
|
{
|
||||||
|
public function store(Request $request)
|
||||||
|
{
|
||||||
|
$this->authorize('update', Asset::class);
|
||||||
|
|
||||||
|
$validated = $request->validate([
|
||||||
|
'id' => 'required',
|
||||||
|
'note' => 'required|string|max:50000',
|
||||||
|
'type' => [
|
||||||
|
'required',
|
||||||
|
Rule::in(['asset']),
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
$item = Asset::findOrFail($validated['id']);
|
||||||
|
|
||||||
|
$this->authorize('update', $item);
|
||||||
|
|
||||||
|
$logaction = new Actionlog;
|
||||||
|
$logaction->item_id = $item->id;
|
||||||
|
$logaction->item_type = get_class($item);
|
||||||
|
$logaction->note = $validated['note'];
|
||||||
|
$logaction->created_by = Auth::id();
|
||||||
|
$logaction->logaction('note added');
|
||||||
|
|
||||||
|
return redirect()
|
||||||
|
->route('hardware.show', $validated['id'])
|
||||||
|
->withFragment('history')
|
||||||
|
->with('success', trans('general.note_added'));
|
||||||
|
}
|
||||||
|
}
|
|
@ -266,7 +266,7 @@ class ReportsController extends Controller
|
||||||
|
|
||||||
$actionlogs = Actionlog::with('item', 'user', 'target', 'location', 'adminuser')
|
$actionlogs = Actionlog::with('item', 'user', 'target', 'location', 'adminuser')
|
||||||
->orderBy('created_at', 'DESC')
|
->orderBy('created_at', 'DESC')
|
||||||
->chunk(20, function ($actionlogs) use ($handle) {
|
->chunk(500, function ($actionlogs) use ($handle) {
|
||||||
$executionTime = microtime(true) - $_SERVER['REQUEST_TIME_FLOAT'];
|
$executionTime = microtime(true) - $_SERVER['REQUEST_TIME_FLOAT'];
|
||||||
Log::debug('Walking results: '.$executionTime);
|
Log::debug('Walking results: '.$executionTime);
|
||||||
$count = 0;
|
$count = 0;
|
||||||
|
@ -748,7 +748,7 @@ class ReportsController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
Log::debug($assets->toSql());
|
Log::debug($assets->toSql());
|
||||||
$assets->orderBy('assets.id', 'ASC')->chunk(20, function ($assets) use ($handle, $customfields, $request) {
|
$assets->orderBy('assets.id', 'ASC')->chunk(500, function ($assets) use ($handle, $customfields, $request) {
|
||||||
|
|
||||||
$executionTime = microtime(true) - $_SERVER['REQUEST_TIME_FLOAT'];
|
$executionTime = microtime(true) - $_SERVER['REQUEST_TIME_FLOAT'];
|
||||||
Log::debug('Walking results: '.$executionTime);
|
Log::debug('Walking results: '.$executionTime);
|
||||||
|
@ -1175,18 +1175,13 @@ class ReportsController extends Controller
|
||||||
}
|
}
|
||||||
$email = $assetItem->assignedTo?->email;
|
$email = $assetItem->assignedTo?->email;
|
||||||
$locale = $assetItem->assignedTo?->locale;
|
$locale = $assetItem->assignedTo?->locale;
|
||||||
// Only send notification if assigned
|
|
||||||
if ($locale && $email) {
|
|
||||||
Mail::to($email)->send((new CheckoutAssetMail($assetItem, $assetItem->assignedTo, $logItem->user, $acceptance, $logItem->note))->locale($locale));
|
|
||||||
|
|
||||||
} elseif ($email) {
|
if (is_null($email) || $email === '') {
|
||||||
Mail::to($email)->send((new CheckoutAssetMail($assetItem, $assetItem->assignedTo, $logItem->user, $acceptance, $logItem->note)));
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($email == ''){
|
|
||||||
return redirect()->route('reports/unaccepted_assets')->with('error', trans('general.no_email'));
|
return redirect()->route('reports/unaccepted_assets')->with('error', trans('general.no_email'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Mail::to($email)->send((new CheckoutAssetMail($assetItem, $assetItem->assignedTo, $logItem->user, $acceptance, $logItem->note, firstTimeSending: false))->locale($locale));
|
||||||
|
|
||||||
return redirect()->route('reports/unaccepted_assets')->with('success', trans('admin/reports/general.reminder_sent'));
|
return redirect()->route('reports/unaccepted_assets')->with('success', trans('admin/reports/general.reminder_sent'));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
5
app/Http/Controllers/SettingsController.php
Executable file → Normal file
5
app/Http/Controllers/SettingsController.php
Executable file → Normal file
|
@ -256,7 +256,7 @@ class SettingsController extends Controller
|
||||||
Artisan::call('migrate', ['--force' => true]);
|
Artisan::call('migrate', ['--force' => true]);
|
||||||
if ((! file_exists(storage_path().'/oauth-private.key')) || (! file_exists(storage_path().'/oauth-public.key'))) {
|
if ((! file_exists(storage_path().'/oauth-private.key')) || (! file_exists(storage_path().'/oauth-public.key'))) {
|
||||||
Artisan::call('migrate', ['--path' => 'vendor/laravel/passport/database/migrations', '--force' => true]);
|
Artisan::call('migrate', ['--path' => 'vendor/laravel/passport/database/migrations', '--force' => true]);
|
||||||
Artisan::call('passport:install');
|
Artisan::call('passport:install', ['--no-interaction' => true]);
|
||||||
}
|
}
|
||||||
|
|
||||||
return view('setup/migrate')
|
return view('setup/migrate')
|
||||||
|
@ -800,6 +800,7 @@ class SettingsController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($setting->save()) {
|
if ($setting->save()) {
|
||||||
|
|
||||||
return redirect()->route('settings.labels.index')
|
return redirect()->route('settings.labels.index')
|
||||||
->with('success', trans('admin/settings/message.update.success'));
|
->with('success', trans('admin/settings/message.update.success'));
|
||||||
}
|
}
|
||||||
|
@ -850,6 +851,7 @@ class SettingsController extends Controller
|
||||||
$setting->ldap_auth_filter_query = $request->input('ldap_auth_filter_query');
|
$setting->ldap_auth_filter_query = $request->input('ldap_auth_filter_query');
|
||||||
$setting->ldap_version = $request->input('ldap_version', 3);
|
$setting->ldap_version = $request->input('ldap_version', 3);
|
||||||
$setting->ldap_active_flag = $request->input('ldap_active_flag');
|
$setting->ldap_active_flag = $request->input('ldap_active_flag');
|
||||||
|
$setting->ldap_invert_active_flag = $request->input('ldap_invert_active_flag');
|
||||||
$setting->ldap_emp_num = $request->input('ldap_emp_num');
|
$setting->ldap_emp_num = $request->input('ldap_emp_num');
|
||||||
$setting->ldap_email = $request->input('ldap_email');
|
$setting->ldap_email = $request->input('ldap_email');
|
||||||
$setting->ldap_manager = $request->input('ldap_manager');
|
$setting->ldap_manager = $request->input('ldap_manager');
|
||||||
|
@ -869,7 +871,6 @@ class SettingsController extends Controller
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($setting->save()) {
|
if ($setting->save()) {
|
||||||
$setting->update_client_side_cert_files();
|
|
||||||
return redirect()->route('settings.ldap.index')
|
return redirect()->route('settings.ldap.index')
|
||||||
->with('success', trans('admin/settings/message.update.success'));
|
->with('success', trans('admin/settings/message.update.success'));
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,16 +26,12 @@ class StatuslabelsController extends Controller
|
||||||
return view('statuslabels.index');
|
return view('statuslabels.index');
|
||||||
}
|
}
|
||||||
|
|
||||||
public function show($id) : View | RedirectResponse
|
public function show(Statuslabel $statuslabel) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', Statuslabel::class);
|
$this->authorize('view', Statuslabel::class);
|
||||||
if ($statuslabel = Statuslabel::find($id)) {
|
|
||||||
return view('statuslabels.view')->with('statuslabel', $statuslabel);
|
return view('statuslabels.view')->with('statuslabel', $statuslabel);
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('statuslabels.index')->with('error', trans('admin/statuslabels/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Statuslabel create.
|
* Statuslabel create.
|
||||||
*
|
*
|
||||||
|
@ -91,20 +87,15 @@ class StatuslabelsController extends Controller
|
||||||
*
|
*
|
||||||
* @param int $statuslabelId
|
* @param int $statuslabelId
|
||||||
*/
|
*/
|
||||||
public function edit($statuslabelId = null) : View | RedirectResponse
|
public function edit(Statuslabel $statuslabel) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Statuslabel::class);
|
$this->authorize('update', Statuslabel::class);
|
||||||
// Check if the Statuslabel exists
|
|
||||||
if (is_null($item = Statuslabel::find($statuslabelId))) {
|
|
||||||
// Redirect to the blogs management page
|
|
||||||
return redirect()->route('statuslabels.index')->with('error', trans('admin/statuslabels/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
$use_statuslabel_type = $item->getStatuslabelType();
|
|
||||||
|
|
||||||
$statuslabel_types = ['' => trans('admin/hardware/form.select_statustype')] + ['undeployable' => trans('admin/hardware/general.undeployable')] + ['pending' => trans('admin/hardware/general.pending')] + ['archived' => trans('admin/hardware/general.archived')] + ['deployable' => trans('admin/hardware/general.deployable')];
|
$statuslabel_types = ['' => trans('admin/hardware/form.select_statustype')] + ['undeployable' => trans('admin/hardware/general.undeployable')] + ['pending' => trans('admin/hardware/general.pending')] + ['archived' => trans('admin/hardware/general.archived')] + ['deployable' => trans('admin/hardware/general.deployable')];
|
||||||
|
|
||||||
return view('statuslabels/edit', compact('item', 'statuslabel_types'))->with('use_statuslabel_type', $use_statuslabel_type);
|
return view('statuslabels/edit', compact('statuslabel_types'))
|
||||||
|
->with('item', $statuslabel)
|
||||||
|
->with('use_statuslabel_type', $statuslabel);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -112,14 +103,9 @@ class StatuslabelsController extends Controller
|
||||||
*
|
*
|
||||||
* @param int $statuslabelId
|
* @param int $statuslabelId
|
||||||
*/
|
*/
|
||||||
public function update(Request $request, $statuslabelId = null) : RedirectResponse
|
public function update(Request $request, Statuslabel $statuslabel) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Statuslabel::class);
|
$this->authorize('update', Statuslabel::class);
|
||||||
// Check if the Statuslabel exists
|
|
||||||
if (is_null($statuslabel = Statuslabel::find($statuslabelId))) {
|
|
||||||
// Redirect to the blogs management page
|
|
||||||
return redirect()->route('statuslabels.index')->with('error', trans('admin/statuslabels/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
if (! $request->filled('statuslabel_types')) {
|
if (! $request->filled('statuslabel_types')) {
|
||||||
return redirect()->back()->withInput()->withErrors(['statuslabel_types' => trans('validation.statuslabel_type')]);
|
return redirect()->back()->withInput()->withErrors(['statuslabel_types' => trans('validation.statuslabel_type')]);
|
||||||
|
|
|
@ -77,17 +77,10 @@ class SuppliersController extends Controller
|
||||||
*
|
*
|
||||||
* @param int $supplierId
|
* @param int $supplierId
|
||||||
*/
|
*/
|
||||||
public function edit($supplierId = null) : View | RedirectResponse
|
public function edit(Supplier $supplier) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Supplier::class);
|
$this->authorize('update', Supplier::class);
|
||||||
// Check if the supplier exists
|
return view('suppliers/edit')->with('item', $supplier);
|
||||||
if (is_null($item = Supplier::find($supplierId))) {
|
|
||||||
// Redirect to the supplier page
|
|
||||||
return redirect()->route('suppliers.index')->with('error', trans('admin/suppliers/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Show the page
|
|
||||||
return view('suppliers/edit', compact('item'));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -95,14 +88,9 @@ class SuppliersController extends Controller
|
||||||
*
|
*
|
||||||
* @param int $supplierId
|
* @param int $supplierId
|
||||||
*/
|
*/
|
||||||
public function update($supplierId, ImageUploadRequest $request) : RedirectResponse
|
public function update(ImageUploadRequest $request, Supplier $supplier) : RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('update', Supplier::class);
|
$this->authorize('update', Supplier::class);
|
||||||
|
|
||||||
if (is_null($supplier = Supplier::find($supplierId))) {
|
|
||||||
return redirect()->route('suppliers.index')->with('error', trans('admin/suppliers/message.does_not_exist'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Save the data
|
// Save the data
|
||||||
$supplier->name = request('name');
|
$supplier->name = request('name');
|
||||||
$supplier->address = request('address');
|
$supplier->address = request('address');
|
||||||
|
@ -163,15 +151,10 @@ class SuppliersController extends Controller
|
||||||
* @param null $supplierId
|
* @param null $supplierId
|
||||||
* @internal param int $assetId
|
* @internal param int $assetId
|
||||||
*/
|
*/
|
||||||
public function show($supplierId = null) : View | RedirectResponse
|
public function show(Supplier $supplier) : View | RedirectResponse
|
||||||
{
|
{
|
||||||
$this->authorize('view', Supplier::class);
|
$this->authorize('view', Supplier::class);
|
||||||
$supplier = Supplier::find($supplierId);
|
|
||||||
|
|
||||||
if (isset($supplier->id)) {
|
|
||||||
return view('suppliers/view', compact('supplier'));
|
return view('suppliers/view', compact('supplier'));
|
||||||
}
|
|
||||||
|
|
||||||
return redirect()->route('suppliers.index')->with('error', trans('admin/suppliers/message.does_not_exist'));
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,6 +15,7 @@ use App\Models\ConsumableAssignment;
|
||||||
use App\Models\Consumable;
|
use App\Models\Consumable;
|
||||||
use App\Models\Setting;
|
use App\Models\Setting;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
|
use App\Notifications\CurrentInventory;
|
||||||
use Carbon\Carbon;
|
use Carbon\Carbon;
|
||||||
use Illuminate\Http\Request;
|
use Illuminate\Http\Request;
|
||||||
use Illuminate\Support\Collection;
|
use Illuminate\Support\Collection;
|
||||||
|
@ -52,6 +53,28 @@ class BulkUsersController extends Controller
|
||||||
return view('users/bulk-edit', compact('users'))
|
return view('users/bulk-edit', compact('users'))
|
||||||
->with('groups', Group::pluck('name', 'id'));
|
->with('groups', Group::pluck('name', 'id'));
|
||||||
|
|
||||||
|
// bulk send assigned inventory
|
||||||
|
} elseif ($request->input('bulk_actions') == 'send_assigned') {
|
||||||
|
$this->authorize('update', User::class);
|
||||||
|
|
||||||
|
$users_without_email = 0;
|
||||||
|
foreach ($users as $user) {
|
||||||
|
if (empty($user->email)) {
|
||||||
|
$users_without_email++;
|
||||||
|
} else {
|
||||||
|
$user->notify((new CurrentInventory($user)));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($users_without_email == 0) {
|
||||||
|
return redirect()->back()->with('success', trans_choice('admin/users/general.users_notified', $users->count()));
|
||||||
|
} else {
|
||||||
|
return redirect()->back()->with('warning', trans_choice('admin/users/general.users_notified_warning', $users->count(), ['no_email' => $users_without_email]));
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
// bulk delete, display the bulk delete confirmation form
|
// bulk delete, display the bulk delete confirmation form
|
||||||
} elseif ($request->input('bulk_actions') == 'delete') {
|
} elseif ($request->input('bulk_actions') == 'delete') {
|
||||||
$this->authorize('delete', User::class);
|
$this->authorize('delete', User::class);
|
||||||
|
|
|
@ -22,15 +22,9 @@ class UserFilesController extends Controller
|
||||||
*@author [A. Gianotto] [<snipe@snipe.net>]
|
*@author [A. Gianotto] [<snipe@snipe.net>]
|
||||||
* @since [v1.6]
|
* @since [v1.6]
|
||||||
*/
|
*/
|
||||||
public function store(UploadFileRequest $request, $userId = null)
|
public function store(UploadFileRequest $request, User $user)
|
||||||
{
|
{
|
||||||
$user = User::find($userId);
|
|
||||||
$destinationPath = config('app.private_uploads').'/users';
|
|
||||||
|
|
||||||
if (isset($user->id)) {
|
|
||||||
$this->authorize('update', $user);
|
$this->authorize('update', $user);
|
||||||
|
|
||||||
$logActions = [];
|
|
||||||
$files = $request->file('file');
|
$files = $request->file('file');
|
||||||
|
|
||||||
if (is_null($files)) {
|
if (is_null($files)) {
|
||||||
|
@ -53,12 +47,9 @@ class UserFilesController extends Controller
|
||||||
if (! $logAction->save()) {
|
if (! $logAction->save()) {
|
||||||
return JsonResponse::create(['error' => 'Failed validation: '.print_r($logAction->getErrors(), true)], 500);
|
return JsonResponse::create(['error' => 'Failed validation: '.print_r($logAction->getErrors(), true)], 500);
|
||||||
}
|
}
|
||||||
$logActions[] = $logAction;
|
|
||||||
}
|
|
||||||
// dd($logActions);
|
|
||||||
return redirect()->back()->withFragment('files')->with('success', trans('admin/users/message.upload.success'));
|
return redirect()->back()->withFragment('files')->with('success', trans('admin/users/message.upload.success'));
|
||||||
}
|
}
|
||||||
return redirect()->back()->with('error', trans('admin/users/message.upload.nofiles'));
|
|
||||||
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -110,7 +101,7 @@ class UserFilesController extends Controller
|
||||||
* @return mixed
|
* @return mixed
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function show($userId = null, $fileId = null)
|
public function show(User $user, $fileId = null)
|
||||||
{
|
{
|
||||||
|
|
||||||
|
|
||||||
|
@ -118,8 +109,6 @@ class UserFilesController extends Controller
|
||||||
return redirect()->route('users.show')->with('error', 'Invalid file request');
|
return redirect()->route('users.show')->with('error', 'Invalid file request');
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($user = User::find($userId)) {
|
|
||||||
|
|
||||||
$this->authorize('view', $user);
|
$this->authorize('view', $user);
|
||||||
|
|
||||||
if ($log = Actionlog::whereNotNull('filename')->where('item_id', $user->id)->find($fileId)) {
|
if ($log = Actionlog::whereNotNull('filename')->where('item_id', $user->id)->find($fileId)) {
|
||||||
|
@ -135,12 +124,6 @@ class UserFilesController extends Controller
|
||||||
// The log record doesn't exist somehow
|
// The log record doesn't exist somehow
|
||||||
return redirect()->route('users.show', ['user' => $user])->with('error', trans('general.log_record_not_found'));
|
return redirect()->route('users.show', ['user' => $user])->with('error', trans('general.log_record_not_found'));
|
||||||
|
|
||||||
|
|
||||||
return redirect()->back()->with('error', trans('general.file_not_found'));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Redirect to the user management page if the user doesn't exist
|
|
||||||
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', ['id' => $userId]));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -182,11 +182,11 @@ class UsersController extends Controller
|
||||||
* @internal param int $id
|
* @internal param int $id
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function edit($id)
|
public function edit(User $user)
|
||||||
{
|
{
|
||||||
|
|
||||||
$this->authorize('update', User::class);
|
$this->authorize('update', User::class);
|
||||||
$user = User::with(['assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc'])->withTrashed()->find($id);
|
$user = User::with(['assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc'])->withTrashed()->find($user->id);
|
||||||
|
|
||||||
if ($user) {
|
if ($user) {
|
||||||
|
|
||||||
|
@ -201,7 +201,6 @@ class UsersController extends Controller
|
||||||
return view('users/edit', compact('user', 'groups', 'userGroups', 'permissions', 'userPermissions'))->with('item', $user);
|
return view('users/edit', compact('user', 'groups', 'userGroups', 'permissions', 'userPermissions'))->with('item', $user);
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', compact('id')));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -324,7 +323,7 @@ class UsersController extends Controller
|
||||||
* @return \Illuminate\Http\RedirectResponse
|
* @return \Illuminate\Http\RedirectResponse
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function destroy(DeleteUserRequest $request, $id = null)
|
public function destroy(DeleteUserRequest $request, $id)
|
||||||
{
|
{
|
||||||
$this->authorize('delete', User::class);
|
$this->authorize('delete', User::class);
|
||||||
|
|
||||||
|
@ -333,13 +332,6 @@ class UsersController extends Controller
|
||||||
$this->authorize('delete', $user);
|
$this->authorize('delete', $user);
|
||||||
|
|
||||||
if ($user->delete()) {
|
if ($user->delete()) {
|
||||||
if (Storage::disk('public')->exists('avatars/' . $user->avatar)) {
|
|
||||||
try {
|
|
||||||
Storage::disk('public')->delete('avatars/' . $user->avatar);
|
|
||||||
} catch (\Exception $e) {
|
|
||||||
Log::debug($e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return redirect()->route('users.index')->with('success', trans('admin/users/message.success.delete'));
|
return redirect()->route('users.index')->with('success', trans('admin/users/message.success.delete'));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -398,25 +390,20 @@ class UsersController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function show($userId = null)
|
public function show(User $user)
|
||||||
{
|
{
|
||||||
// Make sure the user can view users at all
|
// Make sure the user can view users at all
|
||||||
$this->authorize('view', User::class);
|
$this->authorize('view', User::class);
|
||||||
|
|
||||||
$user = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($userId);
|
$user = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($user->id);
|
||||||
|
|
||||||
// Make sure they can view this particular user
|
// Make sure they can view this particular user
|
||||||
$this->authorize('view', $user);
|
$this->authorize('view', $user);
|
||||||
|
|
||||||
if ($user) {
|
|
||||||
$userlog = $user->userlog->load('item');
|
$userlog = $user->userlog->load('item');
|
||||||
return view('users/view', compact('user', 'userlog'))->with('settings', Setting::getSettings());
|
return view('users/view', compact('user', 'userlog'))->with('settings', Setting::getSettings());
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', ['id' => $userId]));
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Return a view containing a pre-populated new user form,
|
* Return a view containing a pre-populated new user form,
|
||||||
|
@ -428,7 +415,7 @@ class UsersController extends Controller
|
||||||
* @return \Illuminate\Contracts\View\View
|
* @return \Illuminate\Contracts\View\View
|
||||||
* @throws \Illuminate\Auth\Access\AuthorizationException
|
* @throws \Illuminate\Auth\Access\AuthorizationException
|
||||||
*/
|
*/
|
||||||
public function getClone(Request $request, $id = null)
|
public function getClone(Request $request, User $user)
|
||||||
{
|
{
|
||||||
$this->authorize('create', User::class);
|
$this->authorize('create', User::class);
|
||||||
|
|
||||||
|
@ -438,7 +425,7 @@ class UsersController extends Controller
|
||||||
app('request')->request->set('permissions', $permissions);
|
app('request')->request->set('permissions', $permissions);
|
||||||
|
|
||||||
|
|
||||||
$user_to_clone = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($id);
|
$user_to_clone = User::with('assets', 'assets.model', 'consumables', 'accessories', 'licenses', 'userloc')->withTrashed()->find($user->id);
|
||||||
// Make sure they can view this particular user
|
// Make sure they can view this particular user
|
||||||
$this->authorize('view', $user_to_clone);
|
$this->authorize('view', $user_to_clone);
|
||||||
|
|
||||||
|
@ -468,10 +455,10 @@ class UsersController extends Controller
|
||||||
->with('user', $user)
|
->with('user', $user)
|
||||||
->with('groups', Group::pluck('name', 'id'))
|
->with('groups', Group::pluck('name', 'id'))
|
||||||
->with('userGroups', $userGroups)
|
->with('userGroups', $userGroups)
|
||||||
->with('clone_user', $user_to_clone);
|
->with('clone_user', $user_to_clone)
|
||||||
|
->with('item', $user);
|
||||||
}
|
}
|
||||||
|
|
||||||
return redirect()->route('users.index')->with('error', trans('admin/users/message.user_not_found', compact('id')));
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -24,7 +24,7 @@ class DeleteUserRequest extends FormRequest
|
||||||
|
|
||||||
public function prepareForValidation(): void
|
public function prepareForValidation(): void
|
||||||
{
|
{
|
||||||
$user_to_delete = User::withTrashed()->find(request()->route('user'));
|
$user_to_delete = User::withTrashed()->with('managesUsers')->find(request()->route('user'));
|
||||||
|
|
||||||
if ($user_to_delete) {
|
if ($user_to_delete) {
|
||||||
$this->merge([
|
$this->merge([
|
||||||
|
@ -61,7 +61,8 @@ class DeleteUserRequest extends FormRequest
|
||||||
public function messages(): array
|
public function messages(): array
|
||||||
{
|
{
|
||||||
|
|
||||||
$user_to_delete = User::withTrashed()->find(request()->route('user'));
|
$user_to_delete = User::withTrashed()->with('managesUsers')->find(request()->route('user'));
|
||||||
|
|
||||||
$messages = [];
|
$messages = [];
|
||||||
|
|
||||||
if ($user_to_delete) {
|
if ($user_to_delete) {
|
||||||
|
|
|
@ -2,8 +2,11 @@
|
||||||
|
|
||||||
namespace App\Http\Requests;
|
namespace App\Http\Requests;
|
||||||
|
|
||||||
|
|
||||||
|
use App\Models\Labels\Label;
|
||||||
use Illuminate\Foundation\Http\FormRequest;
|
use Illuminate\Foundation\Http\FormRequest;
|
||||||
use Illuminate\Support\Facades\Gate;
|
use Illuminate\Support\Facades\Gate;
|
||||||
|
use Illuminate\Validation\Rule;
|
||||||
|
|
||||||
class StoreLabelSettings extends FormRequest
|
class StoreLabelSettings extends FormRequest
|
||||||
{
|
{
|
||||||
|
@ -22,6 +25,10 @@ class StoreLabelSettings extends FormRequest
|
||||||
*/
|
*/
|
||||||
public function rules(): array
|
public function rules(): array
|
||||||
{
|
{
|
||||||
|
$names = Label::find()?->map(function ($label) {
|
||||||
|
return $label->getName();
|
||||||
|
})->values()->toArray();
|
||||||
|
|
||||||
return [
|
return [
|
||||||
'labels_per_page' => 'numeric',
|
'labels_per_page' => 'numeric',
|
||||||
'labels_width' => 'numeric',
|
'labels_width' => 'numeric',
|
||||||
|
@ -36,6 +43,10 @@ class StoreLabelSettings extends FormRequest
|
||||||
'labels_pagewidth' => 'numeric|nullable',
|
'labels_pagewidth' => 'numeric|nullable',
|
||||||
'labels_pageheight' => 'numeric|nullable',
|
'labels_pageheight' => 'numeric|nullable',
|
||||||
'qr_text' => 'max:31|nullable',
|
'qr_text' => 'max:31|nullable',
|
||||||
|
'label2_template' => [
|
||||||
|
'required',
|
||||||
|
Rule::in($names),
|
||||||
|
],
|
||||||
];
|
];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -26,9 +26,9 @@ class StoreNotificationSettings extends FormRequest
|
||||||
return [
|
return [
|
||||||
'alert_email' => 'email_array|nullable',
|
'alert_email' => 'email_array|nullable',
|
||||||
'admin_cc_email' => 'email_array|nullable',
|
'admin_cc_email' => 'email_array|nullable',
|
||||||
'alert_threshold' => 'numeric|nullable|gt:0',
|
'alert_threshold' => 'numeric|nullable',
|
||||||
'alert_interval' => 'numeric|nullable|gt:0',
|
'alert_interval' => 'numeric|nullable|gt:0',
|
||||||
'audit_warning_days' => 'numeric|nullable|gt:0',
|
'audit_warning_days' => 'numeric|nullable',
|
||||||
'due_checkin_days' => 'numeric|nullable|gt:0',
|
'due_checkin_days' => 'numeric|nullable|gt:0',
|
||||||
'audit_interval' => 'numeric|nullable|gt:0',
|
'audit_interval' => 'numeric|nullable|gt:0',
|
||||||
];
|
];
|
||||||
|
|
|
@ -140,13 +140,13 @@ class ActionlogsTransformer
|
||||||
} else {
|
} else {
|
||||||
if ($actionlog->item) {
|
if ($actionlog->item) {
|
||||||
if ($actionlog->itemType() == 'asset') {
|
if ($actionlog->itemType() == 'asset') {
|
||||||
$file_url = route('show/assetfile', ['assetId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
$file_url = route('show/assetfile', ['asset' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
||||||
} elseif ($actionlog->itemType() == 'accessory') {
|
} elseif ($actionlog->itemType() == 'accessory') {
|
||||||
$file_url = route('show.accessoryfile', ['accessoryId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
$file_url = route('show.accessoryfile', ['accessoryId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
||||||
} elseif ($actionlog->itemType() == 'license') {
|
} elseif ($actionlog->itemType() == 'license') {
|
||||||
$file_url = route('show.licensefile', ['licenseId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
$file_url = route('show.licensefile', ['licenseId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
||||||
} elseif ($actionlog->itemType() == 'user') {
|
} elseif ($actionlog->itemType() == 'user') {
|
||||||
$file_url = route('show/userfile', ['userId' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
$file_url = route('show/userfile', ['user' => $actionlog->item->id, 'fileId' => $actionlog->id]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -309,6 +309,7 @@ class AssetsTransformer
|
||||||
'id' => $accessory_checkout->accessory->id,
|
'id' => $accessory_checkout->accessory->id,
|
||||||
'name' => $accessory_checkout->accessory->name,
|
'name' => $accessory_checkout->accessory->name,
|
||||||
],
|
],
|
||||||
|
'assigned_to' => $accessory_checkout->assigned_to,
|
||||||
'image' => ($accessory_checkout->accessory->image) ? Storage::disk('public')->url('accessories/'.e($accessory_checkout->accessory->image)) : null,
|
'image' => ($accessory_checkout->accessory->image) ? Storage::disk('public')->url('accessories/'.e($accessory_checkout->accessory->image)) : null,
|
||||||
'note' => $accessory_checkout->note ? e($accessory_checkout->note) : null,
|
'note' => $accessory_checkout->note ? e($accessory_checkout->note) : null,
|
||||||
'created_by' => $accessory_checkout->adminuser ? [
|
'created_by' => $accessory_checkout->adminuser ? [
|
||||||
|
|
|
@ -66,6 +66,7 @@ class CategoriesTransformer
|
||||||
'id' => (int) $category->adminuser->id,
|
'id' => (int) $category->adminuser->id,
|
||||||
'name'=> e($category->adminuser->present()->fullName()),
|
'name'=> e($category->adminuser->present()->fullName()),
|
||||||
] : null,
|
] : null,
|
||||||
|
'notes' => Helper::parseEscapedMarkedownInline($category->notes),
|
||||||
'created_at' => Helper::getFormattedDateObject($category->created_at, 'datetime'),
|
'created_at' => Helper::getFormattedDateObject($category->created_at, 'datetime'),
|
||||||
'updated_at' => Helper::getFormattedDateObject($category->updated_at, 'datetime'),
|
'updated_at' => Helper::getFormattedDateObject($category->updated_at, 'datetime'),
|
||||||
];
|
];
|
||||||
|
|
|
@ -40,6 +40,7 @@ class CompaniesTransformer
|
||||||
'id' => (int) $company->adminuser->id,
|
'id' => (int) $company->adminuser->id,
|
||||||
'name'=> e($company->adminuser->present()->fullName()),
|
'name'=> e($company->adminuser->present()->fullName()),
|
||||||
] : null,
|
] : null,
|
||||||
|
'notes' => Helper::parseEscapedMarkedownInline($company->notes),
|
||||||
'created_at' => Helper::getFormattedDateObject($company->created_at, 'datetime'),
|
'created_at' => Helper::getFormattedDateObject($company->created_at, 'datetime'),
|
||||||
'updated_at' => Helper::getFormattedDateObject($company->updated_at, 'datetime'),
|
'updated_at' => Helper::getFormattedDateObject($company->updated_at, 'datetime'),
|
||||||
];
|
];
|
||||||
|
|
|
@ -62,7 +62,7 @@ class ComponentsTransformer
|
||||||
'checkout' => Gate::allows('checkout', Component::class),
|
'checkout' => Gate::allows('checkout', Component::class),
|
||||||
'checkin' => Gate::allows('checkin', Component::class),
|
'checkin' => Gate::allows('checkin', Component::class),
|
||||||
'update' => Gate::allows('update', Component::class),
|
'update' => Gate::allows('update', Component::class),
|
||||||
'delete' => Gate::allows('delete', Component::class),
|
'delete' => $component->isDeletable(),
|
||||||
];
|
];
|
||||||
$array += $permissions_array;
|
$array += $permissions_array;
|
||||||
|
|
||||||
|
|
|
@ -44,6 +44,7 @@ class DepartmentsTransformer
|
||||||
'name' => e($department->location->name),
|
'name' => e($department->location->name),
|
||||||
] : null,
|
] : null,
|
||||||
'users_count' => e($department->users_count),
|
'users_count' => e($department->users_count),
|
||||||
|
'notes' => Helper::parseEscapedMarkedownInline($department->notes),
|
||||||
'created_at' => Helper::getFormattedDateObject($department->created_at, 'datetime'),
|
'created_at' => Helper::getFormattedDateObject($department->created_at, 'datetime'),
|
||||||
'updated_at' => Helper::getFormattedDateObject($department->updated_at, 'datetime'),
|
'updated_at' => Helper::getFormattedDateObject($department->updated_at, 'datetime'),
|
||||||
];
|
];
|
||||||
|
|
|
@ -26,6 +26,7 @@ class GroupsTransformer
|
||||||
'name' => e($group->name),
|
'name' => e($group->name),
|
||||||
'permissions' => json_decode($group->permissions),
|
'permissions' => json_decode($group->permissions),
|
||||||
'users_count' => (int) $group->users_count,
|
'users_count' => (int) $group->users_count,
|
||||||
|
'notes' => Helper::parseEscapedMarkedownInline($group->notes),
|
||||||
'created_by' => ($group->adminuser) ? [
|
'created_by' => ($group->adminuser) ? [
|
||||||
'id' => (int) $group->adminuser->id,
|
'id' => (int) $group->adminuser->id,
|
||||||
'name'=> e($group->adminuser->present()->fullName()),
|
'name'=> e($group->adminuser->present()->fullName()),
|
||||||
|
|
|
@ -55,6 +55,7 @@ class LocationsTransformer
|
||||||
'users_count' => (int) $location->users_count,
|
'users_count' => (int) $location->users_count,
|
||||||
'currency' => ($location->currency) ? e($location->currency) : null,
|
'currency' => ($location->currency) ? e($location->currency) : null,
|
||||||
'ldap_ou' => ($location->ldap_ou) ? e($location->ldap_ou) : null,
|
'ldap_ou' => ($location->ldap_ou) ? e($location->ldap_ou) : null,
|
||||||
|
'notes' => Helper::parseEscapedMarkedownInline($location->notes),
|
||||||
'created_at' => Helper::getFormattedDateObject($location->created_at, 'datetime'),
|
'created_at' => Helper::getFormattedDateObject($location->created_at, 'datetime'),
|
||||||
'updated_at' => Helper::getFormattedDateObject($location->updated_at, 'datetime'),
|
'updated_at' => Helper::getFormattedDateObject($location->updated_at, 'datetime'),
|
||||||
'parent' => ($location->parent) ? [
|
'parent' => ($location->parent) ? [
|
||||||
|
@ -99,6 +100,7 @@ class LocationsTransformer
|
||||||
|
|
||||||
$array = [
|
$array = [
|
||||||
'id' => $accessory_checkout->id,
|
'id' => $accessory_checkout->id,
|
||||||
|
'assigned_to' => $accessory_checkout->assigned_to,
|
||||||
'accessory' => [
|
'accessory' => [
|
||||||
'id' => $accessory_checkout->accessory->id,
|
'id' => $accessory_checkout->accessory->id,
|
||||||
'name' => $accessory_checkout->accessory->name,
|
'name' => $accessory_checkout->accessory->name,
|
||||||
|
|
|
@ -37,6 +37,7 @@ class ManufacturersTransformer
|
||||||
'consumables_count' => (int) $manufacturer->consumables_count,
|
'consumables_count' => (int) $manufacturer->consumables_count,
|
||||||
'accessories_count' => (int) $manufacturer->accessories_count,
|
'accessories_count' => (int) $manufacturer->accessories_count,
|
||||||
'components_count' => (int) $manufacturer->components_count,
|
'components_count' => (int) $manufacturer->components_count,
|
||||||
|
'notes' => Helper::parseEscapedMarkedownInline($manufacturer->notes),
|
||||||
'created_by' => ($manufacturer->adminuser) ? [
|
'created_by' => ($manufacturer->adminuser) ? [
|
||||||
'id' => (int) $manufacturer->adminuser->id,
|
'id' => (int) $manufacturer->adminuser->id,
|
||||||
'name'=> e($manufacturer->adminuser->present()->fullName()),
|
'name'=> e($manufacturer->adminuser->present()->fullName()),
|
||||||
|
|
|
@ -74,6 +74,8 @@ class LocationImporter extends ItemImporter
|
||||||
$this->item['ldap_ou'] = trim($this->findCsvMatch($row, 'ldap_ou'));
|
$this->item['ldap_ou'] = trim($this->findCsvMatch($row, 'ldap_ou'));
|
||||||
$this->item['manager'] = trim($this->findCsvMatch($row, 'manager'));
|
$this->item['manager'] = trim($this->findCsvMatch($row, 'manager'));
|
||||||
$this->item['manager_username'] = trim($this->findCsvMatch($row, 'manager_username'));
|
$this->item['manager_username'] = trim($this->findCsvMatch($row, 'manager_username'));
|
||||||
|
$this->item['notes'] = trim($this->findCsvMatch($row, 'notes'));
|
||||||
|
|
||||||
|
|
||||||
if ($this->findCsvMatch($row, 'parent_location')) {
|
if ($this->findCsvMatch($row, 'parent_location')) {
|
||||||
$this->item['parent_id'] = $this->createOrFetchLocation(trim($this->findCsvMatch($row, 'parent_location')));
|
$this->item['parent_id'] = $this->createOrFetchLocation(trim($this->findCsvMatch($row, 'parent_location')));
|
||||||
|
|
|
@ -1,81 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace App\LegacyEncrypter;
|
|
||||||
|
|
||||||
use Illuminate\Contracts\Encryption\DecryptException;
|
|
||||||
|
|
||||||
abstract class BaseEncrypter
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* The encryption key.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
protected $key;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a MAC for the given value.
|
|
||||||
*
|
|
||||||
* @param string $iv
|
|
||||||
* @param string $value
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
protected function hash($iv, $value)
|
|
||||||
{
|
|
||||||
return hash_hmac('sha256', $iv.$value, $this->key);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the JSON array from the given payload.
|
|
||||||
*
|
|
||||||
* @param string $payload
|
|
||||||
* @return array
|
|
||||||
*
|
|
||||||
* @throws \Illuminate\Contracts\Encryption\DecryptException
|
|
||||||
*/
|
|
||||||
protected function getJsonPayload($payload)
|
|
||||||
{
|
|
||||||
$payload = json_decode(base64_decode($payload), true);
|
|
||||||
|
|
||||||
// If the payload is not valid JSON or does not have the proper keys set we will
|
|
||||||
// assume it is invalid and bail out of the routine since we will not be able
|
|
||||||
// to decrypt the given value. We'll also check the MAC for this encryption.
|
|
||||||
if (! $payload || $this->invalidPayload($payload)) {
|
|
||||||
throw new DecryptException('The payload is invalid.');
|
|
||||||
}
|
|
||||||
|
|
||||||
if (! $this->validMac($payload)) {
|
|
||||||
throw new DecryptException('The MAC is invalid.');
|
|
||||||
}
|
|
||||||
|
|
||||||
return $payload;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Verify that the encryption payload is valid.
|
|
||||||
*
|
|
||||||
* @param array|mixed $data
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
protected function invalidPayload($data)
|
|
||||||
{
|
|
||||||
return ! is_array($data) || ! isset($data['iv']) || ! isset($data['value']) || ! isset($data['mac']);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine if the MAC for the given payload is valid.
|
|
||||||
*
|
|
||||||
* @param array $payload
|
|
||||||
* @return bool
|
|
||||||
*
|
|
||||||
* @throws \RuntimeException
|
|
||||||
*/
|
|
||||||
protected function validMac(array $payload)
|
|
||||||
{
|
|
||||||
$bytes = random_bytes(16);
|
|
||||||
|
|
||||||
$calcMac = hash_hmac('sha256', $this->hash($payload['iv'], $payload['value']), $bytes, true);
|
|
||||||
|
|
||||||
return hash_equals(hash_hmac('sha256', $payload['mac'], $bytes, true), $calcMac);
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -1,214 +0,0 @@
|
||||||
<?php
|
|
||||||
|
|
||||||
namespace App\LegacyEncrypter;
|
|
||||||
|
|
||||||
use Exception;
|
|
||||||
use Illuminate\Contracts\Encryption\DecryptException;
|
|
||||||
use Illuminate\Contracts\Encryption\Encrypter as EncrypterContract;
|
|
||||||
use Illuminate\Contracts\Encryption\EncryptException;
|
|
||||||
use RuntimeException;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* @deprecated since version 5.1. Use Illuminate\Encryption\Encrypter.
|
|
||||||
*/
|
|
||||||
class McryptEncrypter extends BaseEncrypter implements EncrypterContract
|
|
||||||
{
|
|
||||||
/**
|
|
||||||
* The algorithm used for encryption.
|
|
||||||
*
|
|
||||||
* @var string
|
|
||||||
*/
|
|
||||||
protected $cipher;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* The block size of the cipher.
|
|
||||||
*
|
|
||||||
* @var int
|
|
||||||
*/
|
|
||||||
protected $block;
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Create a new encrypter instance.
|
|
||||||
*
|
|
||||||
* @param string $key
|
|
||||||
* @param string $cipher
|
|
||||||
* @return void
|
|
||||||
*
|
|
||||||
* @throws \RuntimeException
|
|
||||||
*/
|
|
||||||
public function __construct($key, $cipher = MCRYPT_RIJNDAEL_128)
|
|
||||||
{
|
|
||||||
$key = (string) $key;
|
|
||||||
|
|
||||||
if (static::supported($key, $cipher)) {
|
|
||||||
$this->key = $key;
|
|
||||||
$this->cipher = $cipher;
|
|
||||||
$this->block = mcrypt_get_iv_size($this->cipher, MCRYPT_MODE_CBC);
|
|
||||||
} else {
|
|
||||||
throw new RuntimeException('The only supported ciphers are MCRYPT_RIJNDAEL_128 and MCRYPT_RIJNDAEL_256.');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine if the given key and cipher combination is valid.
|
|
||||||
*
|
|
||||||
* @param string $key
|
|
||||||
* @param string $cipher
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
public static function supported($key, $cipher)
|
|
||||||
{
|
|
||||||
return defined('MCRYPT_RIJNDAEL_128') &&
|
|
||||||
($cipher === MCRYPT_RIJNDAEL_128 || $cipher === MCRYPT_RIJNDAEL_256);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Encrypt the given value.
|
|
||||||
*
|
|
||||||
* @param string $value
|
|
||||||
* @return string
|
|
||||||
*
|
|
||||||
* @throws \Illuminate\Contracts\Encryption\EncryptException
|
|
||||||
*/
|
|
||||||
public function encrypt($value, $serialize = true)
|
|
||||||
{
|
|
||||||
$iv = mcrypt_create_iv($this->getIvSize(), $this->getRandomizer());
|
|
||||||
|
|
||||||
$value = base64_encode($this->padAndMcrypt($value, $iv));
|
|
||||||
|
|
||||||
// Once we have the encrypted value we will go ahead base64_encode the input
|
|
||||||
// vector and create the MAC for the encrypted value so we can verify its
|
|
||||||
// authenticity. Then, we'll JSON encode the data in a "payload" array.
|
|
||||||
$mac = $this->hash($iv = base64_encode($iv), $value);
|
|
||||||
|
|
||||||
$json = json_encode(compact('iv', 'value', 'mac'));
|
|
||||||
|
|
||||||
if (! is_string($json)) {
|
|
||||||
throw new EncryptException('Could not encrypt the data.');
|
|
||||||
}
|
|
||||||
|
|
||||||
return base64_encode($json);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Pad and use mcrypt on the given value and input vector.
|
|
||||||
*
|
|
||||||
* @param string $value
|
|
||||||
* @param string $iv
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
protected function padAndMcrypt($value, $iv)
|
|
||||||
{
|
|
||||||
$value = $this->addPadding(serialize($value));
|
|
||||||
|
|
||||||
return mcrypt_encrypt($this->cipher, $this->key, $value, MCRYPT_MODE_CBC, $iv);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Decrypt the given value.
|
|
||||||
*
|
|
||||||
* @param string $payload
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
public function decrypt($payload, $unserialize = true)
|
|
||||||
{
|
|
||||||
$payload = $this->getJsonPayload($payload);
|
|
||||||
|
|
||||||
// We'll go ahead and remove the PKCS7 padding from the encrypted value before
|
|
||||||
// we decrypt it. Once we have the de-padded value, we will grab the vector
|
|
||||||
// and decrypt the data, passing back the unserialized from of the value.
|
|
||||||
$value = base64_decode($payload['value']);
|
|
||||||
|
|
||||||
$iv = base64_decode($payload['iv']);
|
|
||||||
|
|
||||||
return unserialize($this->stripPadding($this->mcryptDecrypt($value, $iv)));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Run the mcrypt decryption routine for the value.
|
|
||||||
*
|
|
||||||
* @param string $value
|
|
||||||
* @param string $iv
|
|
||||||
* @return string
|
|
||||||
*
|
|
||||||
* @throws \Illuminate\Contracts\Encryption\DecryptException
|
|
||||||
*/
|
|
||||||
protected function mcryptDecrypt($value, $iv)
|
|
||||||
{
|
|
||||||
try {
|
|
||||||
return mcrypt_decrypt($this->cipher, $this->key, $value, MCRYPT_MODE_CBC, $iv);
|
|
||||||
} catch (Exception $e) {
|
|
||||||
throw new DecryptException($e->getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Add PKCS7 padding to a given value.
|
|
||||||
*
|
|
||||||
* @param string $value
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
protected function addPadding($value)
|
|
||||||
{
|
|
||||||
$pad = $this->block - (strlen($value) % $this->block);
|
|
||||||
|
|
||||||
return $value.str_repeat(chr($pad), $pad);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Remove the padding from the given value.
|
|
||||||
*
|
|
||||||
* @param string $value
|
|
||||||
* @return string
|
|
||||||
*/
|
|
||||||
protected function stripPadding($value)
|
|
||||||
{
|
|
||||||
$pad = ord($value[($len = strlen($value)) - 1]);
|
|
||||||
|
|
||||||
return $this->paddingIsValid($pad, $value) ? substr($value, 0, $len - $pad) : $value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Determine if the given padding for a value is valid.
|
|
||||||
*
|
|
||||||
* @param string $pad
|
|
||||||
* @param string $value
|
|
||||||
* @return bool
|
|
||||||
*/
|
|
||||||
protected function paddingIsValid($pad, $value)
|
|
||||||
{
|
|
||||||
$beforePad = strlen($value) - $pad;
|
|
||||||
|
|
||||||
return substr($value, $beforePad) == str_repeat(substr($value, -1), $pad);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the IV size for the cipher.
|
|
||||||
*
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
protected function getIvSize()
|
|
||||||
{
|
|
||||||
return mcrypt_get_iv_size($this->cipher, MCRYPT_MODE_CBC);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the random data source available for the OS.
|
|
||||||
*
|
|
||||||
* @return int
|
|
||||||
*/
|
|
||||||
protected function getRandomizer()
|
|
||||||
{
|
|
||||||
if (defined('MCRYPT_DEV_URANDOM')) {
|
|
||||||
return MCRYPT_DEV_URANDOM;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (defined('MCRYPT_DEV_RANDOM')) {
|
|
||||||
return MCRYPT_DEV_RANDOM;
|
|
||||||
}
|
|
||||||
|
|
||||||
mt_srand();
|
|
||||||
|
|
||||||
return MCRYPT_RAND;
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -115,7 +115,7 @@ class CheckoutableListener
|
||||||
}
|
}
|
||||||
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_fail') );
|
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) .trans('admin/settings/message.webhook.webhook_fail') );
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
Log::error(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [
|
Log::warning(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [
|
||||||
'error' => $e->getMessage(),
|
'error' => $e->getMessage(),
|
||||||
'webhook_endpoint' => Setting::getSettings()->webhook_endpoint,
|
'webhook_endpoint' => Setting::getSettings()->webhook_endpoint,
|
||||||
'event' => $event,
|
'event' => $event,
|
||||||
|
@ -211,7 +211,7 @@ class CheckoutableListener
|
||||||
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) . trans('admin/settings/message.webhook.webhook_fail'));
|
return redirect()->back()->with('warning', ucfirst(Setting::getSettings()->webhook_selected) . trans('admin/settings/message.webhook.webhook_fail'));
|
||||||
}
|
}
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
Log::error(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [
|
Log::warning(ucfirst(Setting::getSettings()->webhook_selected) . ' webhook notification failed:', [
|
||||||
'error' => $e->getMessage(),
|
'error' => $e->getMessage(),
|
||||||
'webhook_endpoint' => Setting::getSettings()->webhook_endpoint,
|
'webhook_endpoint' => Setting::getSettings()->webhook_endpoint,
|
||||||
'event' => $event,
|
'event' => $event,
|
||||||
|
|
|
@ -17,7 +17,6 @@ use App\Events\ItemAccepted;
|
||||||
use App\Events\ItemDeclined;
|
use App\Events\ItemDeclined;
|
||||||
use App\Events\LicenseCheckedIn;
|
use App\Events\LicenseCheckedIn;
|
||||||
use App\Events\LicenseCheckedOut;
|
use App\Events\LicenseCheckedOut;
|
||||||
use App\Events\NoteAdded;
|
|
||||||
use App\Models\Actionlog;
|
use App\Models\Actionlog;
|
||||||
use App\Models\User;
|
use App\Models\User;
|
||||||
use App\Models\LicenseSeat;
|
use App\Models\LicenseSeat;
|
||||||
|
@ -129,23 +128,6 @@ class LogListener
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Note is added to action log
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
public function onNoteAdded(NoteAdded $event)
|
|
||||||
{
|
|
||||||
$logaction = new Actionlog();
|
|
||||||
$logaction->item_id = $event->itemNoteAddedOn->id;
|
|
||||||
$logaction->item_type = get_class($event->itemNoteAddedOn);
|
|
||||||
$logaction->note = $event->note; //this is the received alphanumeric text from the box
|
|
||||||
$logaction->created_by = $event->noteAddedBy->id;
|
|
||||||
$logaction->action_type = 'note_added';
|
|
||||||
$logaction->save();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Register the listeners for the subscriber.
|
* Register the listeners for the subscriber.
|
||||||
*
|
*
|
||||||
|
|
|
@ -12,11 +12,11 @@ class CategoryEditForm extends Component
|
||||||
|
|
||||||
public $originalSendCheckInEmailValue;
|
public $originalSendCheckInEmailValue;
|
||||||
|
|
||||||
public $requireAcceptance;
|
public bool $requireAcceptance;
|
||||||
|
|
||||||
public $sendCheckInEmail;
|
public bool $sendCheckInEmail;
|
||||||
|
|
||||||
public $useDefaultEula;
|
public bool $useDefaultEula;
|
||||||
|
|
||||||
public function mount()
|
public function mount()
|
||||||
{
|
{
|
||||||
|
|
|
@ -342,6 +342,7 @@ class Importer extends Component
|
||||||
'manager_username' => trans('general.importer.manager_username'),
|
'manager_username' => trans('general.importer.manager_username'),
|
||||||
'manager' => trans('general.importer.manager_full_name'),
|
'manager' => trans('general.importer.manager_full_name'),
|
||||||
'parent_location' => trans('admin/locations/table.parent'),
|
'parent_location' => trans('admin/locations/table.parent'),
|
||||||
|
'notes' => trans('general.notes'),
|
||||||
];
|
];
|
||||||
|
|
||||||
$this->assetmodels_fields = [
|
$this->assetmodels_fields = [
|
||||||
|
|
|
@ -20,10 +20,12 @@ class CheckoutAssetMail extends Mailable
|
||||||
{
|
{
|
||||||
use Queueable, SerializesModels;
|
use Queueable, SerializesModels;
|
||||||
|
|
||||||
|
private bool $firstTimeSending;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a new message instance.
|
* Create a new message instance.
|
||||||
*/
|
*/
|
||||||
public function __construct(Asset $asset, $checkedOutTo, User $checkedOutBy, $acceptance, $note)
|
public function __construct(Asset $asset, $checkedOutTo, User $checkedOutBy, $acceptance, $note, bool $firstTimeSending = true)
|
||||||
{
|
{
|
||||||
$this->item = $asset;
|
$this->item = $asset;
|
||||||
$this->admin = $checkedOutBy;
|
$this->admin = $checkedOutBy;
|
||||||
|
@ -36,6 +38,8 @@ class CheckoutAssetMail extends Mailable
|
||||||
$this->last_checkout = '';
|
$this->last_checkout = '';
|
||||||
$this->expected_checkin = '';
|
$this->expected_checkin = '';
|
||||||
|
|
||||||
|
$this->firstTimeSending = $firstTimeSending;
|
||||||
|
|
||||||
if ($this->item->last_checkout) {
|
if ($this->item->last_checkout) {
|
||||||
$this->last_checkout = Helper::getFormattedDateObject($this->item->last_checkout, 'date',
|
$this->last_checkout = Helper::getFormattedDateObject($this->item->last_checkout, 'date',
|
||||||
false);
|
false);
|
||||||
|
@ -56,7 +60,7 @@ class CheckoutAssetMail extends Mailable
|
||||||
|
|
||||||
return new Envelope(
|
return new Envelope(
|
||||||
from: $from,
|
from: $from,
|
||||||
subject: trans('mail.Asset_Checkout_Notification'),
|
subject: $this->getSubject(),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -107,4 +111,13 @@ class CheckoutAssetMail extends Mailable
|
||||||
{
|
{
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function getSubject(): string
|
||||||
|
{
|
||||||
|
if ($this->firstTimeSending) {
|
||||||
|
return trans('mail.Asset_Checkout_Notification');
|
||||||
|
}
|
||||||
|
|
||||||
|
return trans('mail.unaccepted_asset_reminder');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
62
app/Mail/ExpiringAssetsMail.php
Normal file
62
app/Mail/ExpiringAssetsMail.php
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Mail;
|
||||||
|
|
||||||
|
use Illuminate\Bus\Queueable;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Mail\Mailable;
|
||||||
|
use Illuminate\Mail\Mailables\Address;
|
||||||
|
use Illuminate\Mail\Mailables\Content;
|
||||||
|
use Illuminate\Mail\Mailables\Envelope;
|
||||||
|
use Illuminate\Queue\SerializesModels;
|
||||||
|
|
||||||
|
class ExpiringAssetsMail extends Mailable
|
||||||
|
{
|
||||||
|
use Queueable, SerializesModels;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new message instance.
|
||||||
|
*/
|
||||||
|
public function __construct($params, $threshold)
|
||||||
|
{
|
||||||
|
$this->assets = $params;
|
||||||
|
$this->threshold = $threshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the message envelope.
|
||||||
|
*/
|
||||||
|
public function envelope(): Envelope
|
||||||
|
{
|
||||||
|
$from = new Address(config('mail.from.address'), config('mail.from.name'));
|
||||||
|
|
||||||
|
return new Envelope(
|
||||||
|
from: $from,
|
||||||
|
subject: trans('mail.Expiring_Assets_Report'),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the message content definition.
|
||||||
|
*/
|
||||||
|
public function content(): Content
|
||||||
|
{
|
||||||
|
return new Content(
|
||||||
|
markdown: 'notifications.markdown.report-expiring-assets',
|
||||||
|
with: [
|
||||||
|
'assets' => $this->assets,
|
||||||
|
'threshold' => $this->threshold,
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the attachments for the message.
|
||||||
|
*
|
||||||
|
* @return array<int, \Illuminate\Mail\Mailables\Attachment>
|
||||||
|
*/
|
||||||
|
public function attachments(): array
|
||||||
|
{
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
62
app/Mail/ExpiringLicenseMail.php
Normal file
62
app/Mail/ExpiringLicenseMail.php
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Mail;
|
||||||
|
|
||||||
|
use Illuminate\Bus\Queueable;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Mail\Mailable;
|
||||||
|
use Illuminate\Mail\Mailables\Address;
|
||||||
|
use Illuminate\Mail\Mailables\Content;
|
||||||
|
use Illuminate\Mail\Mailables\Envelope;
|
||||||
|
use Illuminate\Queue\SerializesModels;
|
||||||
|
|
||||||
|
class ExpiringLicenseMail extends Mailable
|
||||||
|
{
|
||||||
|
use Queueable, SerializesModels;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new message instance.
|
||||||
|
*/
|
||||||
|
public function __construct($params, $threshold)
|
||||||
|
{
|
||||||
|
$this->licenses = $params;
|
||||||
|
$this->threshold = $threshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the message envelope.
|
||||||
|
*/
|
||||||
|
public function envelope(): Envelope
|
||||||
|
{
|
||||||
|
$from = new Address(config('mail.from.address'), config('mail.from.name'));
|
||||||
|
|
||||||
|
return new Envelope(
|
||||||
|
from: $from,
|
||||||
|
subject: trans('mail.Expiring_Licenses_Report'),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the message content definition.
|
||||||
|
*/
|
||||||
|
public function content(): Content
|
||||||
|
{
|
||||||
|
return new Content(
|
||||||
|
markdown: 'notifications.markdown.report-expiring-licenses',
|
||||||
|
with: [
|
||||||
|
'licenses' => $this->licenses,
|
||||||
|
'threshold' => $this->threshold,
|
||||||
|
]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the attachments for the message.
|
||||||
|
*
|
||||||
|
* @return array<int, \Illuminate\Mail\Mailables\Attachment>
|
||||||
|
*/
|
||||||
|
public function attachments(): array
|
||||||
|
{
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
65
app/Mail/SendUpcomingAuditMail.php
Normal file
65
app/Mail/SendUpcomingAuditMail.php
Normal file
|
@ -0,0 +1,65 @@
|
||||||
|
<?php
|
||||||
|
|
||||||
|
namespace App\Mail;
|
||||||
|
|
||||||
|
use Illuminate\Bus\Queueable;
|
||||||
|
use Illuminate\Contracts\Queue\ShouldQueue;
|
||||||
|
use Illuminate\Mail\Mailable;
|
||||||
|
use Illuminate\Mail\Mailables\Address;
|
||||||
|
use Illuminate\Mail\Mailables\Content;
|
||||||
|
use Illuminate\Mail\Mailables\Envelope;
|
||||||
|
use Illuminate\Queue\SerializesModels;
|
||||||
|
|
||||||
|
class SendUpcomingAuditMail extends Mailable
|
||||||
|
{
|
||||||
|
use Queueable, SerializesModels;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create a new message instance.
|
||||||
|
*/
|
||||||
|
public function __construct($params, $threshold)
|
||||||
|
{
|
||||||
|
$this->assets = $params;
|
||||||
|
$this->threshold = $threshold;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the message envelope.
|
||||||
|
*/
|
||||||
|
public function envelope(): Envelope
|
||||||
|
{
|
||||||
|
$from = new Address(config('mail.from.address'), config('mail.from.name'));
|
||||||
|
|
||||||
|
return new Envelope(
|
||||||
|
from: $from,
|
||||||
|
subject: trans_choice('mail.upcoming-audits', $this->assets->count(), ['count' => $this->assets->count(), 'threshold' => $this->threshold]),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the message content definition.
|
||||||
|
*/
|
||||||
|
public function content(): Content
|
||||||
|
{
|
||||||
|
|
||||||
|
|
||||||
|
return new Content(
|
||||||
|
|
||||||
|
markdown: 'notifications.markdown.upcoming-audits',
|
||||||
|
with: [
|
||||||
|
'assets' => $this->assets,
|
||||||
|
'threshold' => $this->threshold,
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the attachments for the message.
|
||||||
|
*
|
||||||
|
* @return array<int, \Illuminate\Mail\Mailables\Attachment>
|
||||||
|
*/
|
||||||
|
public function attachments(): array
|
||||||
|
{
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
}
|
|
@ -19,9 +19,10 @@ class UnacceptedAssetReminderMail extends Mailable
|
||||||
*/
|
*/
|
||||||
public function __construct($checkout_info, $count)
|
public function __construct($checkout_info, $count)
|
||||||
{
|
{
|
||||||
|
|
||||||
$this->count = $count;
|
$this->count = $count;
|
||||||
$this->target = $checkout_info['acceptance']?->assignedTo;
|
$this->target = $checkout_info?->assignedTo;
|
||||||
$this->acceptance = $checkout_info['acceptance'];
|
$this->acceptance = $checkout_info;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -57,7 +57,7 @@ class AccessoryCheckout extends Model
|
||||||
*/
|
*/
|
||||||
public function adminuser()
|
public function adminuser()
|
||||||
{
|
{
|
||||||
return $this->hasOne(\App\Models\User::class, 'created_by');
|
return $this->hasOne(\App\Models\User::class, 'id', 'created_by');
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -118,7 +118,7 @@ class AccessoryCheckout extends Model
|
||||||
$query->where('assigned_type', '=', Location::class);
|
$query->where('assigned_type', '=', Location::class);
|
||||||
}
|
}
|
||||||
|
|
||||||
public function scopeAssetAssigned(Builder $query): void
|
public function scopeAssetsAssigned(Builder $query): void
|
||||||
{
|
{
|
||||||
$query->where('assigned_type', '=', Asset::class);
|
$query->where('assigned_type', '=', Asset::class);
|
||||||
}
|
}
|
||||||
|
|
|
@ -293,14 +293,20 @@ class Actionlog extends SnipeModel
|
||||||
public function daysUntilNextAudit($monthInterval = 12, $asset = null)
|
public function daysUntilNextAudit($monthInterval = 12, $asset = null)
|
||||||
{
|
{
|
||||||
$now = Carbon::now();
|
$now = Carbon::now();
|
||||||
$last_audit_date = $this->created_at;
|
$last_audit_date = $this->created_at; // this is the action log's created at, not the asset itself
|
||||||
$next_audit = $last_audit_date->addMonth($monthInterval);
|
$next_audit = $last_audit_date->addMonth($monthInterval); // this actually *modifies* the $last_audit_date
|
||||||
$next_audit_days = $now->diffInDays($next_audit);
|
$next_audit_days = round($now->diffInDays($next_audit, true));
|
||||||
|
$override_default_next = $next_audit;
|
||||||
|
|
||||||
// Override the default setting for interval if the asset has its own next audit date
|
// Override the default setting for interval if the asset has its own next audit date
|
||||||
if (($asset) && ($asset->next_audit_date)) {
|
if (($asset) && ($asset->next_audit_date)) {
|
||||||
$override_default_next = \Carbon::parse($asset->next_audit_date);
|
$override_default_next = Carbon::parse($asset->next_audit_date);
|
||||||
$next_audit_days = $override_default_next->diffInDays($now);
|
$next_audit_days = round($override_default_next->diffInDays($now, true));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show as negative number if the next audit date is before the audit date we're looking at
|
||||||
|
if ($this->created_at > $override_default_next) {
|
||||||
|
$next_audit_days = '-'.$next_audit_days;
|
||||||
}
|
}
|
||||||
|
|
||||||
return $next_audit_days;
|
return $next_audit_days;
|
||||||
|
|
|
@ -413,6 +413,17 @@ class Asset extends Depreciable
|
||||||
return $this->rules;
|
return $this->rules;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public function customFieldsForCheckinCheckout($checkin_checkout) {
|
||||||
|
// Check to see if any of the custom fields were included on the form and if they have any values
|
||||||
|
if (($this->model) && ($this->model->fieldset) && ($this->model->fieldset->fields)) {
|
||||||
|
foreach ($this->model->fieldset->fields as $field) {
|
||||||
|
if (($field->{$checkin_checkout} == 1) && (request()->has($field->db_column))){
|
||||||
|
$this->{$field->db_column} = request()->get($field->db_column);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Establishes the asset -> depreciation relationship
|
* Establishes the asset -> depreciation relationship
|
||||||
|
@ -523,6 +534,18 @@ class Asset extends Depreciable
|
||||||
return $this->morphMany(self::class, 'assigned', 'assigned_type', 'assigned_to')->withTrashed();
|
return $this->morphMany(self::class, 'assigned', 'assigned_type', 'assigned_to')->withTrashed();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Establishes the accessory -> asset assignment relationship
|
||||||
|
*
|
||||||
|
* @author A. Gianotto <snipe@snipe.net>
|
||||||
|
* @since [v3.0]
|
||||||
|
* @return \Illuminate\Database\Eloquent\Relations\Relation
|
||||||
|
*/
|
||||||
|
public function assignedAccessories()
|
||||||
|
{
|
||||||
|
return $this->morphMany(\App\Models\AccessoryCheckout::class, 'assigned', 'assigned_type', 'assigned_to');
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the asset's location based on the assigned user
|
* Get the asset's location based on the assigned user
|
||||||
|
|
|
@ -234,7 +234,7 @@ class AssetModel extends SnipeModel
|
||||||
*/
|
*/
|
||||||
public function adminuser()
|
public function adminuser()
|
||||||
{
|
{
|
||||||
return $this->belongsTo(\App\Models\User::class, 'created_by');
|
return $this->belongsTo(\App\Models\User::class, 'created_by')->withTrashed();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -71,6 +71,7 @@ class Category extends SnipeModel
|
||||||
'require_acceptance',
|
'require_acceptance',
|
||||||
'use_default_eula',
|
'use_default_eula',
|
||||||
'created_by',
|
'created_by',
|
||||||
|
'notes',
|
||||||
];
|
];
|
||||||
|
|
||||||
use Searchable;
|
use Searchable;
|
||||||
|
@ -80,7 +81,7 @@ class Category extends SnipeModel
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
protected $searchableAttributes = ['name', 'category_type'];
|
protected $searchableAttributes = ['name', 'category_type', 'notes'];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The relations and their attributes that should be included when searching the model.
|
* The relations and their attributes that should be included when searching the model.
|
||||||
|
|
|
@ -6,6 +6,7 @@ use App\Models\Traits\Searchable;
|
||||||
use App\Presenters\Presentable;
|
use App\Presenters\Presentable;
|
||||||
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
use Illuminate\Database\Eloquent\Factories\HasFactory;
|
||||||
use Illuminate\Database\Eloquent\SoftDeletes;
|
use Illuminate\Database\Eloquent\SoftDeletes;
|
||||||
|
use Illuminate\Support\Facades\Gate;
|
||||||
use Watson\Validating\ValidatingTrait;
|
use Watson\Validating\ValidatingTrait;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -104,6 +105,13 @@ class Component extends SnipeModel
|
||||||
];
|
];
|
||||||
|
|
||||||
|
|
||||||
|
public function isDeletable()
|
||||||
|
{
|
||||||
|
return Gate::allows('delete', $this)
|
||||||
|
&& ($this->numCheckedOut() === 0)
|
||||||
|
&& ($this->deleted_at == '');
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Establishes the components -> action logs -> uploads relationship
|
* Establishes the components -> action logs -> uploads relationship
|
||||||
*
|
*
|
||||||
|
@ -234,13 +242,24 @@ class Component extends SnipeModel
|
||||||
// In case there are elements checked out to assets that belong to a different company
|
// In case there are elements checked out to assets that belong to a different company
|
||||||
// than this asset and full multiple company support is on we'll remove the global scope,
|
// than this asset and full multiple company support is on we'll remove the global scope,
|
||||||
// so they are included in the count.
|
// so they are included in the count.
|
||||||
foreach ($this->assets()->withoutGlobalScope(new CompanyableScope)->get() as $checkout) {
|
return $this->uncontrainedAssets->sum('pivot.assigned_qty');
|
||||||
$checkedout += $checkout->pivot->assigned_qty;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return $checkedout;
|
|
||||||
|
/**
|
||||||
|
* @return \Illuminate\Database\Eloquent\Relations\BelongsToMany
|
||||||
|
*
|
||||||
|
* This allows us to get the assets with assigned components without the company restriction
|
||||||
|
*/
|
||||||
|
public function uncontrainedAssets() {
|
||||||
|
|
||||||
|
return $this->belongsToMany(\App\Models\Asset::class, 'components_assets')
|
||||||
|
->withPivot('id', 'assigned_qty', 'created_at', 'created_by', 'note')
|
||||||
|
->withoutGlobalScope(new CompanyableScope);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check how many items within a component are remaining
|
* Check how many items within a component are remaining
|
||||||
*
|
*
|
||||||
|
|
|
@ -149,11 +149,6 @@ class CustomField extends Model
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
// This is just a dumb thing we have to include because Laraval/Doctrine doesn't
|
|
||||||
// play well with enums or a table that EVER had enums. :(
|
|
||||||
$platform = Schema::getConnection()->getDoctrineSchemaManager()->getDatabasePlatform();
|
|
||||||
$platform->registerDoctrineTypeMapping('enum', 'string');
|
|
||||||
|
|
||||||
// Rename the field if the name has changed
|
// Rename the field if the name has changed
|
||||||
Schema::table(self::$table_name, function ($table) use ($custom_field) {
|
Schema::table(self::$table_name, function ($table) use ($custom_field) {
|
||||||
$table->renameColumn($custom_field->convertUnicodeDbSlug($custom_field->getOriginal('name')), $custom_field->convertUnicodeDbSlug());
|
$table->renameColumn($custom_field->convertUnicodeDbSlug($custom_field->getOriginal('name')), $custom_field->convertUnicodeDbSlug());
|
||||||
|
|
|
@ -18,7 +18,8 @@ class Group extends SnipeModel
|
||||||
|
|
||||||
protected $fillable = [
|
protected $fillable = [
|
||||||
'name',
|
'name',
|
||||||
'permissions'
|
'permissions',
|
||||||
|
'notes',
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -37,7 +38,7 @@ class Group extends SnipeModel
|
||||||
*
|
*
|
||||||
* @var array
|
* @var array
|
||||||
*/
|
*/
|
||||||
protected $searchableAttributes = ['name', 'created_at'];
|
protected $searchableAttributes = ['name', 'created_at', 'notes'];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The relations and their attributes that should be included when searching the model.
|
* The relations and their attributes that should be included when searching the model.
|
||||||
|
|
|
@ -411,14 +411,14 @@ abstract class Label
|
||||||
/**
|
/**
|
||||||
* Checks the template is internally valid
|
* Checks the template is internally valid
|
||||||
*/
|
*/
|
||||||
public final function validate() {
|
public final function validate() : void {
|
||||||
$this->validateUnits();
|
$this->validateUnits();
|
||||||
$this->validateSize();
|
$this->validateSize();
|
||||||
$this->validateMargins();
|
$this->validateMargins();
|
||||||
$this->validateSupport();
|
$this->validateSupport();
|
||||||
}
|
}
|
||||||
|
|
||||||
private function validateUnits() {
|
private function validateUnits() : void {
|
||||||
$validUnits = [ 'pt', 'mm', 'cm', 'in' ];
|
$validUnits = [ 'pt', 'mm', 'cm', 'in' ];
|
||||||
$unit = $this->getUnit();
|
$unit = $this->getUnit();
|
||||||
if (!in_array(strtolower($unit), $validUnits)) {
|
if (!in_array(strtolower($unit), $validUnits)) {
|
||||||
|
@ -430,7 +430,7 @@ abstract class Label
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function validateSize() {
|
private function validateSize() : void {
|
||||||
$width = $this->getWidth();
|
$width = $this->getWidth();
|
||||||
if (!is_numeric($width) || is_string($width)) {
|
if (!is_numeric($width) || is_string($width)) {
|
||||||
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
|
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
|
||||||
|
@ -450,7 +450,7 @@ abstract class Label
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function validateMargins() {
|
private function validateMargins() : void {
|
||||||
$marginTop = $this->getMarginTop();
|
$marginTop = $this->getMarginTop();
|
||||||
if (!is_numeric($marginTop) || is_string($marginTop)) {
|
if (!is_numeric($marginTop) || is_string($marginTop)) {
|
||||||
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
|
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
|
||||||
|
@ -488,7 +488,7 @@ abstract class Label
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private function validateSupport() {
|
private function validateSupport() : void {
|
||||||
$support1D = $this->getSupport1DBarcode();
|
$support1D = $this->getSupport1DBarcode();
|
||||||
if (!is_bool($support1D)) {
|
if (!is_bool($support1D)) {
|
||||||
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
|
throw new \UnexpectedValueException(trans('admin/labels/message.invalid_return_type', [
|
||||||
|
|
Some files were not shown because too many files have changed in this diff Show more
Loading…
Add table
Reference in a new issue