diff --git a/app/admin/controller/Base.php b/app/admin/controller/Base.php index 3fd8e72fc..9ae1bad09 100644 --- a/app/admin/controller/Base.php +++ b/app/admin/controller/Base.php @@ -70,7 +70,7 @@ class Base extends Controller * @param array $data * @throws \think\db\exception\DbException */ - protected function _form_filter(array $data) + protected function _form_filter(array &$data) { if ($this->request->isGet()) { $this->types = SystemBase::types(); diff --git a/app/wechat/controller/Config.php b/app/wechat/controller/Config.php index c63da47fd..479f5484e 100644 --- a/app/wechat/controller/Config.php +++ b/app/wechat/controller/Config.php @@ -19,6 +19,7 @@ namespace app\wechat\controller; use app\wechat\service\WechatService; use think\admin\Builder; use think\admin\Controller; +use think\admin\Storage; use think\admin\storage\LocalStorage; /** @@ -134,15 +135,6 @@ class Config extends Controller } } - /** - * 微信支付测试 - * @auth true - */ - public function payment_test() - { - $this->fetch(); - } - /** * 微信支付修改 * @auth true @@ -151,27 +143,53 @@ class Config extends Controller public function payment_save() { if ($this->request->isPost()) { - if ($this->request->post('wechat.mch_ssl_type') === 'p12') { - if (!LocalStorage::instance()->has(input('wechat.mch_ssl_p12', '-'), true)) { - $this->error('商户证书 P12 证书不能为空!'); - } - $content = LocalStorage::instance()->get(input('wechat.mch_ssl_p12', '-'), true); - if (!openssl_pkcs12_read($content, $certs, input('wechat.mch_id'))) { - $this->error('商户账号与 P12 证书不匹配!'); - } - } elseif ($this->request->post('wechat.mch_ssl_type') === 'pem') { - if (!LocalStorage::instance()->has(input('wechat.mch_ssl_key', '-'), true)) { + $local = LocalStorage::instance(); + $wechat = $this->request->post('wechat'); + if ($wechat['mch_ssl_type'] === 'pem') { + if (empty($wechat['mch_ssl_key']) || !$local->has($wechat['mch_ssl_key'], true)) { $this->error('商户证书 KEY 不能为空!'); } - if (!LocalStorage::instance()->has(input('wechat.mch_ssl_cer', '-'), true)) { + if (empty($wechat['mch_ssl_cer']) || !$local->has($wechat['mch_ssl_cer'], true)) { $this->error('商户证书 CERT 不能为空!'); } } - foreach ($this->request->post() as $k => $v) sysconf($k, $v); + if ($wechat['mch_ssl_type'] === 'p12') { + if (empty($wechat['mch_ssl_p12']) || !$local->has($wechat['mch_ssl_p12'], true)) { + $this->error('商户证书 P12 不能为空!'); + } + $content = $local->get($wechat['mch_ssl_p12'], true); + if (openssl_pkcs12_read($content, $certs, $wechat['mch_id'])) { + $wechat['mch_ssl_key'] = $local->set(Storage::name($certs['pkey'], 'pem'), $certs['pkey'], true)['url']; + $wechat['mch_ssl_cer'] = $local->set(Storage::name($certs['cert'], 'pem'), $certs['cert'], true)['url']; + } else { + $this->error('商户账号与 P12 证书不匹配!'); + } + } + // 记录文本格式参数,兼容分布式部署 + sysdata('plugin.wechat.payment.config', [ + 'appid' => WechatService::getAppid(), + 'mch_id' => $wechat['mch_id'], + 'mch_key' => $wechat['mch_key'], + 'mch_v3_key' => $wechat['mch_v3_key'], + 'ssl_key_text' => $local->get($wechat['mch_ssl_key'], true), + 'ssl_cer_text' => $local->get($wechat['mch_ssl_cer'], true), + ]); + // 记录证书路径参数,兼容历史参数 + foreach ($wechat as $k => $v) sysconf("wechat.{$k}", $v); + // 记录操作历史并返回保存结果 sysoplog('微信授权配置', '修改微信支付配置成功'); $this->success('微信支付配置成功!'); } else { $this->error('抱歉,访问方式错误!'); } } + + /** + * 微信支付测试 + * @auth true + */ + public function payment_test() + { + $this->fetch(); + } } diff --git a/app/wechat/controller/News.php b/app/wechat/controller/News.php index 6cdd42ad8..92d45e932 100644 --- a/app/wechat/controller/News.php +++ b/app/wechat/controller/News.php @@ -73,7 +73,7 @@ class News extends Controller public function add() { if ($this->request->isGet()) { - $this->title = '新建图文'; + $this->title = '新建微信图文'; $this->fetch('form'); } else { $update = [ @@ -103,14 +103,14 @@ class News extends Controller if ($this->request->get('output') === 'json') { $this->success('获取数据成功!', MediaService::news($this->id)); } else { - $this->title = '编辑图文'; + $this->title = '编辑微信图文'; $this->fetch('form'); } } else { $ids = $this->_buildArticle($this->request->post('data', [])); [$map, $data] = [['id' => $this->id], ['article_id' => $ids]]; - if (WechatNews::mk()->where($map)->update($data)) { - $this->success('更新成功!', 'javascript:history.back()'); + if (WechatNews::mk()->where($map)->update($data) !== false) { + $this->success('图文更新成功!', 'javascript:history.back()'); } else { $this->error('更新失败,请稍候再试!'); } @@ -136,7 +136,7 @@ class News extends Controller $ids = []; foreach ($data as $vo) { if (empty($vo['digest'])) { - $vo['digest'] = mb_substr(strip_tags(str_replace(["\s", ' '], '', $vo['content'])), 0, 120); + $vo['digest'] = mb_substr(strip_tags(preg_replace('#(\s+| )#', '', $vo['content'])), 0, 120); } $vo['create_at'] = date('Y-m-d H:i:s'); if (empty($vo['id'])) { diff --git a/app/wechat/service/WechatService.php b/app/wechat/service/WechatService.php index baaf4d305..c0025d445 100644 --- a/app/wechat/service/WechatService.php +++ b/app/wechat/service/WechatService.php @@ -20,6 +20,7 @@ use think\admin\Exception; use think\admin\extend\JsonRpcClient; use think\admin\Library; use think\admin\Service; +use think\admin\Storage; use think\admin\storage\LocalStorage; use think\exception\HttpResponseException; @@ -181,17 +182,53 @@ class WechatService extends Service 'mch_v3_key' => sysconf('wechat.mch_v3_key'), 'cache_path' => syspath('runtime/wechat'), ]; + if (in_array($sslType = strtolower(sysconf('wechat.mch_ssl_type')), ['p12', 'pem'])) { + [$local = LocalStorage::instance(), $options = static::withWxpayCert($options)]; + if ((empty($options['ssl_cer']) || empty($options['ssl_key'])) && $sslType === 'p12') { + if (openssl_pkcs12_read($local->get(sysconf('wechat.mch_ssl_p12'), true), $certs, $options['mch_id'])) { + sysconf('wechat.mch_ssl_cer', $local->set(Storage::name($certs['pkey'], 'pem'), $certs['pkey'], true)['url']); + sysconf('wechat.mch_ssl_key', $local->set(Storage::name($certs['cert'], 'pem'), $certs['cert'], true)['url']); + static::withWxpayCert($options); + } else { + throw new Exception('商户账号与 P12 证书不匹配!'); + } + } + } + return $options; + } + + /** + * 处理支付证书配置 + * @param array $options + * @return array + * @throws \think\admin\Exception + */ + private static function withWxpayCert(array &$options): array + { + // 文本模式主要是为了解决分布式部署 $local = LocalStorage::instance(); - switch (strtolower(sysconf('wechat.mch_ssl_type'))) { - case 'p12': - $options['ssl_p12'] = $local->path(sysconf('wechat.mch_ssl_p12'), true); - break; - case 'pem': - $options['ssl_cer'] = $local->path(sysconf('wechat.mch_ssl_cer'), true); - $options['ssl_key'] = $local->path(sysconf('wechat.mch_ssl_key'), true); - $options['cert_public'] = $local->path(sysconf('wechat.mch_ssl_cer'), true); - $options['cert_private'] = $local->path(sysconf('wechat.mch_ssl_key'), true); - break; + if (!empty($data = sysdata('plugin.wechat.payment.config'))) { + if (empty($data['ssl_key_text']) || empty($data['ssl_cer_text'])) { + throw new Exception('商户证书不能为空!'); + } + $name1 = Storage::name($data['ssl_cer_text'], 'pem'); + $name2 = Storage::name($data['ssl_key_text'], 'pem'); + if ($local->has($name1, true) && $local->has($name2, true)) { + $sslCer = $local->set($name1, $data['ssl_cer_text'], true)['file']; + $sslKey = $local->set($name2, $data['ssl_key_text'], true)['file']; + } else { + $sslCer = $local->path($name1, true); + $sslKey = $local->path($name2, true); + } + $options['ssl_cer'] = $sslCer; + $options['ssl_key'] = $sslKey; + $options['cert_public'] = $sslCer; + $options['cert_private'] = $sslKey; + } else { + $sslCer = $local->path(sysconf('wechat.mch_ssl_cer'), true); + $sslKey = $local->path(sysconf('wechat.mch_ssl_key'), true); + if (is_file($sslCer)) $options['cert_public'] = $options['ssl_cer'] = $sslCer; + if (is_file($sslKey)) $options['cert_private'] = $options['ssl_key'] = $sslKey; } return $options; } diff --git a/app/wechat/view/config/payment.html b/app/wechat/view/config/payment.html index 5ae02af77..83b2f572f 100644 --- a/app/wechat/view/config/payment.html +++ b/app/wechat/view/config/payment.html @@ -7,104 +7,94 @@ {/block} {block name="content"} -
i.pages||(i.curr=e,t.render())});e&&o.on(e,"change",function(){var e=this.value;i.curr*e>i.count&&(i.curr=Math.ceil(i.count/e)),i.limit=e,t.render()}),r&&o.on(r,"click",function(){l()})}},t.prototype.skip=function(t){var i,e;t&&(i=this,(e=t[c]("input")[0])&&o.on(e,"keyup",function(e){var a=this.value,e=e.keyCode;/^(37|38|39|40)$/.test(e)||(/\D/.test(a)&&(this.value=a.replace(/\D/,"")),13===e&&i.jump(t,!0))}))},t.prototype.render=function(e){var a=this,t=a.config,i=a.type(),n=a.view(),i=(2===i?t.elem&&(t.elem.innerHTML=n):3===i?t.elem.html(n):r[u](t.elem)&&(r[u](t.elem).innerHTML=n),t.jump&&t.jump(t,e),r[u]("layui-laypage-"+t.index));a.jump(i),t.hash&&!e&&(location.hash="!"+t.hash+"="+t.curr),a.skip(i)},{render:function(e){return new t(e).index},index:layui.laypage?layui.laypage.index+1e4:0,on:function(a,e,t){return a.attachEvent?a.attachEvent("on"+e,function(e){e.target=e.srcElement,t.call(a,e)}):a.addEventListener(e,t,!1),this}});e("laypage",o)});!function(i,p){"use strict";var n=i.layui&&layui.define,l={getPath:i.lay&&lay.getPath?lay.getPath:"",link:function(e,t,a){v.path&&i.lay&&lay.layui&&lay.layui.link(v.path+e,t,a)}},e=i.LAYUI_GLOBAL||{},d="layui-laydate-id",v={v:"5.5.0",config:{weekStart:0},index:i.laydate&&i.laydate.v?1e5:0,path:e.laydate_dir||l.getPath,set:function(e){var t=this;return t.config=lay.extend({},t.config,e),t},ready:function(e){var t="laydate",a=(n?"modules/":"")+"laydate.css?v="+v.v;return n?layui["layui.all"]?"function"==typeof e&&e():layui.addcss(a,e,t):l.link(a,e,t),this}},s=function(){var t=this,e=t.config.id;return(s.that[e]=t).inst={hint:function(e){t.hint.call(t,e)},reload:function(e){t.reload.call(t,e)},config:t.config}},a="laydate",x="layui-this",k="laydate-disabled",h=[100,2e5],D="layui-laydate-static",w="layui-laydate-list",o="laydate-selected",r="layui-laydate-hint",y="laydate-day-prev",m="laydate-day-next",C=".laydate-btns-confirm",M="laydate-time-text",L="laydate-btns-time",T="layui-laydate-preview",E="layui-laydate-main",S="layui-laydate-shade",I=function(e){var t,a=this,n=(a.index=++v.index,a.config=lay.extend({},a.config,v.config,e),lay(e.elem||a.config.elem));return 1
\u5fc5\u987b\u9075\u5faa\u4e0b\u8ff0\u683c\u5f0f\uff1a
","
\u5df2\u4e3a\u4f60\u91cd\u7f6e"],preview:"\u5f53\u524d\u9009\u4e2d\u7684\u7ed3\u679c"},en:{weeks:["Su","Mo","Tu","We","Th","Fr","Sa"],time:["Hours","Minutes","Seconds"],timeTips:"Select Time",startTime:"Start Time",endTime:"End Time",dateTips:"Select Date",month:["Jan","Feb","Mar","Apr","May","Jun","Jul","Aug","Sep","Oct","Nov","Dec"],tools:{confirm:"Confirm",clear:"Clear",now:"Now"},timeout:"End time cannot be less than start Time
Please re-select",invalidDate:"Invalid date",formatError:["The date format error
Must be followed\uff1a
","
It has been reset"],preview:"The selected result"}};return e[this.config.lang]||e.cn},I.prototype.reload=function(e){this.config=lay.extend({},this.config,e),this.init()},I.prototype.init=function(){var r=this,o=r.config,e="static"===o.position,t={year:"yyyy",month:"yyyy-MM",date:"yyyy-MM-dd",time:"HH:mm:ss",datetime:"yyyy-MM-dd HH:mm:ss"};o.elem=lay(o.elem),o.eventElem=lay(o.eventElem),o.elem[0]&&("array"!==layui.type(o.theme)&&(o.theme=[o.theme]),o.fullPanel&&("datetime"!==o.type||o.range)&&delete o.fullPanel,r.rangeStr=o.range?"string"==typeof o.range?o.range:"-":"",r.rangeLinked=!(!o.range||!o.rangeLinked||"date"!==o.type&&"datetime"!==o.type),r.autoCalendarModel=function(){var e=r.rangeLinked;return r.rangeLinked=o.range&&("date"===o.type||"datetime"===o.type)&&(!r.startDate||!r.endDate||r.startDate&&r.endDate&&r.startDate.year===r.endDate.year&&r.startDate.month===r.endDate.month),lay(r.elem)[r.rangeLinked?"addClass":"removeClass"]("layui-laydate-linkage"),r.rangeLinked!=e},r.autoCalendarModel.auto=r.rangeLinked&&"auto"===o.rangeLinked,"array"===layui.type(o.range)&&(r.rangeElem=[lay(o.range[0]),lay(o.range[1])]),t[o.type]||(i.console&&console.error&&console.error("laydate type error:'"+o.type+"' is not supported"),o.type="date"),o.format===t.date&&(o.format=t[o.type]||t.date),r.format=s.formatArr(o.format),o.weekStart&&!/^[0-6]$/.test(o.weekStart)&&(t=r.lang(),o.weekStart=t.weeks.indexOf(o.weekStart),-1===o.weekStart&&(o.weekStart=0)),r.EXP_IF="",r.EXP_SPLIT="",lay.each(r.format,function(e,t){e=new RegExp(u).test(t)?"\\d{"+(new RegExp(u).test(r.format[0===e?e+1:e-1]||"")?/^yyyy|y$/.test(t)?4:t.length:/^yyyy$/.test(t)?"1,4":/^y$/.test(t)?"1,308":"1,2")+"}":"\\"+t;r.EXP_IF=r.EXP_IF+e,r.EXP_SPLIT=r.EXP_SPLIT+"("+e+")"}),r.EXP_IF_ONE=new RegExp("^"+r.EXP_IF+"$"),r.EXP_IF=new RegExp("^"+(o.range?r.EXP_IF+"\\s\\"+r.rangeStr+"\\s"+r.EXP_IF:r.EXP_IF)+"$"),r.EXP_SPLIT=new RegExp("^"+r.EXP_SPLIT+"$",""),r.isInput(o.elem[0])||"focus"===o.trigger&&(o.trigger="click"),o.elem.attr("lay-key",r.index),o.eventElem.attr("lay-key",r.index),o.elem.attr(d,o.id),o.mark=lay.extend({},o.calendar&&"cn"===o.lang?{"0-1-1":"\u5143\u65e6","0-2-14":"\u60c5\u4eba","0-3-8":"\u5987\u5973","0-3-12":"\u690d\u6811","0-4-1":"\u611a\u4eba","0-5-1":"\u52b3\u52a8","0-5-4":"\u9752\u5e74","0-6-1":"\u513f\u7ae5","0-9-10":"\u6559\u5e08","0-10-1":"\u56fd\u5e86","0-12-25":"\u5723\u8bde"}:{},o.mark),lay.each(["min","max"],function(e,t){var a=[],n=[];if("number"==typeof o[t])var i=o[t],l=new Date,l=r.newDate({year:l.getFullYear(),month:l.getMonth(),date:l.getDate(),hours:e?23:0,minutes:e?59:0,seconds:e?59:0}).getTime(),e=new Date(i?i<864e5?l+864e5*i:i:l),a=[e.getFullYear(),e.getMonth()+1,e.getDate()],n=[e.getHours(),e.getMinutes(),e.getSeconds()];else if("string"==typeof o[t])a=(o[t].match(/\d+-\d+-\d+/)||[""])[0].split("-"),n=(o[t].match(/\d+:\d+:\d+/)||[""])[0].split(":");else if("object"==typeof o[t])return o[t];o[t]={year:0|a[0]||(new Date).getFullYear(),month:a[1]?(0|a[1])-1:(new Date).getMonth(),date:0|a[2]||(new Date).getDate(),hours:0|n[0],minutes:0|n[1],seconds:0|n[2]}}),r.elemID="layui-laydate"+o.elem.attr("lay-key"),(o.show||e)&&r.render(),e||r.events(),o.value&&o.isInitValue&&("date"===layui.type(o.value)?r.setValue(r.parse(0,r.systemDate(o.value))):r.setValue(o.value)))},I.prototype.render=function(){var a,n,i,l,r=this,o=r.config,d=r.lang(),s="static"===o.position,y=r.elem=lay.elem("div",{id:r.elemID,"class":["layui-laydate",o.range?" layui-laydate-range":"",r.rangeLinked?" layui-laydate-linkage":"",s?" "+D:"",o.fullPanel?" laydate-theme-fullpanel":"",(a="",lay.each(o.theme,function(e,t){"default"===t||/^#/.test(t)||(a+=" laydate-theme-"+t)}),a)].join("")}),m=r.elemMain=[],u=r.elemHeader=[],c=r.elemCont=[],h=r.table=[],e=r.footer=lay.elem("div",{"class":"layui-laydate-footer"}),t=r.shortcut=lay.elem("ul",{"class":"layui-laydate-shortcut"}),f=(o.zIndex&&(y.style.zIndex=o.zIndex),lay.each(new Array(2),function(e){if(!o.range&&0
"+m.time[t]+"