From 0fb3315c4cd094bdeac45dece0b813b8dbe42e10 Mon Sep 17 00:00:00 2001 From: Eldad Fux Date: Wed, 1 Nov 2023 22:44:19 -0400 Subject: [PATCH 01/26] Upgrade to PHP 8.2 --- Dockerfile | 2 +- app/console | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Dockerfile b/Dockerfile index 33b143465..849be37ea 100755 --- a/Dockerfile +++ b/Dockerfile @@ -29,7 +29,7 @@ ENV VITE_APPWRITE_GROWTH_ENDPOINT=$VITE_APPWRITE_GROWTH_ENDPOINT RUN npm ci RUN npm run build -FROM appwrite/base:0.4.3 as final +FROM appwrite/base:0.5.0 as final LABEL maintainer="team@appwrite.io" diff --git a/app/console b/app/console index 9810ce858..e96573898 160000 --- a/app/console +++ b/app/console @@ -1 +1 @@ -Subproject commit 9810ce85812ca26c95b7d35196848c92e8ba813d +Subproject commit e9657389879c8d76a9b3a0d3486c1d86f43c3bb9 From d91f961c90856ed3e9a0892bffb4a1016d386d75 Mon Sep 17 00:00:00 2001 From: Eldad Fux Date: Wed, 1 Nov 2023 23:09:04 -0400 Subject: [PATCH 02/26] Fixed errors --- composer.json | 8 +++--- composer.lock | 78 +++++++++++++++++++++++++++------------------------ 2 files changed, 45 insertions(+), 41 deletions(-) diff --git a/composer.json b/composer.json index 4ce175abe..a10ddd2e4 100644 --- a/composer.json +++ b/composer.json @@ -69,14 +69,14 @@ "utopia-php/vcs": "0.6.*", "utopia-php/websocket": "0.1.*", "matomo/device-detector": "6.1.*", - "dragonmantank/cron-expression": "3.3.2", + "dragonmantank/cron-expression": "3.3.3", "influxdb/influxdb-php": "1.15.2", - "phpmailer/phpmailer": "6.8.0", + "phpmailer/phpmailer": "6.8.1", "chillerlan/php-qrcode": "4.3.4", "adhocore/jwt": "1.1.2", "webonyx/graphql-php": "14.11.*", - "slickdeals/statsd": "3.1.0", - "league/csv": "9.7.1" + "slickdeals/statsd": "3.1.2", + "league/csv": "9.11.0" }, "repositories": [ { diff --git a/composer.lock b/composer.lock index e8334179f..83f05aa9f 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "9afc62ce9c6ba587b9c028e11494e026", + "content-hash": "2475f162a1e4a3cc1ee0d2f99e0b2de1", "packages": [ { "name": "adhocore/jwt", @@ -341,16 +341,16 @@ }, { "name": "dragonmantank/cron-expression", - "version": "v3.3.2", + "version": "v3.3.3", "source": { "type": "git", "url": "https://github.com/dragonmantank/cron-expression.git", - "reference": "782ca5968ab8b954773518e9e49a6f892a34b2a8" + "reference": "adfb1f505deb6384dc8b39804c5065dd3c8c8c0a" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/dragonmantank/cron-expression/zipball/782ca5968ab8b954773518e9e49a6f892a34b2a8", - "reference": "782ca5968ab8b954773518e9e49a6f892a34b2a8", + "url": "https://api.github.com/repos/dragonmantank/cron-expression/zipball/adfb1f505deb6384dc8b39804c5065dd3c8c8c0a", + "reference": "adfb1f505deb6384dc8b39804c5065dd3c8c8c0a", "shasum": "" }, "require": { @@ -390,7 +390,7 @@ ], "support": { "issues": "https://github.com/dragonmantank/cron-expression/issues", - "source": "https://github.com/dragonmantank/cron-expression/tree/v3.3.2" + "source": "https://github.com/dragonmantank/cron-expression/tree/v3.3.3" }, "funding": [ { @@ -398,7 +398,7 @@ "type": "github" } ], - "time": "2022-09-10T18:51:20+00:00" + "time": "2023-08-10T19:36:49+00:00" }, { "name": "guzzlehttp/guzzle", @@ -853,34 +853,38 @@ }, { "name": "league/csv", - "version": "9.7.1", + "version": "9.11.0", "source": { "type": "git", "url": "https://github.com/thephpleague/csv.git", - "reference": "0ec57e8264ec92565974ead0d1724cf1026e10c1" + "reference": "33149c4bea4949aa4fa3d03fb11ed28682168b39" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/thephpleague/csv/zipball/0ec57e8264ec92565974ead0d1724cf1026e10c1", - "reference": "0ec57e8264ec92565974ead0d1724cf1026e10c1", + "url": "https://api.github.com/repos/thephpleague/csv/zipball/33149c4bea4949aa4fa3d03fb11ed28682168b39", + "reference": "33149c4bea4949aa4fa3d03fb11ed28682168b39", "shasum": "" }, "require": { "ext-json": "*", "ext-mbstring": "*", - "php": "^7.3 || ^8.0" + "php": "^8.1.2" }, "require-dev": { - "ext-curl": "*", + "doctrine/collections": "^2.1.3", "ext-dom": "*", - "friendsofphp/php-cs-fixer": "^2.16", - "phpstan/phpstan": "^0.12.0", - "phpstan/phpstan-phpunit": "^0.12.0", - "phpstan/phpstan-strict-rules": "^0.12.0", - "phpunit/phpunit": "^9.5" + "ext-xdebug": "*", + "friendsofphp/php-cs-fixer": "^v3.22.0", + "phpbench/phpbench": "^1.2.14", + "phpstan/phpstan": "^1.10.26", + "phpstan/phpstan-deprecation-rules": "^1.1.3", + "phpstan/phpstan-phpunit": "^1.3.13", + "phpstan/phpstan-strict-rules": "^1.5.1", + "phpunit/phpunit": "^10.3.1", + "symfony/var-dumper": "^6.3.3" }, "suggest": { - "ext-dom": "Required to use the XMLConverter and or the HTMLConverter classes", + "ext-dom": "Required to use the XMLConverter and the HTMLConverter classes", "ext-iconv": "Needed to ease transcoding CSV using iconv stream filters" }, "type": "library", @@ -910,7 +914,7 @@ } ], "description": "CSV data manipulation made easy in PHP", - "homepage": "http://csv.thephpleague.com", + "homepage": "https://csv.thephpleague.com", "keywords": [ "convert", "csv", @@ -933,7 +937,7 @@ "type": "github" } ], - "time": "2021-04-17T16:32:08+00:00" + "time": "2023-09-23T10:09:54+00:00" }, { "name": "matomo/device-detector", @@ -1125,16 +1129,16 @@ }, { "name": "phpmailer/phpmailer", - "version": "v6.8.0", + "version": "v6.8.1", "source": { "type": "git", "url": "https://github.com/PHPMailer/PHPMailer.git", - "reference": "df16b615e371d81fb79e506277faea67a1be18f1" + "reference": "e88da8d679acc3824ff231fdc553565b802ac016" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/df16b615e371d81fb79e506277faea67a1be18f1", - "reference": "df16b615e371d81fb79e506277faea67a1be18f1", + "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/e88da8d679acc3824ff231fdc553565b802ac016", + "reference": "e88da8d679acc3824ff231fdc553565b802ac016", "shasum": "" }, "require": { @@ -1144,13 +1148,13 @@ "php": ">=5.5.0" }, "require-dev": { - "dealerdirect/phpcodesniffer-composer-installer": "^0.7.2", + "dealerdirect/phpcodesniffer-composer-installer": "^1.0", "doctrine/annotations": "^1.2.6 || ^1.13.3", "php-parallel-lint/php-console-highlighter": "^1.0.0", "php-parallel-lint/php-parallel-lint": "^1.3.2", "phpcompatibility/php-compatibility": "^9.3.5", "roave/security-advisories": "dev-latest", - "squizlabs/php_codesniffer": "^3.7.1", + "squizlabs/php_codesniffer": "^3.7.2", "yoast/phpunit-polyfills": "^1.0.4" }, "suggest": { @@ -1193,7 +1197,7 @@ "description": "PHPMailer is a full-featured email creation and transfer class for PHP", "support": { "issues": "https://github.com/PHPMailer/PHPMailer/issues", - "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.8.0" + "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.8.1" }, "funding": [ { @@ -1201,7 +1205,7 @@ "type": "github" } ], - "time": "2023-03-06T14:43:22+00:00" + "time": "2023-08-29T08:26:30+00:00" }, { "name": "psr/http-client", @@ -1409,23 +1413,23 @@ }, { "name": "slickdeals/statsd", - "version": "3.1.0", + "version": "3.1.2", "source": { "type": "git", "url": "https://github.com/Slickdeals/statsd-php.git", - "reference": "225588a0a079e145359049f6e5e23eedb1b4c17f" + "reference": "03fc5d8d51aff9f651f97101033aad31c08d00de" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/Slickdeals/statsd-php/zipball/225588a0a079e145359049f6e5e23eedb1b4c17f", - "reference": "225588a0a079e145359049f6e5e23eedb1b4c17f", + "url": "https://api.github.com/repos/Slickdeals/statsd-php/zipball/03fc5d8d51aff9f651f97101033aad31c08d00de", + "reference": "03fc5d8d51aff9f651f97101033aad31c08d00de", "shasum": "" }, "require": { "php": ">= 7.3 || ^8" }, "replace": { - "domnikl/statsd": "self.version" + "domnikl/statsd": "<=3.0.2" }, "require-dev": { "friendsofphp/php-cs-fixer": "^3.0", @@ -1459,9 +1463,9 @@ ], "support": { "issues": "https://github.com/Slickdeals/statsd-php/issues", - "source": "https://github.com/Slickdeals/statsd-php/tree/3.1.0" + "source": "https://github.com/Slickdeals/statsd-php/tree/3.1.2" }, - "time": "2021-06-04T20:33:46+00:00" + "time": "2022-08-15T16:22:48+00:00" }, { "name": "symfony/deprecation-contracts", @@ -5822,5 +5826,5 @@ "platform-overrides": { "php": "8.0" }, - "plugin-api-version": "2.3.0" + "plugin-api-version": "2.6.0" } From 8dcf836ec97b9a1d137d34c990e1617872960c00 Mon Sep 17 00:00:00 2001 From: Eldad Fux Date: Thu, 2 Nov 2023 00:16:30 -0400 Subject: [PATCH 03/26] Updated resources --- tests/resources/initials.png | Bin 3075 -> 3059 bytes tests/resources/qr/qr-default.png | Bin 14593 -> 29105 bytes tests/resources/qr/qr-size-200-margin-10.png | Bin 3750 -> 5496 bytes tests/resources/qr/qr-size-200.png | Bin 6075 -> 10890 bytes 4 files changed, 0 insertions(+), 0 deletions(-) diff --git a/tests/resources/initials.png b/tests/resources/initials.png index ea6bf729b50f2b71b746690522a2cb0f2d1f0d17..3e51f228cc6092ca37b101ffa7a900dda52a0f50 100644 GIT binary patch delta 10 RcmZpc_$)p_WuwV?ZU7fn1Q`GT delta 24 fcmew?-YhXeg_kAW(btiIVPik{pG4M;ikG+nVQB~b diff --git a/tests/resources/qr/qr-default.png b/tests/resources/qr/qr-default.png index a7da496d9fd62ecb6b0afc8d1f2bfb154a660cbf..58af53bb9327542558a7889c4e0874c9e614bf79 100644 GIT binary patch literal 29105 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 Y2JX$sTS8|^237#l;tFClqDDdg56b>FCIA2c literal 14593 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 delta 3420 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 Y{Xa@22OtL6Rn+wZey)&{uktJW2h%)g!~g&Q diff --git a/tests/resources/qr/qr-size-200.png b/tests/resources/qr/qr-size-200.png index 202c0cc1970196532f0fde06446de707ad7ab718..a44e4861e33de3cebfd107c7d0ce39403e71e269 100644 GIT binary patch delta 10650 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 T51yTuN6d%U3-}p>%k+N$(fcmC delta 5746 zcmZ{oRaDdsw8j6!P|_eMEezeQ(gO_LEg{lK*GI>UFd!l|Am9+vEg%ftD$?B{Atf<{ z(tTZb-Pikg&ROd`oQJdbZ^s(93MH|V;yNj4D*!-EBJqt49*R{7z6m12V_^>%+FJqu zntpYdf>GeY{>zX++{gYNW3(0?CP+F|9C`ue|M1b*~hyj~g}F z|IfT$PUY4dO_yC06X|c{Qkm%O=6Z_Bmg0>x&qYL>f`}qmaMcLee`bb!3b~uR?waSi z{uJWj6RFO*m-rK&6?W#jk2za#t*$>?K-vZ({sskDv?HF!4Lh>#lY}%GhH1 z6c%mmux(fj;q7UK3tHl0rf3oX20!^xcQJ(gD4dA$7qBciR~c#Y!UwERQ!Z;mymw6{ zIX@U^l)<$x?QG-*2kx@MVjpNaJJ0>s_L+0@H+CQ6cnNQB>Tz;V3g!-A2ew1Gsn3ih zl8ci#wKv*Xf-`twgwo52{mcejY8##lqP8w;i~k9`HO&x0!Xo49WET(N!!`Z0UWI1K z>3sVYSrase=BbBE&UTsv*G>apQAYA?q8@hUN(d?_HefM@LefXUuwTc~s@X@?88%~e zuI}lm0l)D$8#VRp=>QcJYV=85rA-nSl|Nh9YD+`L%p1LINj!B9tjD;Iep>Ua=wYWe z-cya|U4puOw9UyWW}$C#W`DOK!Rl?G_iPyYLrUaZJwAgG8H!e*C_j-X!srY%tl5gv zrrN#HX(a$e=>Wv?m^)6ctlJEo84jSZ{BU3cjsvimDVeZU2$8jL(AaOpY#hQ10;b>i zT;e^5savr+1XHgx5dvOEom(`};ZX}*;A2qbCOCNUCwacZ%u4G#a5FSr((e+w^C$P& z*XJ0Iq*`%F$N>X9iDOi&3w-$l8z6N}=Zg;@I=Yy-!r}qwb%7!}Tc{ocj3m!AZNUR3 zHe8F!l(%9|3?F}$5-}uqs*_`TXkl4PqY_=5kzc$~74KE4KyXcp3df@5xL_M6Rl*Zw zjlD7{Jxy(f26F$2f|hW2Pvy4N-b3*?Lzbr6@F_|v3wwaE41OX&8vT@`v!@=ER3i}y zLK%{g5n8)lT%T+FX7W8JVPLWU8VMl$Or3DLMwfk-t)ccLlQC1qMSXBrty>(x8WJ*e zyrfQaYQR(a(_W|$CB~hjyM{l7jTEq6rMS$pX<~zLrNDkzqB367m6eIoHeNb6#^jnN1v6$jN!Qb?%H$m|K1C;DiI= zTG0$CSYyf%08@&=)ej%Oo%VeQli$zs1%S~T6R16>VPD&=Efu8qN#cp#n|_@MVO_~z zm#N3v<+!sNN=LUJkFUv53ZU;alvP)n_!#nXrc_t@NvxYX{VzEWlX&mSGt*to4De9V z1gV;%i#q`9JyS7^ByZ#e>h?<(c~_?^y)A*)cNIu{Y!RHh`Nz04%;Gg8%Gk7=gNvmm zU@)2CP%;moxH$>WGd%tz3VQxq-e42dQ~ZQq0;J)2^%jZmzhb;lr*-p!%QohD*yGar z;_(6y^e5YR!b<@032Y2_1=RGuRF}Y0f3GR>@1W%Sr8U#2#1G$MeoP~$h`xDknkzk4 zgk9zG_`;j#D23%7V>%wH$Of}a>B{~YPn|Y;gk&OrH!XT9YD&H|9A3)G9R#sV)M zO0*10PLc|6b9#urRVr!@7}9t`$-uJnVl-cV6-LE0n|-1>jK*Y2_%l$=Vram zLPB&bg7FP2$#*hJl!iNsN{goj?~>&x!%dE-$(~tlS}adaMex`sS(K{COm`@*<{<>R zJAJS~kAAfkJU)bWp5k6L4sd@bOj!O*)TYfQUlH-m-UQ>*75ES#wp*az9t2l%1S_R@ zEUf{#1@y?8ow;0GsAJ^zdp8N#3ELFI;QhL-jGX)X)k;4hOFBAsj!wZRl`dDZ+GM#~ zIWHdd+dsQUHur=!-~kaCkHA0#={_#vZYCva`S|6Il@l0n_9z6f7!M3!y~r_S(dG;R zxI<H2`vUxqgo|0YMWXrS2gu0vXPK@g%~lwQ?hD2i=?DYI->Ln;aIPy6n-= zj4|~0k>M-E!s$&9QtS6!zDb6lXiYj=qvR?;akr(wfur)CO4L0|uuaw`B~IN#vSvjI z?%_m#rdhc0L`_}bOvKLD9zN;TCJWiO%g7gjZCrKSzU@nj`W34t=|5smStP7yRLI%l zHAb3432ZICi@k+!f!W%RG46wq=Hd94*c|nTpxm(mJ}$r%qIcz4{lxsSnOy1t zIo?h{?_R`-bRL>8My3M_h7v8Q3s?rb!+7%imi#PW18mDWXQRr`^oaCQBncPfLl$tY zU&%a+pX$p!YcOPcIr{mcRX5+$bSPnfX*p{V4>r~zpbRHI)gCWtku@0H`S@p?t0f+d zYD`sd-UXeG_r6=ZtNEfFNK!fVi$K>QJ8u7NDmD7sOE*dUthql%T;3$#639)fB>rQC zvFi`agqm^6#*f;3Ec5B5iakZ$JBnumN6HZ=j6mN;Ej#fAuam(WZP4~oOE(@DVWQKq z*2|1!kRH4*miR-JswfD5PlZ<~B(>~qDO&=mD$-I$8)VI&BXqQ)O#Bj$u-s{71N?#U zyMcVxY#!*%cyRdPq1xB{j@9)X_eoGW(`hPzSb-e$j&@K`3(JLC69C)kIR`jnE0#&$ z+QS!&C75&lMqV+wnjuUX0H9vzOK%YW=*Brx3tl}{tDow+F3ImG@gpuTvzZo7U|J`} z#d`eU?84#^jrGq8MRWjeCU|5o?}Nd@FpcbNFHW0ROpR2$K7zIC{N@X+HamsgF}Z@x z@RHbzNe3IZbF^-*#2n2B$=k|2ru|m>R3#C#o*dP!FqU1Ym1WlBPulU}#!uh&m(0rQ zB%HrEzJyQS8k>GPereYThRz2$LxF&N+8etYj=0+@c*eYBqA9>~-gDN6uk0@WtQs&{ z91^V`qVN|wl=(4NP3v8FnJGzgwA(y)`E~IF;PN}Yg$8lKCx-*LT{*-Pa7VGx8v%E5 zUxe+f?#}B!JD5Fs0&JM#4fFD zDSfEtADkcy73tgjxAD6)O%)bTh_^nPz%zm#ZEI#@@!sWG9~dy@C6J_-kfVmHls!Vw zCpuAGw(xD%)ePr+-`%4tdI69KLVY>>8x=mk-lA&gW*<3-^&j~-Y;fv;9>7`VN+qq~ z8)0;}XQJC+c@_;3NsXinhKSm*_h>xGw4~E9Xt~k3=pB>(tKs zG5qik2v9Y;GdDFZ0Cf$ACZEDVz6IQokDq=6)p-bN2UJ&|0d{uzfs2!L6o_)Wxv?qc zZ9asof5j#cYnKp@O<(d(d-3$d*?u>dcI=NKa_X`(hVkGG^O$)LXDm(TuoT!aW8){9 ze(T5$ZS&Jo8lTJ8M^Ml7O^I(QnF4@SXNd(k%R7LX7`Sj_!^?$V!C{$XxC=oT$Zy9| zZ#DPN7>BO&?!-&iv=)lnC@-dNRk zNqr8NHOmvnHq9ixt)-y6{AzVP3bkM~yXE*?c6$K{`9o_xYh?w0_e;Yhwd$t5k=#&m z#r9s z^ni3vNYwgXe*VxpnyQdviFF_`(Vz^!MYTgBZrYVlzMo48o)Vvz&$qgq-b`mjJ*hp! z;V_&0n$UcqZwQ69`w02ejFVsfqY#m^6WEIfWypwP%?xEUK5NFAsIOoIwyw2;eN9YN z3vYAFm1&z28VjosmT}|Wa8`$P*tpZ@~kCC7rA; z?UXMdi}ER9LaoYy8&3x!XsK=ZDoocGxQnAy&vNkrLLV&Hof2*CE&$}B`zPyOJ`da@ z?of*fLT@SG=kNa#K7Msc^dyt*Z?@;qNWYjqT%U>OS@eSQoY&v8z;0Tp7p(KN^z3pz zxdyc-Fuu~Cv>Ch|7NW>7B550Ls;bRyBbjJk=Xnz!N>A@7MMFYs-#xDWE?jG00io0U zNSo+4DdIaW>s-zSF~vZxRp|t2eBNVbLDpK9w7{(I2HtEa~WMoAbB z{r!@&E+Z#c$=&ARV6Q)_%Fk^BEH$^r7wv!j=byBvh(Ql(jWT;O^~A07RV>Hh*&~V| zXh~muWY*E^DYR(#N{&TJ_hsrXSsZPkEnO6;x~Mz#gX#YxOUk9a0Gccm*LCls2bBz& zc|F@kHf6ZX%ah6vtlg?BBtS$hkp>9*0%IWer7SKmJVOB-1xVupmB}hz^_>2@2r&h0 zxk-gSVoY%B?--`+D;~U5%KW51hE;SwP5jNb11j46{A=QWh%nkQiAQL4&7cTe&n?n} zy3psfpf1V1Yfb=r)0@f(jr1H&o0eG@Ne8B~cnftp>|#kvN5I&T7@CzCp4FKZ2+_o! zy&N4~?_pi+r<~z?QsAGk`3~MZ@z*7A`h6$uS=;D#`{lk4jsvX;5u;~tD3jpAG?8}N zBr;)5;`Q^5R`w_ih@Oo$9Vl&*f%7f1DP=nqIUuthsp~c)*Bi0E!r>HtNLSLQr7RHG z#T#tdoI^&xEwiaz40?^mc+8gO%Kbs2%V$todRR?lgQB*GfUXQlx#a}rl{p(}N+$Jz z->ltxO9TS$&ExjT^biy)5G`2U;Z{thm&NYCTSV6HQhqe}v*A7@g@F5D5nHqAHb`)j z!Sm-pUxVkg+DZ47N69}q4yyWZ=y&ZQq6-GbIA>wH(OVxI+0kQAP}BNyRRXv0>vd!-tIxm` z;WNYS+hY*$_HL1#VeITD#By%Nd(Q#|s!wdPvxQBGzq+s?16&TDO9qRcWbZx2L0n}I zGQxqLwTghlJ@M1KGskh-1@cl#@cLic=QEE#iqx3|?o>V`DYGkPqzKFEz7|)GX-u=* zqect=BcxA({{^@{kGNH3#}~0HeO(@ViPNoP)TB60Vf#Bh)H@-LR*8Vmh#56Do;4Z5 zO(I4WvjehA2e>Du<`yqLrQMsz6=RAi@Zeq&1cJG5eY=Hd1#^4X$Z-9qUKIJtQuw_n zZ^FwJlp)%-0j1I@DV@Cxf{i#sa9Nh_Z689t(X}8xj|nhzvE8b#_0L&L%uQ8Tw`0&O zxbqKy>qKN~jH}!ESP!~h(^2lI>kFdSVgpC1@fl9?XSd2`T#c*Nwn>L`Xzu^ggzMJF z^sW6E?;Q#%A&SCxtG&MMj2G0e(Xq3r0V&ycoeQ!B=#u+zN5tItAN!o_2WJpZq_C3w z`KWSjhIHM*u(Z}*>N#&|SKKa#Snp#dTZ|Ba6(3!fk zz|f9e7y|z$2EK3llooiftR33|H^!yh!{2G|^SK|nZx>lWx(cbb`-4_g$-$wQhs3Hr;`w#bz$>`oL7^V;7 z2seMuaGnBk73>Gh!#H^dXO6b5>LZ#B9Tt(h8+y-ZK4N3V+wyxjjRN5gg5+F`n2$Yz z-seSSj6XM^%-^$JNo8Ytb2T4EiKQmuo)3u*((y!%=lrBCe@~_P^O*1H{QlDq8~123 zo_gMCEaK&-dH!toG}6~gvPI=!s4pH)7nZK^pU>cw%aFZPvy`r9X{uz|?xO+5u-rV5 z!x5ga(ZmVCPPgqeHg`BQj~ESWJ(J=CV$#vO``ZmzHuC3Rrbot9u4|r~{yQ&dxC?aIqx+KHQdUXK z{mWRt;(x~{M%{xSbzW!JqF5AC4!9rFiwFpd2#5$92}?+ciAsq* Date: Thu, 2 Nov 2023 07:09:31 -0400 Subject: [PATCH 04/26] Fixed errors --- tests/e2e/Services/GraphQL/AvatarsTest.php | 12 ++++++------ 1 file changed, 6 insertions(+), 6 deletions(-) diff --git a/tests/e2e/Services/GraphQL/AvatarsTest.php b/tests/e2e/Services/GraphQL/AvatarsTest.php index b95e3b251..9f3fa7a3b 100644 --- a/tests/e2e/Services/GraphQL/AvatarsTest.php +++ b/tests/e2e/Services/GraphQL/AvatarsTest.php @@ -29,7 +29,7 @@ class AvatarsTest extends Scope 'x-appwrite-project' => $projectId, ], $this->getHeaders()), $graphQLPayload); - $this->assertEquals(18767, \strlen($creditCardIcon['body'])); + $this->assertEquals(18546, \strlen($creditCardIcon['body'])); return $creditCardIcon['body']; } @@ -50,7 +50,7 @@ class AvatarsTest extends Scope 'x-appwrite-project' => $projectId, ], $this->getHeaders()), $graphQLPayload); - $this->assertEquals(11100, \strlen($browserIcon['body'])); + $this->assertEquals(13312, \strlen($browserIcon['body'])); return $browserIcon['body']; } @@ -71,7 +71,7 @@ class AvatarsTest extends Scope 'x-appwrite-project' => $projectId, ], $this->getHeaders()), $graphQLPayload); - $this->assertEquals(7460, \strlen($countryFlag['body'])); + $this->assertEquals(8814, \strlen($countryFlag['body'])); return $countryFlag['body']; } @@ -92,7 +92,7 @@ class AvatarsTest extends Scope 'x-appwrite-project' => $projectId, ], $this->getHeaders()), $graphQLPayload); - $this->assertEquals(36036, \strlen($image['body'])); + $this->assertEquals(52585, \strlen($image['body'])); return $image['body']; } @@ -134,7 +134,7 @@ class AvatarsTest extends Scope 'x-appwrite-project' => $projectId, ], $this->getHeaders()), $graphQLPayload); - $this->assertEquals(14771, \strlen($qrCode['body'])); + $this->assertEquals(29428, \strlen($qrCode['body'])); return $qrCode['body']; } @@ -155,7 +155,7 @@ class AvatarsTest extends Scope 'x-appwrite-project' => $projectId, ], $this->getHeaders()), $graphQLPayload); - $this->assertEquals(5041, \strlen($initials['body'])); + $this->assertEquals(5025, \strlen($initials['body'])); return $initials['body']; } From 77caf64076c6f56bc18bd173a538cd6775cd2000 Mon Sep 17 00:00:00 2001 From: Bradley Schofield Date: Mon, 6 Nov 2023 10:08:11 +0000 Subject: [PATCH 05/26] Add XDebug to Appwrite --- CONTRIBUTING.md | 8 ++++---- Dockerfile | 4 ++-- dev/xdebug.ini | 6 ++++++ dev/yasd_init.php.stub | 4 ---- 4 files changed, 12 insertions(+), 10 deletions(-) create mode 100644 dev/xdebug.ini delete mode 100644 dev/yasd_init.php.stub diff --git a/CONTRIBUTING.md b/CONTRIBUTING.md index ccd61e742..7aa97050d 100644 --- a/CONTRIBUTING.md +++ b/CONTRIBUTING.md @@ -345,13 +345,13 @@ Things to remember when releasing SDKs: ## Debug -Appwrite uses [yasd](https://github.com/swoole/yasd) debugger, which can be made available during build of Appwrite. You can connect to the debugger using VS Code's [PHP Debug](https://marketplace.visualstudio.com/items?itemName=felixfbecker.php-debug) extension. +Appwrite uses [XDebug](https://github.com/xdebug/xdebug) debugger, which can be made available during build of Appwrite. You can connect to the debugger using VS Code's [PHP Debug](https://marketplace.visualstudio.com/items?itemName=felixfbecker.php-debug) extension. If you are in PHP Storm you don't need any plugin. Below are the settings required for remote debugger connection: -1. Create an init file. -2. Duplicate **dev/yasd_init.php.stub** file and name it **dev/yasd_init.php**. -3. Set **DEBUG** build arg in **appwrite** service in **docker-compose.yml** file. +1. Set **DEBUG** build arg in **appwrite** service in **docker-compose.yml** file. +2. If needed edit the **dev/xdebug.ini** file to your needs. +3. Launch your Appwrite instance while your debugger is listening for connections. ### VS Code Launch Configuration diff --git a/Dockerfile b/Dockerfile index 849be37ea..7826c8621 100755 --- a/Dockerfile +++ b/Dockerfile @@ -56,6 +56,7 @@ COPY ./public /usr/src/code/public COPY ./bin /usr/local/bin COPY ./docs /usr/src/code/docs COPY ./src /usr/src/code/src +COPY ./dev /usr/src/code/dev # Set Volumes RUN mkdir -p /storage/uploads && \ @@ -110,8 +111,7 @@ RUN chmod +x /usr/local/bin/hamster && \ RUN mkdir -p /etc/letsencrypt/live/ && chmod -Rf 755 /etc/letsencrypt/live/ # Enable Extensions -RUN if [ "$DEBUG" == "true" ]; then printf "zend_extension=yasd \nyasd.debug_mode=remote \nyasd.init_file=/usr/src/code/dev/yasd_init.php \nyasd.remote_port=9005 \nyasd.log_level=-1" >> /usr/local/etc/php/conf.d/yasd.ini; fi - +RUN if [ "$DEBUG" == "true" ]; then cp /usr/src/code/dev/xdebug.ini /usr/local/etc/php/conf.d/xdebug.ini; fi RUN if [ "$DEBUG" == "true" ]; then echo "opcache.enable=0" >> /usr/local/etc/php/conf.d/appwrite.ini; fi RUN echo "opcache.preload_user=www-data" >> /usr/local/etc/php/conf.d/appwrite.ini RUN echo "opcache.preload=/usr/src/code/app/preload.php" >> /usr/local/etc/php/conf.d/appwrite.ini diff --git a/dev/xdebug.ini b/dev/xdebug.ini new file mode 100644 index 000000000..e29c8bd46 --- /dev/null +++ b/dev/xdebug.ini @@ -0,0 +1,6 @@ +zend_extension=xdebug + +[xdebug] +xdebug.mode=develop,debug +xdebug.client_host=host.docker.internal +xdebug.start_with_request=yes \ No newline at end of file diff --git a/dev/yasd_init.php.stub b/dev/yasd_init.php.stub deleted file mode 100644 index efcc37c0b..000000000 --- a/dev/yasd_init.php.stub +++ /dev/null @@ -1,4 +0,0 @@ - Date: Wed, 8 Nov 2023 11:24:58 +0000 Subject: [PATCH 06/26] Update base and remove XDebug if non-debug --- Dockerfile | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index 7826c8621..52bbd4e63 100755 --- a/Dockerfile +++ b/Dockerfile @@ -29,7 +29,7 @@ ENV VITE_APPWRITE_GROWTH_ENDPOINT=$VITE_APPWRITE_GROWTH_ENDPOINT RUN npm ci RUN npm run build -FROM appwrite/base:0.5.0 as final +FROM appwrite/base:0.6.0 as final LABEL maintainer="team@appwrite.io" @@ -113,6 +113,8 @@ RUN mkdir -p /etc/letsencrypt/live/ && chmod -Rf 755 /etc/letsencrypt/live/ # Enable Extensions RUN if [ "$DEBUG" == "true" ]; then cp /usr/src/code/dev/xdebug.ini /usr/local/etc/php/conf.d/xdebug.ini; fi RUN if [ "$DEBUG" == "true" ]; then echo "opcache.enable=0" >> /usr/local/etc/php/conf.d/appwrite.ini; fi +RUN if [ "$DEBUG" = "false" ]; then rm -rf /usr/src/code/dev; fi +RUN if [ "$DEBUG" = "false" ]; then rm -f /usr/local/lib/php/extensions/no-debug-non-zts-20220829/xdebug.so RUN echo "opcache.preload_user=www-data" >> /usr/local/etc/php/conf.d/appwrite.ini RUN echo "opcache.preload=/usr/src/code/app/preload.php" >> /usr/local/etc/php/conf.d/appwrite.ini RUN echo "opcache.enable_cli=1" >> /usr/local/etc/php/conf.d/appwrite.ini From d885b6218d3bf94c9de0572f51e16a1d4853a3a7 Mon Sep 17 00:00:00 2001 From: Bradley Schofield Date: Wed, 8 Nov 2023 11:26:27 +0000 Subject: [PATCH 07/26] Update Dockerfile --- Dockerfile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index 52bbd4e63..e70eed415 100755 --- a/Dockerfile +++ b/Dockerfile @@ -114,7 +114,7 @@ RUN mkdir -p /etc/letsencrypt/live/ && chmod -Rf 755 /etc/letsencrypt/live/ RUN if [ "$DEBUG" == "true" ]; then cp /usr/src/code/dev/xdebug.ini /usr/local/etc/php/conf.d/xdebug.ini; fi RUN if [ "$DEBUG" == "true" ]; then echo "opcache.enable=0" >> /usr/local/etc/php/conf.d/appwrite.ini; fi RUN if [ "$DEBUG" = "false" ]; then rm -rf /usr/src/code/dev; fi -RUN if [ "$DEBUG" = "false" ]; then rm -f /usr/local/lib/php/extensions/no-debug-non-zts-20220829/xdebug.so +RUN if [ "$DEBUG" = "false" ]; then rm -f /usr/local/lib/php/extensions/no-debug-non-zts-20220829/xdebug.so; fi RUN echo "opcache.preload_user=www-data" >> /usr/local/etc/php/conf.d/appwrite.ini RUN echo "opcache.preload=/usr/src/code/app/preload.php" >> /usr/local/etc/php/conf.d/appwrite.ini RUN echo "opcache.enable_cli=1" >> /usr/local/etc/php/conf.d/appwrite.ini From 3322711d9e5220118c5ba048f6661698ca5663cc Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Tue, 9 Jan 2024 18:16:30 +1300 Subject: [PATCH 08/26] Fix lock --- composer.lock | 58 +++++++++++++++++++++++++++------------------------ 1 file changed, 31 insertions(+), 27 deletions(-) diff --git a/composer.lock b/composer.lock index eb9b32a31..87a52c233 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "359b1e3bd27ac7362c6f8d145e64ae36", + "content-hash": "0ccb571d2088720a1babfb614a09e99f", "packages": [ { "name": "adhocore/jwt", @@ -277,16 +277,16 @@ }, { "name": "chillerlan/php-settings-container", - "version": "2.1.4", + "version": "2.1.5", "source": { "type": "git", "url": "https://github.com/chillerlan/php-settings-container.git", - "reference": "1beb7df3c14346d4344b0b2e12f6f9a74feabd4a" + "reference": "f705310389264c3578fdd9ffb15aa2cd6d91772e" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/chillerlan/php-settings-container/zipball/1beb7df3c14346d4344b0b2e12f6f9a74feabd4a", - "reference": "1beb7df3c14346d4344b0b2e12f6f9a74feabd4a", + "url": "https://api.github.com/repos/chillerlan/php-settings-container/zipball/f705310389264c3578fdd9ffb15aa2cd6d91772e", + "reference": "f705310389264c3578fdd9ffb15aa2cd6d91772e", "shasum": "" }, "require": { @@ -294,8 +294,10 @@ "php": "^7.4 || ^8.0" }, "require-dev": { - "phan/phan": "^5.3", - "phpunit/phpunit": "^9.5" + "phan/phan": "^5.4", + "phpcsstandards/php_codesniffer": "^3.8", + "phpmd/phpmd": "^2.13", + "phpunit/phpunit": "^9.6" }, "type": "library", "autoload": { @@ -337,7 +339,7 @@ "type": "ko_fi" } ], - "time": "2022-07-05T22:32:14+00:00" + "time": "2024-01-05T23:20:55+00:00" }, { "name": "dragonmantank/cron-expression", @@ -2076,12 +2078,12 @@ "version": "0.31.1", "source": { "type": "git", - "url": "https://github.com/utopia-php/framework.git", + "url": "https://github.com/utopia-php/http.git", "reference": "e50d2d16f4bc31319043f3f6d3dbea36c6fd6b68" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/utopia-php/framework/zipball/e50d2d16f4bc31319043f3f6d3dbea36c6fd6b68", + "url": "https://api.github.com/repos/utopia-php/http/zipball/e50d2d16f4bc31319043f3f6d3dbea36c6fd6b68", "reference": "e50d2d16f4bc31319043f3f6d3dbea36c6fd6b68", "shasum": "" }, @@ -2111,8 +2113,8 @@ "upf" ], "support": { - "issues": "https://github.com/utopia-php/framework/issues", - "source": "https://github.com/utopia-php/framework/tree/0.31.1" + "issues": "https://github.com/utopia-php/http/issues", + "source": "https://github.com/utopia-php/http/tree/0.31.1" }, "time": "2023-12-08T18:47:29+00:00" }, @@ -3493,25 +3495,27 @@ }, { "name": "nikic/php-parser", - "version": "v4.18.0", + "version": "v5.0.0", "source": { "type": "git", "url": "https://github.com/nikic/PHP-Parser.git", - "reference": "1bcbb2179f97633e98bbbc87044ee2611c7d7999" + "reference": "4a21235f7e56e713259a6f76bf4b5ea08502b9dc" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/1bcbb2179f97633e98bbbc87044ee2611c7d7999", - "reference": "1bcbb2179f97633e98bbbc87044ee2611c7d7999", + "url": "https://api.github.com/repos/nikic/PHP-Parser/zipball/4a21235f7e56e713259a6f76bf4b5ea08502b9dc", + "reference": "4a21235f7e56e713259a6f76bf4b5ea08502b9dc", "shasum": "" }, "require": { + "ext-ctype": "*", + "ext-json": "*", "ext-tokenizer": "*", - "php": ">=7.0" + "php": ">=7.4" }, "require-dev": { "ircmaxell/php-yacc": "^0.0.7", - "phpunit/phpunit": "^6.5 || ^7.0 || ^8.0 || ^9.0" + "phpunit/phpunit": "^7.0 || ^8.0 || ^9.0" }, "bin": [ "bin/php-parse" @@ -3519,7 +3523,7 @@ "type": "library", "extra": { "branch-alias": { - "dev-master": "4.9-dev" + "dev-master": "5.0-dev" } }, "autoload": { @@ -3543,9 +3547,9 @@ ], "support": { "issues": "https://github.com/nikic/PHP-Parser/issues", - "source": "https://github.com/nikic/PHP-Parser/tree/v4.18.0" + "source": "https://github.com/nikic/PHP-Parser/tree/v5.0.0" }, - "time": "2023-12-10T21:03:43+00:00" + "time": "2024-01-07T17:17:35+00:00" }, { "name": "phar-io/manifest", @@ -3897,16 +3901,16 @@ }, { "name": "phpstan/phpdoc-parser", - "version": "1.24.5", + "version": "1.25.0", "source": { "type": "git", "url": "https://github.com/phpstan/phpdoc-parser.git", - "reference": "fedf211ff14ec8381c9bf5714e33a7a552dd1acc" + "reference": "bd84b629c8de41aa2ae82c067c955e06f1b00240" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/phpstan/phpdoc-parser/zipball/fedf211ff14ec8381c9bf5714e33a7a552dd1acc", - "reference": "fedf211ff14ec8381c9bf5714e33a7a552dd1acc", + "url": "https://api.github.com/repos/phpstan/phpdoc-parser/zipball/bd84b629c8de41aa2ae82c067c955e06f1b00240", + "reference": "bd84b629c8de41aa2ae82c067c955e06f1b00240", "shasum": "" }, "require": { @@ -3938,9 +3942,9 @@ "description": "PHPDoc parser with support for nullable, intersection and generic types", "support": { "issues": "https://github.com/phpstan/phpdoc-parser/issues", - "source": "https://github.com/phpstan/phpdoc-parser/tree/1.24.5" + "source": "https://github.com/phpstan/phpdoc-parser/tree/1.25.0" }, - "time": "2023-12-16T09:33:33+00:00" + "time": "2024-01-04T17:06:16+00:00" }, { "name": "phpunit/php-code-coverage", From f925d4e55613a43f6637dc09e8bf489e271688e4 Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Tue, 9 Jan 2024 18:19:08 +1300 Subject: [PATCH 09/26] Update docker base --- Dockerfile | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Dockerfile b/Dockerfile index 09f3e0f8a..5c7522f1b 100755 --- a/Dockerfile +++ b/Dockerfile @@ -29,7 +29,7 @@ ENV VITE_APPWRITE_GROWTH_ENDPOINT=$VITE_APPWRITE_GROWTH_ENDPOINT RUN npm ci RUN npm run build -FROM appwrite/base:0.6.0 as final +FROM appwrite/base:0.7.0 as final LABEL maintainer="team@appwrite.io" From 5039592128b8ac49be677c84db0b5c8444212e49 Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Wed, 10 Jan 2024 16:39:14 +1300 Subject: [PATCH 10/26] Bump base --- Dockerfile | 2 +- tests/e2e/Services/GraphQL/ScopeTest.php | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Dockerfile b/Dockerfile index 5c7522f1b..4373616f5 100755 --- a/Dockerfile +++ b/Dockerfile @@ -29,7 +29,7 @@ ENV VITE_APPWRITE_GROWTH_ENDPOINT=$VITE_APPWRITE_GROWTH_ENDPOINT RUN npm ci RUN npm run build -FROM appwrite/base:0.7.0 as final +FROM appwrite/base:0.7.1 as final LABEL maintainer="team@appwrite.io" diff --git a/tests/e2e/Services/GraphQL/ScopeTest.php b/tests/e2e/Services/GraphQL/ScopeTest.php index a8b5b7cea..ebf535268 100644 --- a/tests/e2e/Services/GraphQL/ScopeTest.php +++ b/tests/e2e/Services/GraphQL/ScopeTest.php @@ -34,7 +34,7 @@ class ScopeTest extends Scope 'x-appwrite-key' => $apiKey, ], $gqlPayload); - $message = "app.${projectId}@service.localhost (role: applications) missing scope (databases.write)"; + $message = "app.{$projectId}@service.localhost (role: applications) missing scope (databases.write)"; $this->assertArrayHasKey('errors', $database['body']); $this->assertEquals($message, $database['body']['errors'][0]['message']); } From eb7339fdf72a7a02aeba9d6d2ace665ad0caa8e2 Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 22 Jan 2024 13:01:46 +0200 Subject: [PATCH 11/26] composer.lock --- composer.lock | 52 +++++++++++++++++++++++++-------------------------- 1 file changed, 26 insertions(+), 26 deletions(-) diff --git a/composer.lock b/composer.lock index 87a52c233..716a8c98e 100644 --- a/composer.lock +++ b/composer.lock @@ -2276,16 +2276,16 @@ }, { "name": "utopia-php/messaging", - "version": "0.8.0", + "version": "0.8.1", "source": { "type": "git", "url": "https://github.com/utopia-php/messaging.git", - "reference": "64eca3faf02a79831f219d4f3ae05cd278a88b4b" + "reference": "bfb5014d3a8752901e50da1ae21bf309a6af5006" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/utopia-php/messaging/zipball/64eca3faf02a79831f219d4f3ae05cd278a88b4b", - "reference": "64eca3faf02a79831f219d4f3ae05cd278a88b4b", + "url": "https://api.github.com/repos/utopia-php/messaging/zipball/bfb5014d3a8752901e50da1ae21bf309a6af5006", + "reference": "bfb5014d3a8752901e50da1ae21bf309a6af5006", "shasum": "" }, "require": { @@ -2320,9 +2320,9 @@ ], "support": { "issues": "https://github.com/utopia-php/messaging/issues", - "source": "https://github.com/utopia-php/messaging/tree/0.8.0" + "source": "https://github.com/utopia-php/messaging/tree/0.8.1" }, - "time": "2023-12-15T06:44:08+00:00" + "time": "2024-01-10T23:55:03+00:00" }, { "name": "utopia-php/migration", @@ -3144,16 +3144,16 @@ "packages-dev": [ { "name": "appwrite/sdk-generator", - "version": "0.36.0", + "version": "0.36.2", "source": { "type": "git", "url": "https://github.com/appwrite/sdk-generator.git", - "reference": "3a10f1f895ed71120442ff71eb6adec3fd6b4e8a" + "reference": "0aa67479d75f0e0cb7b60454031534d7f0abaece" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/appwrite/sdk-generator/zipball/3a10f1f895ed71120442ff71eb6adec3fd6b4e8a", - "reference": "3a10f1f895ed71120442ff71eb6adec3fd6b4e8a", + "url": "https://api.github.com/repos/appwrite/sdk-generator/zipball/0aa67479d75f0e0cb7b60454031534d7f0abaece", + "reference": "0aa67479d75f0e0cb7b60454031534d7f0abaece", "shasum": "" }, "require": { @@ -3189,9 +3189,9 @@ "description": "Appwrite PHP library for generating API SDKs for multiple programming languages and platforms", "support": { "issues": "https://github.com/appwrite/sdk-generator/issues", - "source": "https://github.com/appwrite/sdk-generator/tree/0.36.0" + "source": "https://github.com/appwrite/sdk-generator/tree/0.36.2" }, - "time": "2023-11-20T10:03:06+00:00" + "time": "2024-01-19T01:04:35+00:00" }, { "name": "doctrine/deprecations", @@ -3774,16 +3774,16 @@ }, { "name": "phpdocumentor/type-resolver", - "version": "1.7.3", + "version": "1.8.0", "source": { "type": "git", "url": "https://github.com/phpDocumentor/TypeResolver.git", - "reference": "3219c6ee25c9ea71e3d9bbaf39c67c9ebd499419" + "reference": "fad452781b3d774e3337b0c0b245dd8e5a4455fc" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/phpDocumentor/TypeResolver/zipball/3219c6ee25c9ea71e3d9bbaf39c67c9ebd499419", - "reference": "3219c6ee25c9ea71e3d9bbaf39c67c9ebd499419", + "url": "https://api.github.com/repos/phpDocumentor/TypeResolver/zipball/fad452781b3d774e3337b0c0b245dd8e5a4455fc", + "reference": "fad452781b3d774e3337b0c0b245dd8e5a4455fc", "shasum": "" }, "require": { @@ -3826,9 +3826,9 @@ "description": "A PSR-5 based resolver of Class names, Types and Structural Element Names", "support": { "issues": "https://github.com/phpDocumentor/TypeResolver/issues", - "source": "https://github.com/phpDocumentor/TypeResolver/tree/1.7.3" + "source": "https://github.com/phpDocumentor/TypeResolver/tree/1.8.0" }, - "time": "2023-08-12T11:01:26+00:00" + "time": "2024-01-11T11:49:22+00:00" }, { "name": "phpspec/prophecy", @@ -5384,16 +5384,16 @@ }, { "name": "squizlabs/php_codesniffer", - "version": "3.8.0", + "version": "3.8.1", "source": { "type": "git", "url": "https://github.com/PHPCSStandards/PHP_CodeSniffer.git", - "reference": "5805f7a4e4958dbb5e944ef1e6edae0a303765e7" + "reference": "14f5fff1e64118595db5408e946f3a22c75807f7" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/PHPCSStandards/PHP_CodeSniffer/zipball/5805f7a4e4958dbb5e944ef1e6edae0a303765e7", - "reference": "5805f7a4e4958dbb5e944ef1e6edae0a303765e7", + "url": "https://api.github.com/repos/PHPCSStandards/PHP_CodeSniffer/zipball/14f5fff1e64118595db5408e946f3a22c75807f7", + "reference": "14f5fff1e64118595db5408e946f3a22c75807f7", "shasum": "" }, "require": { @@ -5403,11 +5403,11 @@ "php": ">=5.4.0" }, "require-dev": { - "phpunit/phpunit": "^4.0 || ^5.0 || ^6.0 || ^7.0 || ^8.0 || ^9.0" + "phpunit/phpunit": "^4.0 || ^5.0 || ^6.0 || ^7.0 || ^8.0 || ^9.3.4" }, "bin": [ - "bin/phpcs", - "bin/phpcbf" + "bin/phpcbf", + "bin/phpcs" ], "type": "library", "extra": { @@ -5460,7 +5460,7 @@ "type": "open_collective" } ], - "time": "2023-12-08T12:32:31+00:00" + "time": "2024-01-11T20:47:48+00:00" }, { "name": "swoole/ide-helper", From d0de6b57e9427f6ab1824f7cded94beb6789e347 Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 22 Jan 2024 17:12:12 +0200 Subject: [PATCH 12/26] jfif error --- composer.lock | 90 +++++++++++----------- phpunit.xml | 2 +- tests/e2e/Services/Storage/StorageBase.php | 8 ++ 3 files changed, 56 insertions(+), 44 deletions(-) diff --git a/composer.lock b/composer.lock index bc4f6ed2e..f5ed34880 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "b493981ce1e062708a4f86b0ceff315c", + "content-hash": "f4d494d258630f4b7e9427becf73cd7b", "packages": [ { "name": "adhocore/jwt", @@ -343,16 +343,16 @@ }, { "name": "dragonmantank/cron-expression", - "version": "v3.3.2", + "version": "v3.3.3", "source": { "type": "git", "url": "https://github.com/dragonmantank/cron-expression.git", - "reference": "782ca5968ab8b954773518e9e49a6f892a34b2a8" + "reference": "adfb1f505deb6384dc8b39804c5065dd3c8c8c0a" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/dragonmantank/cron-expression/zipball/782ca5968ab8b954773518e9e49a6f892a34b2a8", - "reference": "782ca5968ab8b954773518e9e49a6f892a34b2a8", + "url": "https://api.github.com/repos/dragonmantank/cron-expression/zipball/adfb1f505deb6384dc8b39804c5065dd3c8c8c0a", + "reference": "adfb1f505deb6384dc8b39804c5065dd3c8c8c0a", "shasum": "" }, "require": { @@ -392,7 +392,7 @@ ], "support": { "issues": "https://github.com/dragonmantank/cron-expression/issues", - "source": "https://github.com/dragonmantank/cron-expression/tree/v3.3.2" + "source": "https://github.com/dragonmantank/cron-expression/tree/v3.3.3" }, "funding": [ { @@ -400,7 +400,7 @@ "type": "github" } ], - "time": "2022-09-10T18:51:20+00:00" + "time": "2023-08-10T19:36:49+00:00" }, { "name": "guzzlehttp/guzzle", @@ -855,34 +855,38 @@ }, { "name": "league/csv", - "version": "9.7.1", + "version": "9.11.0", "source": { "type": "git", "url": "https://github.com/thephpleague/csv.git", - "reference": "0ec57e8264ec92565974ead0d1724cf1026e10c1" + "reference": "33149c4bea4949aa4fa3d03fb11ed28682168b39" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/thephpleague/csv/zipball/0ec57e8264ec92565974ead0d1724cf1026e10c1", - "reference": "0ec57e8264ec92565974ead0d1724cf1026e10c1", + "url": "https://api.github.com/repos/thephpleague/csv/zipball/33149c4bea4949aa4fa3d03fb11ed28682168b39", + "reference": "33149c4bea4949aa4fa3d03fb11ed28682168b39", "shasum": "" }, "require": { "ext-json": "*", "ext-mbstring": "*", - "php": "^7.3 || ^8.0" + "php": "^8.1.2" }, "require-dev": { - "ext-curl": "*", + "doctrine/collections": "^2.1.3", "ext-dom": "*", - "friendsofphp/php-cs-fixer": "^2.16", - "phpstan/phpstan": "^0.12.0", - "phpstan/phpstan-phpunit": "^0.12.0", - "phpstan/phpstan-strict-rules": "^0.12.0", - "phpunit/phpunit": "^9.5" + "ext-xdebug": "*", + "friendsofphp/php-cs-fixer": "^v3.22.0", + "phpbench/phpbench": "^1.2.14", + "phpstan/phpstan": "^1.10.26", + "phpstan/phpstan-deprecation-rules": "^1.1.3", + "phpstan/phpstan-phpunit": "^1.3.13", + "phpstan/phpstan-strict-rules": "^1.5.1", + "phpunit/phpunit": "^10.3.1", + "symfony/var-dumper": "^6.3.3" }, "suggest": { - "ext-dom": "Required to use the XMLConverter and or the HTMLConverter classes", + "ext-dom": "Required to use the XMLConverter and the HTMLConverter classes", "ext-iconv": "Needed to ease transcoding CSV using iconv stream filters" }, "type": "library", @@ -912,7 +916,7 @@ } ], "description": "CSV data manipulation made easy in PHP", - "homepage": "http://csv.thephpleague.com", + "homepage": "https://csv.thephpleague.com", "keywords": [ "convert", "csv", @@ -935,7 +939,7 @@ "type": "github" } ], - "time": "2021-04-17T16:32:08+00:00" + "time": "2023-09-23T10:09:54+00:00" }, { "name": "matomo/device-detector", @@ -1127,16 +1131,16 @@ }, { "name": "phpmailer/phpmailer", - "version": "v6.8.0", + "version": "v6.8.1", "source": { "type": "git", "url": "https://github.com/PHPMailer/PHPMailer.git", - "reference": "df16b615e371d81fb79e506277faea67a1be18f1" + "reference": "e88da8d679acc3824ff231fdc553565b802ac016" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/df16b615e371d81fb79e506277faea67a1be18f1", - "reference": "df16b615e371d81fb79e506277faea67a1be18f1", + "url": "https://api.github.com/repos/PHPMailer/PHPMailer/zipball/e88da8d679acc3824ff231fdc553565b802ac016", + "reference": "e88da8d679acc3824ff231fdc553565b802ac016", "shasum": "" }, "require": { @@ -1146,13 +1150,13 @@ "php": ">=5.5.0" }, "require-dev": { - "dealerdirect/phpcodesniffer-composer-installer": "^0.7.2", + "dealerdirect/phpcodesniffer-composer-installer": "^1.0", "doctrine/annotations": "^1.2.6 || ^1.13.3", "php-parallel-lint/php-console-highlighter": "^1.0.0", "php-parallel-lint/php-parallel-lint": "^1.3.2", "phpcompatibility/php-compatibility": "^9.3.5", "roave/security-advisories": "dev-latest", - "squizlabs/php_codesniffer": "^3.7.1", + "squizlabs/php_codesniffer": "^3.7.2", "yoast/phpunit-polyfills": "^1.0.4" }, "suggest": { @@ -1195,7 +1199,7 @@ "description": "PHPMailer is a full-featured email creation and transfer class for PHP", "support": { "issues": "https://github.com/PHPMailer/PHPMailer/issues", - "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.8.0" + "source": "https://github.com/PHPMailer/PHPMailer/tree/v6.8.1" }, "funding": [ { @@ -1203,7 +1207,7 @@ "type": "github" } ], - "time": "2023-03-06T14:43:22+00:00" + "time": "2023-08-29T08:26:30+00:00" }, { "name": "psr/http-client", @@ -1411,23 +1415,23 @@ }, { "name": "slickdeals/statsd", - "version": "3.1.0", + "version": "3.1.2", "source": { "type": "git", "url": "https://github.com/Slickdeals/statsd-php.git", - "reference": "225588a0a079e145359049f6e5e23eedb1b4c17f" + "reference": "03fc5d8d51aff9f651f97101033aad31c08d00de" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/Slickdeals/statsd-php/zipball/225588a0a079e145359049f6e5e23eedb1b4c17f", - "reference": "225588a0a079e145359049f6e5e23eedb1b4c17f", + "url": "https://api.github.com/repos/Slickdeals/statsd-php/zipball/03fc5d8d51aff9f651f97101033aad31c08d00de", + "reference": "03fc5d8d51aff9f651f97101033aad31c08d00de", "shasum": "" }, "require": { "php": ">= 7.3 || ^8" }, "replace": { - "domnikl/statsd": "self.version" + "domnikl/statsd": "<=3.0.2" }, "require-dev": { "friendsofphp/php-cs-fixer": "^3.0", @@ -1461,9 +1465,9 @@ ], "support": { "issues": "https://github.com/Slickdeals/statsd-php/issues", - "source": "https://github.com/Slickdeals/statsd-php/tree/3.1.0" + "source": "https://github.com/Slickdeals/statsd-php/tree/3.1.2" }, - "time": "2021-06-04T20:33:46+00:00" + "time": "2022-08-15T16:22:48+00:00" }, { "name": "symfony/deprecation-contracts", @@ -3140,16 +3144,16 @@ "packages-dev": [ { "name": "appwrite/sdk-generator", - "version": "0.36.1", + "version": "0.36.2", "source": { "type": "git", "url": "https://github.com/appwrite/sdk-generator.git", - "reference": "ca4700bfbbb8bcf1c0d5a49fc5efc38da98d0992" + "reference": "0aa67479d75f0e0cb7b60454031534d7f0abaece" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/appwrite/sdk-generator/zipball/ca4700bfbbb8bcf1c0d5a49fc5efc38da98d0992", - "reference": "ca4700bfbbb8bcf1c0d5a49fc5efc38da98d0992", + "url": "https://api.github.com/repos/appwrite/sdk-generator/zipball/0aa67479d75f0e0cb7b60454031534d7f0abaece", + "reference": "0aa67479d75f0e0cb7b60454031534d7f0abaece", "shasum": "" }, "require": { @@ -3185,9 +3189,9 @@ "description": "Appwrite PHP library for generating API SDKs for multiple programming languages and platforms", "support": { "issues": "https://github.com/appwrite/sdk-generator/issues", - "source": "https://github.com/appwrite/sdk-generator/tree/0.36.1" + "source": "https://github.com/appwrite/sdk-generator/tree/0.36.2" }, - "time": "2024-01-18T06:24:47+00:00" + "time": "2024-01-19T01:04:35+00:00" }, { "name": "doctrine/deprecations", @@ -5892,5 +5896,5 @@ "platform-overrides": { "php": "8.0" }, - "plugin-api-version": "2.3.0" + "plugin-api-version": "2.6.0" } diff --git a/phpunit.xml b/phpunit.xml index 90ebd4225..e77286605 100644 --- a/phpunit.xml +++ b/phpunit.xml @@ -6,7 +6,7 @@ convertNoticesToExceptions="true" convertWarningsToExceptions="true" processIsolation="false" - stopOnFailure="false" + stopOnFailure="true" > diff --git a/tests/e2e/Services/Storage/StorageBase.php b/tests/e2e/Services/Storage/StorageBase.php index c4a15585e..a80e366fc 100644 --- a/tests/e2e/Services/Storage/StorageBase.php +++ b/tests/e2e/Services/Storage/StorageBase.php @@ -473,6 +473,14 @@ trait StorageBase $this->assertEquals(201, $fileJfif['headers']['status-code']); $this->assertNotEmpty($fileJfif['body']['$id']); + + // shmuel + $data = file_get_contents(__DIR__ . '/../../../resources/disk-a/preview-test.jfif'); + $image = new \Imagick(); + $image->readImageBlob($data); + // End + + // TEST preview JXL $preview = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $fileJfif['body']['$id'] . '/preview', array_merge([ 'content-type' => 'application/json', From 52c386358950b8e44b71d1b70d92550d5e3b182a Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 22 Jan 2024 17:55:20 +0200 Subject: [PATCH 13/26] Comment failing tests --- tests/e2e/Services/Storage/StorageBase.php | 126 ++++++++++----------- 1 file changed, 63 insertions(+), 63 deletions(-) diff --git a/tests/e2e/Services/Storage/StorageBase.php b/tests/e2e/Services/Storage/StorageBase.php index a80e366fc..1c760ca3e 100644 --- a/tests/e2e/Services/Storage/StorageBase.php +++ b/tests/e2e/Services/Storage/StorageBase.php @@ -474,70 +474,70 @@ trait StorageBase $this->assertNotEmpty($fileJfif['body']['$id']); - // shmuel - $data = file_get_contents(__DIR__ . '/../../../resources/disk-a/preview-test.jfif'); - $image = new \Imagick(); - $image->readImageBlob($data); - // End +// // shmuel +// $data = file_get_contents(__DIR__ . '/../../../resources/disk-a/preview-test.jfif'); +// $image = new \Imagick(); +// $image->readImageBlob($data); +// // End - - // TEST preview JXL - $preview = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $fileJfif['body']['$id'] . '/preview', array_merge([ - 'content-type' => 'application/json', - 'x-appwrite-project' => $this->getProject()['$id'], - ], $this->getHeaders())); - - $this->assertEquals(200, $preview['headers']['status-code']); - $this->assertEquals('image/jpeg', $preview['headers']['content-type']); - $this->assertNotEmpty($preview['body']); - - //new image preview features - $file3 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ - 'content-type' => 'application/json', - 'x-appwrite-project' => $this->getProject()['$id'], - ], $this->getHeaders()), [ - 'width' => 300, - 'height' => 100, - 'borderRadius' => '50', - 'opacity' => '0.5', - 'output' => 'png', - 'rotation' => '45', - ]); - - $this->assertEquals(200, $file3['headers']['status-code']); - $this->assertEquals('image/png', $file3['headers']['content-type']); - $this->assertNotEmpty($file3['body']); - - $image = new \Imagick(); - $image->readImageBlob($file3['body']); - $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.png'); - - $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); - $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); - $this->assertEquals('PNG', $image->getImageFormat()); - - $file4 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ - 'content-type' => 'application/json', - 'x-appwrite-project' => $this->getProject()['$id'], - ], $this->getHeaders()), [ - 'width' => 200, - 'height' => 80, - 'borderWidth' => '5', - 'borderColor' => 'ff0000', - 'output' => 'jpg', - ]); - - $this->assertEquals(200, $file4['headers']['status-code']); - $this->assertEquals('image/jpeg', $file4['headers']['content-type']); - $this->assertNotEmpty($file4['body']); - - $image = new \Imagick(); - $image->readImageBlob($file4['body']); - $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.jpg'); - - $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); - $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); - $this->assertEquals('JPEG', $image->getImageFormat()); +// +// // TEST preview JXL +// $preview = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $fileJfif['body']['$id'] . '/preview', array_merge([ +// 'content-type' => 'application/json', +// 'x-appwrite-project' => $this->getProject()['$id'], +// ], $this->getHeaders())); +// +// $this->assertEquals(200, $preview['headers']['status-code']); +// $this->assertEquals('image/jpeg', $preview['headers']['content-type']); +// $this->assertNotEmpty($preview['body']); +// +// //new image preview features +// $file3 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ +// 'content-type' => 'application/json', +// 'x-appwrite-project' => $this->getProject()['$id'], +// ], $this->getHeaders()), [ +// 'width' => 300, +// 'height' => 100, +// 'borderRadius' => '50', +// 'opacity' => '0.5', +// 'output' => 'png', +// 'rotation' => '45', +// ]); +// +// $this->assertEquals(200, $file3['headers']['status-code']); +// $this->assertEquals('image/png', $file3['headers']['content-type']); +// $this->assertNotEmpty($file3['body']); +// +// $image = new \Imagick(); +// $image->readImageBlob($file3['body']); +// $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.png'); +// +// $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); +// $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); +// $this->assertEquals('PNG', $image->getImageFormat()); +// +// $file4 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ +// 'content-type' => 'application/json', +// 'x-appwrite-project' => $this->getProject()['$id'], +// ], $this->getHeaders()), [ +// 'width' => 200, +// 'height' => 80, +// 'borderWidth' => '5', +// 'borderColor' => 'ff0000', +// 'output' => 'jpg', +// ]); +// +// $this->assertEquals(200, $file4['headers']['status-code']); +// $this->assertEquals('image/jpeg', $file4['headers']['content-type']); +// $this->assertNotEmpty($file4['body']); +// +// $image = new \Imagick(); +// $image->readImageBlob($file4['body']); +// $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.jpg'); +// +// $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); +// $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); +// $this->assertEquals('JPEG', $image->getImageFormat()); $file5 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/download', array_merge([ 'content-type' => 'application/json', From 37bdb17cc62dadb5f49ea7fc19a3c66e2067cd10 Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 22 Jan 2024 19:24:12 +0200 Subject: [PATCH 14/26] Deprecated changes --- app/controllers/api/avatars.php | 2 +- app/controllers/api/databases.php | 2 +- app/controllers/api/projects.php | 2 +- app/controllers/api/users.php | 2 +- src/Appwrite/Auth/Hash/Scryptmodified.php | 3 ++- src/Appwrite/Auth/Validator/PersonalData.php | 8 ++++---- 6 files changed, 10 insertions(+), 9 deletions(-) diff --git a/app/controllers/api/avatars.php b/app/controllers/api/avatars.php index b6395774e..0ac1cee05 100644 --- a/app/controllers/api/avatars.php +++ b/app/controllers/api/avatars.php @@ -521,7 +521,7 @@ App::get('/v1/avatars/initials') // if there is no space, try to split by `_` underscore $words = (count($words) == 1) ? \explode('_', \strtoupper($name)) : $words; - $initials = null; + $initials = ''; $code = 0; foreach ($words as $key => $w) { diff --git a/app/controllers/api/databases.php b/app/controllers/api/databases.php index b3104f673..8b44dfe72 100644 --- a/app/controllers/api/databases.php +++ b/app/controllers/api/databases.php @@ -3934,7 +3934,7 @@ App::get('/v1/databases/:databaseId/collections/:collectionId/usage') }; $stats[$metric][] = [ 'value' => 0, - 'date' => DateTime::formatTz(DateTime::addSeconds(new \DateTime($stats[$metric][$last]['date'] ?? null), -1 * $diff)), + 'date' => DateTime::formatTz(DateTime::addSeconds(new \DateTime($stats[$metric][$last]['date'] ?? 'now'), -1 * $diff)), ]; $backfill--; } diff --git a/app/controllers/api/projects.php b/app/controllers/api/projects.php index 4e13747a5..fe4a32dec 100644 --- a/app/controllers/api/projects.php +++ b/app/controllers/api/projects.php @@ -385,7 +385,7 @@ App::get('/v1/projects/:projectId/usage') }; $stats[$metric][] = [ 'value' => 0, - 'date' => DateTime::formatTz(DateTime::addSeconds(new \DateTime($stats[$metric][$last]['date'] ?? null), -1 * $diff)), + 'date' => DateTime::formatTz(DateTime::addSeconds(new \DateTime($stats[$metric][$last]['date'] ?? 'now'), -1 * $diff)), ]; $backfill--; } diff --git a/app/controllers/api/users.php b/app/controllers/api/users.php index e090b3230..1d12d62b3 100644 --- a/app/controllers/api/users.php +++ b/app/controllers/api/users.php @@ -1844,7 +1844,7 @@ App::get('/v1/users/usage') }; $stats[$metric][] = [ 'value' => 0, - 'date' => DateTime::formatTz(DateTime::addSeconds(new \DateTime($stats[$metric][$last]['date'] ?? null), -1 * $diff)), + 'date' => DateTime::formatTz(DateTime::addSeconds(new \DateTime($stats[$metric][$last]['date'] ?? 'now'), -1 * $diff)), ]; $backfill--; } diff --git a/src/Appwrite/Auth/Hash/Scryptmodified.php b/src/Appwrite/Auth/Hash/Scryptmodified.php index 2d1cd4f16..7717f324e 100644 --- a/src/Appwrite/Auth/Hash/Scryptmodified.php +++ b/src/Appwrite/Auth/Hash/Scryptmodified.php @@ -60,7 +60,8 @@ class Scryptmodified extends Hash $saltBytes = \base64_decode($options['salt']); $saltSeparatorBytes = \base64_decode($options['saltSeparator']); - $derivedKey = \scrypt(\utf8_encode($password), $saltBytes . $saltSeparatorBytes, 16384, 8, 1, 64); + $password = mb_convert_encoding($password, 'UTF-8'); + $derivedKey = \scrypt($password, $saltBytes . $saltSeparatorBytes, 16384, 8, 1, 64); $derivedKey = \hex2bin($derivedKey); return $derivedKey; diff --git a/src/Appwrite/Auth/Validator/PersonalData.php b/src/Appwrite/Auth/Validator/PersonalData.php index 6f8ed0a8c..f93945755 100644 --- a/src/Appwrite/Auth/Validator/PersonalData.php +++ b/src/Appwrite/Auth/Validator/PersonalData.php @@ -43,10 +43,10 @@ class PersonalData extends Password if (!$this->strict) { $password = strtolower($password); - $this->userId = strtolower($this->userId); - $this->email = strtolower($this->email); - $this->name = strtolower($this->name); - $this->phone = strtolower($this->phone); + $this->userId = strtolower($this->userId ?? ''); + $this->email = strtolower($this->email ?? ''); + $this->name = strtolower($this->name ?? ''); + $this->phone = strtolower($this->phone ?? ''); } if ($this->userId && strpos($password, $this->userId) !== false) { From e8a390bff6374d2025aa093d17a7c525a74c024d Mon Sep 17 00:00:00 2001 From: fogelito Date: Tue, 23 Jan 2024 13:07:08 +0200 Subject: [PATCH 15/26] DSN changes check --- composer.json | 2 +- composer.lock | 14 +-- tests/e2e/Services/Storage/StorageBase.php | 118 ++++++++++----------- 3 files changed, 63 insertions(+), 71 deletions(-) diff --git a/composer.json b/composer.json index 3e6252f2b..b0794d31b 100644 --- a/composer.json +++ b/composer.json @@ -51,7 +51,7 @@ "utopia-php/config": "0.2.*", "utopia-php/database": "0.45.*", "utopia-php/domains": "0.5.*", - "utopia-php/dsn": "0.1.*", + "utopia-php/dsn": "0.2.*", "utopia-php/framework": "0.32.*", "utopia-php/image": "0.5.*", "utopia-php/locale": "0.4.*", diff --git a/composer.lock b/composer.lock index f5ed34880..462783a8b 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "f4d494d258630f4b7e9427becf73cd7b", + "content-hash": "577d31a631d293cfa0ab47cf5308badd", "packages": [ { "name": "adhocore/jwt", @@ -2028,16 +2028,16 @@ }, { "name": "utopia-php/dsn", - "version": "0.1.0", + "version": "0.2.0", "source": { "type": "git", "url": "https://github.com/utopia-php/dsn.git", - "reference": "17a5935eab1b89fb4b95600db50a1b6d5faa6cea" + "reference": "c11f37a12c3f6aaf9fea97ca7cb363dcc93668d7" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/utopia-php/dsn/zipball/17a5935eab1b89fb4b95600db50a1b6d5faa6cea", - "reference": "17a5935eab1b89fb4b95600db50a1b6d5faa6cea", + "url": "https://api.github.com/repos/utopia-php/dsn/zipball/c11f37a12c3f6aaf9fea97ca7cb363dcc93668d7", + "reference": "c11f37a12c3f6aaf9fea97ca7cb363dcc93668d7", "shasum": "" }, "require": { @@ -2069,9 +2069,9 @@ ], "support": { "issues": "https://github.com/utopia-php/dsn/issues", - "source": "https://github.com/utopia-php/dsn/tree/0.1.0" + "source": "https://github.com/utopia-php/dsn/tree/0.2.0" }, - "time": "2022-10-26T10:06:20+00:00" + "time": "2023-11-02T12:01:43+00:00" }, { "name": "utopia-php/framework", diff --git a/tests/e2e/Services/Storage/StorageBase.php b/tests/e2e/Services/Storage/StorageBase.php index 1c760ca3e..c4a15585e 100644 --- a/tests/e2e/Services/Storage/StorageBase.php +++ b/tests/e2e/Services/Storage/StorageBase.php @@ -473,71 +473,63 @@ trait StorageBase $this->assertEquals(201, $fileJfif['headers']['status-code']); $this->assertNotEmpty($fileJfif['body']['$id']); + // TEST preview JXL + $preview = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $fileJfif['body']['$id'] . '/preview', array_merge([ + 'content-type' => 'application/json', + 'x-appwrite-project' => $this->getProject()['$id'], + ], $this->getHeaders())); -// // shmuel -// $data = file_get_contents(__DIR__ . '/../../../resources/disk-a/preview-test.jfif'); -// $image = new \Imagick(); -// $image->readImageBlob($data); -// // End + $this->assertEquals(200, $preview['headers']['status-code']); + $this->assertEquals('image/jpeg', $preview['headers']['content-type']); + $this->assertNotEmpty($preview['body']); -// -// // TEST preview JXL -// $preview = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $fileJfif['body']['$id'] . '/preview', array_merge([ -// 'content-type' => 'application/json', -// 'x-appwrite-project' => $this->getProject()['$id'], -// ], $this->getHeaders())); -// -// $this->assertEquals(200, $preview['headers']['status-code']); -// $this->assertEquals('image/jpeg', $preview['headers']['content-type']); -// $this->assertNotEmpty($preview['body']); -// -// //new image preview features -// $file3 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ -// 'content-type' => 'application/json', -// 'x-appwrite-project' => $this->getProject()['$id'], -// ], $this->getHeaders()), [ -// 'width' => 300, -// 'height' => 100, -// 'borderRadius' => '50', -// 'opacity' => '0.5', -// 'output' => 'png', -// 'rotation' => '45', -// ]); -// -// $this->assertEquals(200, $file3['headers']['status-code']); -// $this->assertEquals('image/png', $file3['headers']['content-type']); -// $this->assertNotEmpty($file3['body']); -// -// $image = new \Imagick(); -// $image->readImageBlob($file3['body']); -// $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.png'); -// -// $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); -// $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); -// $this->assertEquals('PNG', $image->getImageFormat()); -// -// $file4 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ -// 'content-type' => 'application/json', -// 'x-appwrite-project' => $this->getProject()['$id'], -// ], $this->getHeaders()), [ -// 'width' => 200, -// 'height' => 80, -// 'borderWidth' => '5', -// 'borderColor' => 'ff0000', -// 'output' => 'jpg', -// ]); -// -// $this->assertEquals(200, $file4['headers']['status-code']); -// $this->assertEquals('image/jpeg', $file4['headers']['content-type']); -// $this->assertNotEmpty($file4['body']); -// -// $image = new \Imagick(); -// $image->readImageBlob($file4['body']); -// $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.jpg'); -// -// $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); -// $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); -// $this->assertEquals('JPEG', $image->getImageFormat()); + //new image preview features + $file3 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ + 'content-type' => 'application/json', + 'x-appwrite-project' => $this->getProject()['$id'], + ], $this->getHeaders()), [ + 'width' => 300, + 'height' => 100, + 'borderRadius' => '50', + 'opacity' => '0.5', + 'output' => 'png', + 'rotation' => '45', + ]); + + $this->assertEquals(200, $file3['headers']['status-code']); + $this->assertEquals('image/png', $file3['headers']['content-type']); + $this->assertNotEmpty($file3['body']); + + $image = new \Imagick(); + $image->readImageBlob($file3['body']); + $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.png'); + + $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); + $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); + $this->assertEquals('PNG', $image->getImageFormat()); + + $file4 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/preview', array_merge([ + 'content-type' => 'application/json', + 'x-appwrite-project' => $this->getProject()['$id'], + ], $this->getHeaders()), [ + 'width' => 200, + 'height' => 80, + 'borderWidth' => '5', + 'borderColor' => 'ff0000', + 'output' => 'jpg', + ]); + + $this->assertEquals(200, $file4['headers']['status-code']); + $this->assertEquals('image/jpeg', $file4['headers']['content-type']); + $this->assertNotEmpty($file4['body']); + + $image = new \Imagick(); + $image->readImageBlob($file4['body']); + $original = new \Imagick(__DIR__ . '/../../../resources/logo-after.jpg'); + + $this->assertEquals($image->getImageWidth(), $original->getImageWidth()); + $this->assertEquals($image->getImageHeight(), $original->getImageHeight()); + $this->assertEquals('JPEG', $image->getImageFormat()); $file5 = $this->client->call(Client::METHOD_GET, '/storage/buckets/' . $bucketId . '/files/' . $data['fileId'] . '/download', array_merge([ 'content-type' => 'application/json', From 8dd387cfbed7a03ca81798468577086a197a81cf Mon Sep 17 00:00:00 2001 From: fogelito Date: Tue, 23 Jan 2024 17:14:14 +0200 Subject: [PATCH 16/26] Parsing and Throw Error --- app/controllers/api/storage.php | 6 +++++- app/controllers/general.php | 2 +- tests/e2e/Services/Storage/StorageBase.php | 2 +- 3 files changed, 7 insertions(+), 3 deletions(-) diff --git a/app/controllers/api/storage.php b/app/controllers/api/storage.php index 011f83f4b..0242aacce 100644 --- a/app/controllers/api/storage.php +++ b/app/controllers/api/storage.php @@ -976,7 +976,11 @@ App::get('/v1/storage/buckets/:bucketId/files/:fileId/preview') break; } - $image = new Image($source); + try { + $image = new Image($source); + } catch (ImagickException $e) { + throw new Exception(Exception::STORAGE_FILE_TYPE_UNSUPPORTED, $e->getMessage()); + } $image->crop((int) $width, (int) $height, $gravity); diff --git a/app/controllers/general.php b/app/controllers/general.php index 3020cd665..1d0fbdfec 100644 --- a/app/controllers/general.php +++ b/app/controllers/general.php @@ -739,7 +739,7 @@ App::error() 'code' => $code, 'file' => $file, 'line' => $line, - 'trace' => $trace, + 'trace' => \json_encode($trace, JSON_UNESCAPED_UNICODE) === false ? [] : $trace, 'version' => $version, 'type' => $type, ] : [ diff --git a/tests/e2e/Services/Storage/StorageBase.php b/tests/e2e/Services/Storage/StorageBase.php index c4a15585e..b17073cd5 100644 --- a/tests/e2e/Services/Storage/StorageBase.php +++ b/tests/e2e/Services/Storage/StorageBase.php @@ -27,7 +27,7 @@ trait StorageBase 'name' => 'Test Bucket', 'fileSecurity' => true, 'maximumFileSize' => 2000000, //2MB - 'allowedFileExtensions' => ["jpg", "png", 'jfif'], + 'allowedFileExtensions' => ['jpg', 'png', 'jfif'], 'permissions' => [ Permission::read(Role::any()), Permission::create(Role::any()), From 55c72521e9700633c022e9684f18b4b1afdd4c61 Mon Sep 17 00:00:00 2001 From: fogelito Date: Wed, 24 Jan 2024 10:37:28 +0200 Subject: [PATCH 17/26] Upgrade base:0.7.2 --- Dockerfile | 2 +- app/controllers/general.php | 2 +- composer.json | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/Dockerfile b/Dockerfile index 58fa2167e..d78ccf591 100755 --- a/Dockerfile +++ b/Dockerfile @@ -29,7 +29,7 @@ ENV VITE_APPWRITE_GROWTH_ENDPOINT=$VITE_APPWRITE_GROWTH_ENDPOINT RUN npm ci RUN npm run build -FROM appwrite/base:0.7.1 as final +FROM appwrite/base:0.7.2 as final LABEL maintainer="team@appwrite.io" diff --git a/app/controllers/general.php b/app/controllers/general.php index 1d0fbdfec..94060a3cf 100644 --- a/app/controllers/general.php +++ b/app/controllers/general.php @@ -739,7 +739,7 @@ App::error() 'code' => $code, 'file' => $file, 'line' => $line, - 'trace' => \json_encode($trace, JSON_UNESCAPED_UNICODE) === false ? [] : $trace, + 'trace' => \json_encode($trace, JSON_UNESCAPED_UNICODE) === false ? [] : $trace, // check for failing encode 'version' => $version, 'type' => $type, ] : [ diff --git a/composer.json b/composer.json index b0794d31b..ca2fb0a3c 100644 --- a/composer.json +++ b/composer.json @@ -53,7 +53,7 @@ "utopia-php/domains": "0.5.*", "utopia-php/dsn": "0.2.*", "utopia-php/framework": "0.32.*", - "utopia-php/image": "0.5.*", + "utopia-php/image": "0.6.*", "utopia-php/locale": "0.4.*", "utopia-php/logger": "0.3.*", "utopia-php/messaging": "0.8.*", From b4ccfb9a02b466b824e01d2b1c8f15303779debc Mon Sep 17 00:00:00 2001 From: fogelito Date: Wed, 24 Jan 2024 12:31:49 +0200 Subject: [PATCH 18/26] composer.lock --- composer.lock | 22 +++++++++------------- 1 file changed, 9 insertions(+), 13 deletions(-) diff --git a/composer.lock b/composer.lock index 462783a8b..5e3c78684 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "577d31a631d293cfa0ab47cf5308badd", + "content-hash": "d14cf808498f53a1a217350b2e95ec0d", "packages": [ { "name": "adhocore/jwt", @@ -2120,16 +2120,16 @@ }, { "name": "utopia-php/image", - "version": "0.5.4", + "version": "0.6.0", "source": { "type": "git", "url": "https://github.com/utopia-php/image.git", - "reference": "ca5f436f9aa22dedaa6648f24f3687733808e336" + "reference": "88f7209172bdabd81e76ac981c95fac117dc6e08" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/utopia-php/image/zipball/ca5f436f9aa22dedaa6648f24f3687733808e336", - "reference": "ca5f436f9aa22dedaa6648f24f3687733808e336", + "url": "https://api.github.com/repos/utopia-php/image/zipball/88f7209172bdabd81e76ac981c95fac117dc6e08", + "reference": "88f7209172bdabd81e76ac981c95fac117dc6e08", "shasum": "" }, "require": { @@ -2137,6 +2137,8 @@ "php": ">=8.0" }, "require-dev": { + "laravel/pint": "1.2.*", + "phpstan/phpstan": "1.9.x-dev", "phpunit/phpunit": "^9.3", "vimeo/psalm": "4.13.1" }, @@ -2150,12 +2152,6 @@ "license": [ "MIT" ], - "authors": [ - { - "name": "Eldad Fux", - "email": "eldad@appwrite.io" - } - ], "description": "A simple Image manipulation library", "keywords": [ "framework", @@ -2166,9 +2162,9 @@ ], "support": { "issues": "https://github.com/utopia-php/image/issues", - "source": "https://github.com/utopia-php/image/tree/0.5.4" + "source": "https://github.com/utopia-php/image/tree/0.6.0" }, - "time": "2022-05-11T12:30:41+00:00" + "time": "2024-01-24T06:59:44+00:00" }, { "name": "utopia-php/locale", From 61dd4a123cc4dcbdd9e82858ab4b7ce0b6cdf9c7 Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Mon, 29 Jan 2024 19:29:58 +1300 Subject: [PATCH 19/26] Remove descriptions --- app/config/collections.php | 22 ---------------- app/controllers/api/messaging.php | 44 ++++++------------------------- 2 files changed, 8 insertions(+), 58 deletions(-) diff --git a/app/config/collections.php b/app/config/collections.php index 96e7cd7b2..59050b9b4 100644 --- a/app/config/collections.php +++ b/app/config/collections.php @@ -1573,17 +1573,6 @@ $commonCollections = [ 'array' => false, 'filters' => [], ], - [ - '$id' => ID::custom('description'), - 'type' => Database::VAR_STRING, - 'format' => '', - 'size' => 256, - 'signed' => true, - 'required' => false, - 'default' => '', - 'array' => false, - 'filters' => [], - ], [ '$id' => ID::custom('status'), 'type' => Database::VAR_STRING, @@ -1744,17 +1733,6 @@ $commonCollections = [ 'array' => false, 'filters' => [], ], - [ - '$id' => ID::custom('description'), - 'type' => Database::VAR_STRING, - 'format' => '', - 'size' => 2048, - 'signed' => true, - 'required' => false, - 'default' => null, - 'array' => false, - 'filters' => [], - ], [ '$id' => ID::custom('total'), 'type' => Database::VAR_INTEGER, diff --git a/app/controllers/api/messaging.php b/app/controllers/api/messaging.php index a6c2f923e..2628c96a1 100644 --- a/app/controllers/api/messaging.php +++ b/app/controllers/api/messaging.php @@ -1701,17 +1701,15 @@ App::post('/v1/messaging/topics') ->label('sdk.response.model', Response::MODEL_TOPIC) ->param('topicId', '', new CustomId(), 'Topic ID. Choose a custom Topic ID or a new Topic ID.') ->param('name', '', new Text(128), 'Topic Name.') - ->param('description', '', new Text(2048), 'Topic Description.', true) ->inject('queueForEvents') ->inject('dbForProject') ->inject('response') - ->action(function (string $topicId, string $name, string $description, Event $queueForEvents, Database $dbForProject, Response $response) { + ->action(function (string $topicId, string $name, Event $queueForEvents, Database $dbForProject, Response $response) { $topicId = $topicId == 'unique()' ? ID::unique() : $topicId; $topic = new Document([ '$id' => $topicId, 'name' => $name, - 'description' => $description ]); try { @@ -1898,11 +1896,10 @@ App::patch('/v1/messaging/topics/:topicId') ->label('sdk.response.model', Response::MODEL_TOPIC) ->param('topicId', '', new UID(), 'Topic ID.') ->param('name', '', new Text(128), 'Topic Name.', true) - ->param('description', '', new Text(2048), 'Topic Description.', true) ->inject('queueForEvents') ->inject('dbForProject') ->inject('response') - ->action(function (string $topicId, string $name, string $description, Event $queueForEvents, Database $dbForProject, Response $response) { + ->action(function (string $topicId, string $name, Event $queueForEvents, Database $dbForProject, Response $response) { $topic = $dbForProject->getDocument('topics', $topicId); if ($topic->isEmpty()) { @@ -1913,10 +1910,6 @@ App::patch('/v1/messaging/topics/:topicId') $topic->setAttribute('name', $name); } - if (!empty($description)) { - $topic->setAttribute('description', $description); - } - $topic = $dbForProject->updateDocument('topics', $topicId, $topic); $queueForEvents @@ -2301,7 +2294,6 @@ App::post('/v1/messaging/messages/email') ->param('targets', [], new ArrayList(new UID()), 'List of Targets IDs.', true) ->param('cc', [], new ArrayList(new UID()), 'Array of target IDs to be added as CC.', true) ->param('bcc', [], new ArrayList(new UID()), 'Array of target IDs to be added as BCC.', true) - ->param('description', '', new Text(256), 'Description for message.', true) ->param('status', MessageStatus::DRAFT, new WhiteList([MessageStatus::DRAFT, MessageStatus::SCHEDULED, MessageStatus::PROCESSING]), 'Message Status. Value must be one of: ' . implode(', ', [MessageStatus::DRAFT, MessageStatus::SCHEDULED, MessageStatus::PROCESSING]) . '.', true) ->param('html', false, new Boolean(), 'Is content of type HTML', true) ->param('scheduledAt', null, new DatetimeValidator(requireDateInFuture: true), 'Scheduled delivery time for message in [ISO 8601](https://www.iso.org/iso-8601-date-and-time-format.html) format. DateTime value must be in future.', true) @@ -2311,7 +2303,7 @@ App::post('/v1/messaging/messages/email') ->inject('project') ->inject('queueForMessaging') ->inject('response') - ->action(function (string $messageId, string $subject, string $content, array $topics, array $users, array $targets, array $cc, array $bcc, string $description, string $status, bool $html, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { + ->action(function (string $messageId, string $subject, string $content, array $topics, array $users, array $targets, array $cc, array $bcc, string $status, bool $html, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { $messageId = $messageId == 'unique()' ? ID::unique() : $messageId; @@ -2350,7 +2342,6 @@ App::post('/v1/messaging/messages/email') 'topics' => $topics, 'users' => $users, 'targets' => $targets, - 'description' => $description, 'scheduledAt' => $scheduledAt, 'data' => [ 'subject' => $subject, @@ -2420,7 +2411,6 @@ App::post('/v1/messaging/messages/sms') ->param('topics', [], new ArrayList(new UID()), 'List of Topic IDs.', true) ->param('users', [], new ArrayList(new UID()), 'List of User IDs.', true) ->param('targets', [], new ArrayList(new UID()), 'List of Targets IDs.', true) - ->param('description', '', new Text(256), 'Description for Message.', true) ->param('status', MessageStatus::DRAFT, new WhiteList([MessageStatus::DRAFT, MessageStatus::SCHEDULED, MessageStatus::PROCESSING]), 'Message Status. Value must be one of: ' . implode(', ', [MessageStatus::DRAFT, MessageStatus::SCHEDULED, MessageStatus::PROCESSING]) . '.', true) ->param('scheduledAt', null, new DatetimeValidator(requireDateInFuture: true), 'Scheduled delivery time for message in [ISO 8601](https://www.iso.org/iso-8601-date-and-time-format.html) format. DateTime value must be in future.', true) ->inject('queueForEvents') @@ -2429,7 +2419,7 @@ App::post('/v1/messaging/messages/sms') ->inject('project') ->inject('queueForMessaging') ->inject('response') - ->action(function (string $messageId, string $content, array $topics, array $users, array $targets, string $description, string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { + ->action(function (string $messageId, string $content, array $topics, array $users, array $targets, string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { $messageId = $messageId == 'unique()' ? ID::unique() : $messageId; @@ -2466,7 +2456,6 @@ App::post('/v1/messaging/messages/sms') 'topics' => $topics, 'users' => $users, 'targets' => $targets, - 'description' => $description, 'data' => [ 'content' => $content, ], @@ -2532,7 +2521,6 @@ App::post('/v1/messaging/messages/push') ->param('topics', [], new ArrayList(new UID()), 'List of Topic IDs.', true) ->param('users', [], new ArrayList(new UID()), 'List of User IDs.', true) ->param('targets', [], new ArrayList(new UID()), 'List of Targets IDs.', true) - ->param('description', '', new Text(256), 'Description for Message.', true) ->param('data', null, new JSON(), 'Additional Data for push notification.', true) ->param('action', '', new Text(256), 'Action for push notification.', true) ->param('icon', '', new Text(256), 'Icon for push notification. Available only for Android and Web Platform.', true) @@ -2548,7 +2536,7 @@ App::post('/v1/messaging/messages/push') ->inject('project') ->inject('queueForMessaging') ->inject('response') - ->action(function (string $messageId, string $title, string $body, array $topics, array $users, array $targets, string $description, ?array $data, string $action, string $icon, string $sound, string $color, string $tag, string $badge, string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { + ->action(function (string $messageId, string $title, string $body, array $topics, array $users, array $targets, ?array $data, string $action, string $icon, string $sound, string $color, string $tag, string $badge, string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { $messageId = $messageId == 'unique()' ? ID::unique() : $messageId; @@ -2595,7 +2583,6 @@ App::post('/v1/messaging/messages/push') 'topics' => $topics, 'users' => $users, 'targets' => $targets, - 'description' => $description, 'scheduledAt' => $scheduledAt, 'data' => $pushData, 'status' => $status, @@ -2869,7 +2856,6 @@ App::patch('/v1/messaging/messages/email/:messageId') ->param('users', null, new ArrayList(new UID()), 'List of User IDs.', true) ->param('targets', null, new ArrayList(new UID()), 'List of Targets IDs.', true) ->param('subject', null, new Text(998), 'Email Subject.', true) - ->param('description', null, new Text(256), 'Description for Message.', true) ->param('content', null, new Text(64230), 'Email Content.', true) ->param('status', MessageStatus::DRAFT, new WhiteList([MessageStatus::DRAFT, MessageStatus::SCHEDULED, MessageStatus::PROCESSING]), 'Message Status. Value must be one of: ' . implode(', ', [MessageStatus::DRAFT, MessageStatus::SCHEDULED, MessageStatus::PROCESSING]) . '.', true) ->param('html', null, new Boolean(), 'Is content of type HTML', true) @@ -2882,7 +2868,7 @@ App::patch('/v1/messaging/messages/email/:messageId') ->inject('project') ->inject('queueForMessaging') ->inject('response') - ->action(function (string $messageId, ?array $topics, ?array $users, ?array $targets, ?string $subject, ?string $description, ?string $content, ?string $status, ?bool $html, ?array $cc, ?array $bcc, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { + ->action(function (string $messageId, ?array $topics, ?array $users, ?array $targets, ?string $subject, ?string $content, ?string $status, ?bool $html, ?array $cc, ?array $bcc, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { $message = $dbForProject->getDocument('messages', $messageId); if ($message->isEmpty()) { @@ -2933,10 +2919,6 @@ App::patch('/v1/messaging/messages/email/:messageId') $message->setAttribute('data', $data); - if (!\is_null($description)) { - $message->setAttribute('description', $description); - } - if (!\is_null($status)) { $message->setAttribute('status', $status); } @@ -3007,7 +2989,6 @@ App::patch('/v1/messaging/messages/sms/:messageId') ->param('topics', null, new ArrayList(new UID()), 'List of Topic IDs.', true) ->param('users', null, new ArrayList(new UID()), 'List of User IDs.', true) ->param('targets', null, new ArrayList(new UID()), 'List of Targets IDs.', true) - ->param('description', null, new Text(256), 'Description for Message.', true) ->param('content', null, new Text(64230), 'Email Content.', true) ->param('status', null, new WhiteList(['draft', 'cancelled', 'processing']), 'Message Status. Value must be either draft or cancelled or processing.', true) ->param('scheduledAt', null, new DatetimeValidator(requireDateInFuture: true), 'Scheduled delivery time for message in [ISO 8601](https://www.iso.org/iso-8601-date-and-time-format.html) format. DateTime value must be in future.', true) @@ -3017,7 +2998,7 @@ App::patch('/v1/messaging/messages/sms/:messageId') ->inject('project') ->inject('queueForMessaging') ->inject('response') - ->action(function (string $messageId, ?array $topics, ?array $users, ?array $targets, ?string $description, ?string $content, ?string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { + ->action(function (string $messageId, ?array $topics, ?array $users, ?array $targets, ?string $content, ?string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { $message = $dbForProject->getDocument('messages', $messageId); if ($message->isEmpty()) { @@ -3056,10 +3037,6 @@ App::patch('/v1/messaging/messages/sms/:messageId') $message->setAttribute('status', $status); } - if (!\is_null($description)) { - $message->setAttribute('description', $description); - } - if (!\is_null($scheduledAt)) { if (\is_null($message->getAttribute(('scheduleId')))) { $schedule = $dbForConsole->createDocument('schedules', new Document([ @@ -3126,7 +3103,6 @@ App::patch('/v1/messaging/messages/push/:messageId') ->param('topics', null, new ArrayList(new UID()), 'List of Topic IDs.', true) ->param('users', null, new ArrayList(new UID()), 'List of User IDs.', true) ->param('targets', null, new ArrayList(new UID()), 'List of Targets IDs.', true) - ->param('description', null, new Text(256), 'Description for Message.', true) ->param('title', null, new Text(256), 'Title for push notification.', true) ->param('body', null, new Text(64230), 'Body for push notification.', true) ->param('data', null, new JSON(), 'Additional Data for push notification.', true) @@ -3144,7 +3120,7 @@ App::patch('/v1/messaging/messages/push/:messageId') ->inject('project') ->inject('queueForMessaging') ->inject('response') - ->action(function (string $messageId, ?array $topics, ?array $users, ?array $targets, ?string $description, ?string $title, ?string $body, ?array $data, ?string $action, ?string $icon, ?string $sound, ?string $color, ?string $tag, ?int $badge, ?string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { + ->action(function (string $messageId, ?array $topics, ?array $users, ?array $targets, ?string $title, ?string $body, ?array $data, ?string $action, ?string $icon, ?string $sound, ?string $color, ?string $tag, ?int $badge, ?string $status, ?string $scheduledAt, Event $queueForEvents, Database $dbForProject, Database $dbForConsole, Document $project, Messaging $queueForMessaging, Response $response) { $message = $dbForProject->getDocument('messages', $messageId); if ($message->isEmpty()) { @@ -3215,10 +3191,6 @@ App::patch('/v1/messaging/messages/push/:messageId') $message->setAttribute('status', $status); } - if (!\is_null($description)) { - $message->setAttribute('description', $description); - } - if (!\is_null($scheduledAt)) { if (\is_null($message->getAttribute(('scheduleId')))) { $schedule = $dbForConsole->createDocument('schedules', new Document([ From afbbada22af75f32f936e7ae4a39e6708d3d0a1d Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Mon, 29 Jan 2024 19:36:05 +1300 Subject: [PATCH 20/26] Fix tests --- tests/e2e/Services/GraphQL/Base.php | 44 +++++++------------ tests/e2e/Services/GraphQL/MessagingTest.php | 8 ---- .../e2e/Services/Messaging/MessagingBase.php | 20 --------- .../Messaging/MessagingConsoleClientTest.php | 3 +- 4 files changed, 17 insertions(+), 58 deletions(-) diff --git a/tests/e2e/Services/GraphQL/Base.php b/tests/e2e/Services/GraphQL/Base.php index 4683c3d8b..9ffabc8fc 100644 --- a/tests/e2e/Services/GraphQL/Base.php +++ b/tests/e2e/Services/GraphQL/Base.php @@ -2000,11 +2000,10 @@ trait Base } }'; case self::$CREATE_TOPIC: - return 'mutation createTopic($topicId: String!, $name: String!, $description: String!) { - messagingCreateTopic(topicId: $topicId, name: $name, description: $description) { + return 'mutation createTopic($topicId: String!, $name: String!) { + messagingCreateTopic(topicId: $topicId, name: $name) { _id name - description } }'; case self::$LIST_TOPICS: @@ -2014,7 +2013,6 @@ trait Base topics { _id name - description } } }'; @@ -2023,15 +2021,13 @@ trait Base messagingGetTopic(topicId: $topicId) { _id name - description } }'; case self::$UPDATE_TOPIC: - return 'mutation updateTopic($topicId: String!, $name: String!, $description: String!) { - messagingUpdateTopic(topicId: $topicId, name: $name, description: $description) { + return 'mutation updateTopic($topicId: String!, $name: String!) { + messagingUpdateTopic(topicId: $topicId, name: $name) { _id name - description } }'; case self::$DELETE_TOPIC: @@ -2098,8 +2094,8 @@ trait Base } }'; case self::$CREATE_EMAIL: - return 'mutation createEmail($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $subject: String!, $content: String!, $status: String, $description: String, $html: Boolean, $cc: [String], $bcc: [String], $scheduledAt: String) { - messagingCreateEmail(messageId: $messageId, topics: $topics, users: $users, targets: $targets, subject: $subject, content: $content, status: $status, description: $description, html: $html, cc: $cc, bcc: $bcc, scheduledAt: $scheduledAt) { + return 'mutation createEmail($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $subject: String!, $content: String!, $status: String, $html: Boolean, $cc: [String], $bcc: [String], $scheduledAt: String) { + messagingCreateEmail(messageId: $messageId, topics: $topics, users: $users, targets: $targets, subject: $subject, content: $content, status: $status, html: $html, cc: $cc, bcc: $bcc, scheduledAt: $scheduledAt) { _id topics users @@ -2109,12 +2105,11 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$CREATE_SMS: - return 'mutation createSMS($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $content: String!, $status: String, $description: String, $scheduledAt: String) { - messagingCreateSMS(messageId: $messageId, topics: $topics, users: $users, targets: $targets, content: $content, status: $status, description: $description, scheduledAt: $scheduledAt) { + return 'mutation createSMS($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $content: String!, $status: String, $scheduledAt: String) { + messagingCreateSMS(messageId: $messageId, topics: $topics, users: $users, targets: $targets, content: $content, status: $status, scheduledAt: $scheduledAt) { _id topics users @@ -2124,12 +2119,11 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$CREATE_PUSH_NOTIFICATION: - return 'mutation createPushNotification($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $title: String!, $body: String!, $data: Json, $action: String, $icon: String, $sound: String, $color: String, $tag: String, $badge: String, $status: String, $description: String, $scheduledAt: String) { - messagingCreatePushNotification(messageId: $messageId, topics: $topics, users: $users, targets: $targets, title: $title, body: $body, data: $data, action: $action, icon: $icon, sound: $sound, color: $color, tag: $tag, badge: $badge, status: $status, description: $description, scheduledAt: $scheduledAt) { + return 'mutation createPushNotification($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $title: String!, $body: String!, $data: Json, $action: String, $icon: String, $sound: String, $color: String, $tag: String, $badge: String, $status: String, $scheduledAt: String) { + messagingCreatePushNotification(messageId: $messageId, topics: $topics, users: $users, targets: $targets, title: $title, body: $body, data: $data, action: $action, icon: $icon, sound: $sound, color: $color, tag: $tag, badge: $badge, status: $status, scheduledAt: $scheduledAt) { _id topics users @@ -2139,7 +2133,6 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$LIST_MESSAGES: @@ -2157,7 +2150,6 @@ trait Base deliveryErrors deliveredTotal status - description } } }'; @@ -2174,12 +2166,11 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$UPDATE_EMAIL: - return 'mutation updateEmail($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $subject: String, $content: String, $status: String, $description: String, $html: Boolean, $cc: [String], $bcc: [String], $scheduledAt: String) { - messagingUpdateEmail(messageId: $messageId, topics: $topics, users: $users, targets: $targets, subject: $subject, content: $content, status: $status, description: $description, html: $html, cc: $cc, bcc: $bcc, scheduledAt: $scheduledAt) { + return 'mutation updateEmail($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $subject: String, $content: String, $status: String, , $html: Boolean, $cc: [String], $bcc: [String], $scheduledAt: String) { + messagingUpdateEmail(messageId: $messageId, topics: $topics, users: $users, targets: $targets, subject: $subject, content: $content, status: $status, html: $html, cc: $cc, bcc: $bcc, scheduledAt: $scheduledAt) { _id topics users @@ -2189,12 +2180,11 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$UPDATE_SMS: - return 'mutation updateSMS($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $content: String, $status: String, $description: String, $scheduledAt: String) { - messagingUpdateSMS(messageId: $messageId, topics: $topics, users: $users, targets: $targets, content: $content, status: $status, description: $description, scheduledAt: $scheduledAt) { + return 'mutation updateSMS($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $content: String, $status: String, $scheduledAt: String) { + messagingUpdateSMS(messageId: $messageId, topics: $topics, users: $users, targets: $targets, content: $content, status: $status, scheduledAt: $scheduledAt) { _id topics users @@ -2204,12 +2194,11 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$UPDATE_PUSH_NOTIFICATION: - return 'mutation updatePushNotification($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $title: String, $body: String, $data: Json, $action: String, $icon: String, $sound: String, $color: String, $tag: String, $badge: String, $status: String, $description: String, $scheduledAt: String) { - messagingUpdatePushNotification(messageId: $messageId, topics: $topics, users: $users, targets: $targets, title: $title, body: $body, data: $data, action: $action, icon: $icon, sound: $sound, color: $color, tag: $tag, badge: $badge, status: $status, description: $description, scheduledAt: $scheduledAt) { + return 'mutation updatePushNotification($messageId: String!, $topics: [String!], $users: [String!], $targets: [String!], $title: String, $body: String, $data: Json, $action: String, $icon: String, $sound: String, $color: String, $tag: String, $badge: String, $status: String, $scheduledAt: String) { + messagingUpdatePushNotification(messageId: $messageId, topics: $topics, users: $users, targets: $targets, title: $title, body: $body, data: $data, action: $action, icon: $icon, sound: $sound, color: $color, tag: $tag, badge: $badge, status: $status, scheduledAt: $scheduledAt) { _id topics users @@ -2219,7 +2208,6 @@ trait Base deliveryErrors deliveredTotal status - description } }'; case self::$COMPLEX_QUERY: diff --git a/tests/e2e/Services/GraphQL/MessagingTest.php b/tests/e2e/Services/GraphQL/MessagingTest.php index 40134c221..dba655b55 100644 --- a/tests/e2e/Services/GraphQL/MessagingTest.php +++ b/tests/e2e/Services/GraphQL/MessagingTest.php @@ -285,7 +285,6 @@ class MessagingTest extends Scope 'variables' => [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Active users', ], ]; $response = $this->client->call(Client::METHOD_POST, '/graphql', \array_merge([ @@ -296,7 +295,6 @@ class MessagingTest extends Scope $this->assertEquals(200, $response['headers']['status-code']); $this->assertEquals('topic1', $response['body']['data']['messagingCreateTopic']['name']); - $this->assertEquals('Active users', $response['body']['data']['messagingCreateTopic']['description']); return $response['body']['data']['messagingCreateTopic']; } @@ -313,7 +311,6 @@ class MessagingTest extends Scope 'variables' => [ 'topicId' => $topicId, 'name' => 'topic2', - 'description' => 'Inactive users', ], ]; $response = $this->client->call(Client::METHOD_POST, '/graphql', \array_merge([ @@ -324,7 +321,6 @@ class MessagingTest extends Scope $this->assertEquals(200, $response['headers']['status-code']); $this->assertEquals('topic2', $response['body']['data']['messagingUpdateTopic']['name']); - $this->assertEquals('Inactive users', $response['body']['data']['messagingUpdateTopic']['description']); return $topicId; } @@ -368,7 +364,6 @@ class MessagingTest extends Scope $this->assertEquals(200, $response['headers']['status-code']); $this->assertEquals('topic2', $response['body']['data']['messagingGetTopic']['name']); - $this->assertEquals('Inactive users', $response['body']['data']['messagingGetTopic']['description']); } /** @@ -594,7 +589,6 @@ class MessagingTest extends Scope 'variables' => [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Active users', ], ]; $topic = $this->client->call(Client::METHOD_POST, '/graphql', \array_merge([ @@ -801,7 +795,6 @@ class MessagingTest extends Scope 'variables' => [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Active users', ], ]; $topic = $this->client->call(Client::METHOD_POST, '/graphql', \array_merge([ @@ -1006,7 +999,6 @@ class MessagingTest extends Scope 'variables' => [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Active users', ], ]; $topic = $this->client->call(Client::METHOD_POST, '/graphql', \array_merge([ diff --git a/tests/e2e/Services/Messaging/MessagingBase.php b/tests/e2e/Services/Messaging/MessagingBase.php index 0e7fca850..efd72481c 100644 --- a/tests/e2e/Services/Messaging/MessagingBase.php +++ b/tests/e2e/Services/Messaging/MessagingBase.php @@ -269,7 +269,6 @@ trait MessagingBase ]); $this->assertEquals(201, $response['headers']['status-code']); $this->assertEquals('my-app', $response['body']['name']); - $this->assertEquals('', $response['body']['description']); return $response['body']; } @@ -285,11 +284,9 @@ trait MessagingBase 'x-appwrite-key' => $this->getProject()['apiKey'], ], [ 'name' => 'android-app', - 'description' => 'updated-description' ]); $this->assertEquals(200, $response['headers']['status-code']); $this->assertEquals('android-app', $response['body']['name']); - $this->assertEquals('updated-description', $response['body']['description']); return $response['body']['$id']; } @@ -298,17 +295,6 @@ trait MessagingBase */ public function testListTopic(string $topicId) { - $response = $this->client->call(Client::METHOD_GET, '/messaging/topics', [ - 'content-type' => 'application/json', - 'x-appwrite-project' => $this->getProject()['$id'], - 'x-appwrite-key' => $this->getProject()['apiKey'], - ], [ - 'search' => 'updated-description', - ]); - - $this->assertEquals(200, $response['headers']['status-code']); - $this->assertEquals(1, \count($response['body']['topics'])); - $response = $this->client->call(Client::METHOD_GET, '/messaging/topics', [ 'content-type' => 'application/json', 'x-appwrite-project' => $this->getProject()['$id'], @@ -350,7 +336,6 @@ trait MessagingBase ]); $this->assertEquals(200, $response['headers']['status-code']); $this->assertEquals('android-app', $response['body']['name']); - $this->assertEquals('updated-description', $response['body']['description']); $this->assertEquals(0, $response['body']['total']); } @@ -407,7 +392,6 @@ trait MessagingBase $this->assertEquals(200, $topic['headers']['status-code']); $this->assertEquals('android-app', $topic['body']['name']); - $this->assertEquals('updated-description', $topic['body']['description']); $this->assertEquals(1, $topic['body']['total']); return [ @@ -626,7 +610,6 @@ trait MessagingBase $this->assertEquals(200, $topic['headers']['status-code']); $this->assertEquals('android-app', $topic['body']['name']); - $this->assertEquals('updated-description', $topic['body']['description']); $this->assertEquals(0, $topic['body']['total']); } @@ -724,7 +707,6 @@ trait MessagingBase ], [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Test Topic' ]); $this->assertEquals(201, $topic['headers']['status-code']); @@ -881,7 +863,6 @@ trait MessagingBase ], [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Test Topic' ]); $this->assertEquals(201, $topic['headers']['status-code']); @@ -1042,7 +1023,6 @@ trait MessagingBase ], [ 'topicId' => ID::unique(), 'name' => 'topic1', - 'description' => 'Test Topic' ]); $this->assertEquals(201, $topic['headers']['status-code']); diff --git a/tests/e2e/Services/Messaging/MessagingConsoleClientTest.php b/tests/e2e/Services/Messaging/MessagingConsoleClientTest.php index 0baa465b4..89315db27 100644 --- a/tests/e2e/Services/Messaging/MessagingConsoleClientTest.php +++ b/tests/e2e/Services/Messaging/MessagingConsoleClientTest.php @@ -170,7 +170,6 @@ class MessagingConsoleClientTest extends Scope ], $this->getHeaders()), [ 'topicId' => ID::unique(), 'name' => 'my-app', - 'description' => 'web app' ]); $this->assertEquals(201, $topic['headers']['status-code']); @@ -178,7 +177,7 @@ class MessagingConsoleClientTest extends Scope 'content-type' => 'application/json', 'x-appwrite-project' => $this->getProject()['$id'], ], $this->getHeaders()), [ - 'description' => 'updated-description' + 'name' => 'android-app' ]); $this->assertEquals(200, $response['headers']['status-code']); From 91e09f5f439a5761eafbbc1bf517fe588c16b57c Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 29 Jan 2024 08:55:11 +0200 Subject: [PATCH 21/26] Increase labels max value --- composer.lock | 36 ++++++++++++++++-------------------- 1 file changed, 16 insertions(+), 20 deletions(-) diff --git a/composer.lock b/composer.lock index 6a5ed7215..c4b39a427 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "758fc8f128df402e823dd2cb340bf78b", + "content-hash": "5973c7f0f194221f9117010d08c4b10c", "packages": [ { "name": "adhocore/jwt", @@ -1306,16 +1306,16 @@ }, { "name": "utopia-php/dsn", - "version": "0.1.0", + "version": "0.2.0", "source": { "type": "git", "url": "https://github.com/utopia-php/dsn.git", - "reference": "17a5935eab1b89fb4b95600db50a1b6d5faa6cea" + "reference": "c11f37a12c3f6aaf9fea97ca7cb363dcc93668d7" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/utopia-php/dsn/zipball/17a5935eab1b89fb4b95600db50a1b6d5faa6cea", - "reference": "17a5935eab1b89fb4b95600db50a1b6d5faa6cea", + "url": "https://api.github.com/repos/utopia-php/dsn/zipball/c11f37a12c3f6aaf9fea97ca7cb363dcc93668d7", + "reference": "c11f37a12c3f6aaf9fea97ca7cb363dcc93668d7", "shasum": "" }, "require": { @@ -1347,9 +1347,9 @@ ], "support": { "issues": "https://github.com/utopia-php/dsn/issues", - "source": "https://github.com/utopia-php/dsn/tree/0.1.0" + "source": "https://github.com/utopia-php/dsn/tree/0.2.0" }, - "time": "2022-10-26T10:06:20+00:00" + "time": "2023-11-02T12:01:43+00:00" }, { "name": "utopia-php/framework", @@ -1398,16 +1398,16 @@ }, { "name": "utopia-php/image", - "version": "0.5.4", + "version": "0.6.0", "source": { "type": "git", "url": "https://github.com/utopia-php/image.git", - "reference": "ca5f436f9aa22dedaa6648f24f3687733808e336" + "reference": "88f7209172bdabd81e76ac981c95fac117dc6e08" }, "dist": { "type": "zip", - "url": "https://api.github.com/repos/utopia-php/image/zipball/ca5f436f9aa22dedaa6648f24f3687733808e336", - "reference": "ca5f436f9aa22dedaa6648f24f3687733808e336", + "url": "https://api.github.com/repos/utopia-php/image/zipball/88f7209172bdabd81e76ac981c95fac117dc6e08", + "reference": "88f7209172bdabd81e76ac981c95fac117dc6e08", "shasum": "" }, "require": { @@ -1415,6 +1415,8 @@ "php": ">=8.0" }, "require-dev": { + "laravel/pint": "1.2.*", + "phpstan/phpstan": "1.9.x-dev", "phpunit/phpunit": "^9.3", "vimeo/psalm": "4.13.1" }, @@ -1428,12 +1430,6 @@ "license": [ "MIT" ], - "authors": [ - { - "name": "Eldad Fux", - "email": "eldad@appwrite.io" - } - ], "description": "A simple Image manipulation library", "keywords": [ "framework", @@ -1444,9 +1440,9 @@ ], "support": { "issues": "https://github.com/utopia-php/image/issues", - "source": "https://github.com/utopia-php/image/tree/0.5.4" + "source": "https://github.com/utopia-php/image/tree/0.6.0" }, - "time": "2022-05-11T12:30:41+00:00" + "time": "2024-01-24T06:59:44+00:00" }, { "name": "utopia-php/locale", @@ -5174,5 +5170,5 @@ "platform-overrides": { "php": "8.0" }, - "plugin-api-version": "2.3.0" + "plugin-api-version": "2.6.0" } From 53efc930c0e003e0e0f346941cce25f8ab080413 Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 29 Jan 2024 08:58:11 +0200 Subject: [PATCH 22/26] Remove Database::INDEX_ARRAY --- app/controllers/api/databases.php | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/controllers/api/databases.php b/app/controllers/api/databases.php index ccfc93f13..c85cabf8a 100644 --- a/app/controllers/api/databases.php +++ b/app/controllers/api/databases.php @@ -2330,7 +2330,7 @@ App::post('/v1/databases/:databaseId/collections/:collectionId/indexes') ->param('databaseId', '', new UID(), 'Database ID.') ->param('collectionId', '', new UID(), 'Collection ID. You can create a new collection using the Database service [server integration](https://appwrite.io/docs/server/databases#databasesCreateCollection).') ->param('key', null, new Key(), 'Index Key.') - ->param('type', null, new WhiteList([Database::INDEX_KEY, Database::INDEX_FULLTEXT, Database::INDEX_UNIQUE, Database::INDEX_SPATIAL, Database::INDEX_ARRAY]), 'Index type.') + ->param('type', null, new WhiteList([Database::INDEX_KEY, Database::INDEX_FULLTEXT, Database::INDEX_UNIQUE, Database::INDEX_SPATIAL]), 'Index type.') ->param('attributes', null, new ArrayList(new Key(true), APP_LIMIT_ARRAY_PARAMS_SIZE), 'Array of attributes to index. Maximum of ' . APP_LIMIT_ARRAY_PARAMS_SIZE . ' attributes are allowed, each 32 characters long.') ->param('orders', [], new ArrayList(new WhiteList(['ASC', 'DESC'], false, Database::VAR_STRING), APP_LIMIT_ARRAY_PARAMS_SIZE), 'Array of index orders. Maximum of ' . APP_LIMIT_ARRAY_PARAMS_SIZE . ' orders are allowed.', true) ->inject('response') From accf75156c39f03955c6ed816836e08b0b29db05 Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 29 Jan 2024 09:24:44 +0200 Subject: [PATCH 23/26] Remove "league/csv": "9.7.1" --- composer.json | 3 +- composer.lock | 86 +-------------------------------------------------- 2 files changed, 2 insertions(+), 87 deletions(-) diff --git a/composer.json b/composer.json index faa5ad64f..8e5baa102 100644 --- a/composer.json +++ b/composer.json @@ -73,8 +73,7 @@ "phpmailer/phpmailer": "6.8.0", "chillerlan/php-qrcode": "4.3.4", "adhocore/jwt": "1.1.2", - "webonyx/graphql-php": "14.11.*", - "league/csv": "9.7.1" + "webonyx/graphql-php": "14.11.*" }, "repositories": [ { diff --git a/composer.lock b/composer.lock index c4b39a427..40b4a8a70 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "5973c7f0f194221f9117010d08c4b10c", + "content-hash": "f79e94c7f6a698230d2165929f8df11e", "packages": [ { "name": "adhocore/jwt", @@ -461,90 +461,6 @@ }, "time": "2021-10-08T21:21:46+00:00" }, - { - "name": "league/csv", - "version": "9.7.1", - "source": { - "type": "git", - "url": "https://github.com/thephpleague/csv.git", - "reference": "0ec57e8264ec92565974ead0d1724cf1026e10c1" - }, - "dist": { - "type": "zip", - "url": "https://api.github.com/repos/thephpleague/csv/zipball/0ec57e8264ec92565974ead0d1724cf1026e10c1", - "reference": "0ec57e8264ec92565974ead0d1724cf1026e10c1", - "shasum": "" - }, - "require": { - "ext-json": "*", - "ext-mbstring": "*", - "php": "^7.3 || ^8.0" - }, - "require-dev": { - "ext-curl": "*", - "ext-dom": "*", - "friendsofphp/php-cs-fixer": "^2.16", - "phpstan/phpstan": "^0.12.0", - "phpstan/phpstan-phpunit": "^0.12.0", - "phpstan/phpstan-strict-rules": "^0.12.0", - "phpunit/phpunit": "^9.5" - }, - "suggest": { - "ext-dom": "Required to use the XMLConverter and or the HTMLConverter classes", - "ext-iconv": "Needed to ease transcoding CSV using iconv stream filters" - }, - "type": "library", - "extra": { - "branch-alias": { - "dev-master": "9.x-dev" - } - }, - "autoload": { - "files": [ - "src/functions_include.php" - ], - "psr-4": { - "League\\Csv\\": "src" - } - }, - "notification-url": "https://packagist.org/downloads/", - "license": [ - "MIT" - ], - "authors": [ - { - "name": "Ignace Nyamagana Butera", - "email": "nyamsprod@gmail.com", - "homepage": "https://github.com/nyamsprod/", - "role": "Developer" - } - ], - "description": "CSV data manipulation made easy in PHP", - "homepage": "http://csv.thephpleague.com", - "keywords": [ - "convert", - "csv", - "export", - "filter", - "import", - "read", - "transform", - "write" - ], - "support": { - "docs": "https://csv.thephpleague.com", - "issues": "https://github.com/thephpleague/csv/issues", - "rss": "https://github.com/thephpleague/csv/releases.atom", - "source": "https://github.com/thephpleague/csv" - }, - "funding": [ - { - "url": "https://github.com/sponsors/nyamsprod", - "type": "github" - } - ], - "time": "2021-04-17T16:32:08+00:00" - }, { "name": "matomo/device-detector", "version": "6.1.6", From ee6c081641c076d2af4135f4ca8a696d38efb17e Mon Sep 17 00:00:00 2001 From: fogelito Date: Mon, 29 Jan 2024 11:20:25 +0200 Subject: [PATCH 24/26] stopOnFailure --- phpunit.xml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/phpunit.xml b/phpunit.xml index e77286605..90ebd4225 100644 --- a/phpunit.xml +++ b/phpunit.xml @@ -6,7 +6,7 @@ convertNoticesToExceptions="true" convertWarningsToExceptions="true" processIsolation="false" - stopOnFailure="true" + stopOnFailure="false" > From 1be0997db5045d8eeff23e4e14b8dcfb263040f2 Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Tue, 30 Jan 2024 19:11:42 +1300 Subject: [PATCH 25/26] Update composer platform --- composer.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/composer.json b/composer.json index fa4bfb4c8..593c325ec 100644 --- a/composer.json +++ b/composer.json @@ -95,7 +95,7 @@ }, "config": { "platform": { - "php": "8.0" + "php": "8.2" } } } From f0a14a441e7210430d39d8f687e8d281dc49570f Mon Sep 17 00:00:00 2001 From: Jake Barnby Date: Tue, 30 Jan 2024 19:15:11 +1300 Subject: [PATCH 26/26] Add back csv lib --- composer.json | 3 +- composer.lock | 93 +++++++++++++++++++++++++++++++++++++++++++++++++-- 2 files changed, 93 insertions(+), 3 deletions(-) diff --git a/composer.json b/composer.json index 593c325ec..b9d43a822 100644 --- a/composer.json +++ b/composer.json @@ -73,7 +73,8 @@ "phpmailer/phpmailer": "6.8.0", "chillerlan/php-qrcode": "4.3.4", "adhocore/jwt": "1.1.2", - "webonyx/graphql-php": "14.11.*" + "webonyx/graphql-php": "14.11.*", + "league/csv": "^9.14" }, "repositories": [ { diff --git a/composer.lock b/composer.lock index f663ae5fe..34b655a88 100644 --- a/composer.lock +++ b/composer.lock @@ -4,7 +4,7 @@ "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "This file is @generated automatically" ], - "content-hash": "f79e94c7f6a698230d2165929f8df11e", + "content-hash": "b692b61971354d653a0a0aa27bd5a646", "packages": [ { "name": "adhocore/jwt", @@ -461,6 +461,95 @@ }, "time": "2021-10-08T21:21:46+00:00" }, + { + "name": "league/csv", + "version": "9.14.0", + "source": { + "type": "git", + "url": "https://github.com/thephpleague/csv.git", + "reference": "34bf0df7340b60824b9449b5c526fcc3325070d5" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/thephpleague/csv/zipball/34bf0df7340b60824b9449b5c526fcc3325070d5", + "reference": "34bf0df7340b60824b9449b5c526fcc3325070d5", + "shasum": "" + }, + "require": { + "ext-filter": "*", + "ext-json": "*", + "ext-mbstring": "*", + "php": "^8.1.2" + }, + "require-dev": { + "doctrine/collections": "^2.1.4", + "ext-dom": "*", + "ext-xdebug": "*", + "friendsofphp/php-cs-fixer": "^v3.22.0", + "phpbench/phpbench": "^1.2.15", + "phpstan/phpstan": "^1.10.50", + "phpstan/phpstan-deprecation-rules": "^1.1.4", + "phpstan/phpstan-phpunit": "^1.3.15", + "phpstan/phpstan-strict-rules": "^1.5.2", + "phpunit/phpunit": "^10.5.3", + "symfony/var-dumper": "^6.4.0" + }, + "suggest": { + "ext-dom": "Required to use the XMLConverter and the HTMLConverter classes", + "ext-iconv": "Needed to ease transcoding CSV using iconv stream filters" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "9.x-dev" + } + }, + "autoload": { + "files": [ + "src/functions_include.php" + ], + "psr-4": { + "League\\Csv\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Ignace Nyamagana Butera", + "email": "nyamsprod@gmail.com", + "homepage": "https://github.com/nyamsprod/", + "role": "Developer" + } + ], + "description": "CSV data manipulation made easy in PHP", + "homepage": "https://csv.thephpleague.com", + "keywords": [ + "convert", + "csv", + "export", + "filter", + "import", + "read", + "transform", + "write" + ], + "support": { + "docs": "https://csv.thephpleague.com", + "issues": "https://github.com/thephpleague/csv/issues", + "rss": "https://github.com/thephpleague/csv/releases.atom", + "source": "https://github.com/thephpleague/csv" + }, + "funding": [ + { + "url": "https://github.com/sponsors/nyamsprod", + "type": "github" + } + ], + "time": "2023-12-29T07:34:53+00:00" + }, { "name": "matomo/device-detector", "version": "6.1.6", @@ -5084,7 +5173,7 @@ "ext-fileinfo": "*" }, "platform-overrides": { - "php": "8.0" + "php": "8.2" }, "plugin-api-version": "2.6.0" }