微信小程序非开放接口开发,token使用

简介: 前言:之前介绍了些小程序用户登录获取服务器token。现在来介绍下用户拿到token后请求一些权限接口的时候怎么,我们服务器端应该如何处理。今天就用一个商城里非常常见的地址添加接口来举例。

前言:

之前介绍了些小程序用户登录获取服务器token。现在来介绍下用户拿到token后请求一些权限接口的时候怎么,我们服务器端应该如何处理。今天就用一个商城里非常常见的地址添加接口来举例。

目录:

  1. 思路图
  2. token的接收和使用
  3. 接受地址数据,过滤地址数据
  4. 判断是新增还是修改,保存数据
一:思路图
token的使用.png
二:token的接收和使用

我们就从客户端传来token,地址数据,开始说起。

  1. 首先作为非开放接口,我们第一步当然就是接收token,如果没有token,就直接抛出异常。
    那么,我们约定,token是在http请求中的headr传递过来。那我们首先取出token,然后到服务器缓存中查找是否有相对于的value值。如果没有说明token不存在或者过期了,抛出异常。

首先我们在我们的service层下的token服务中构建一个通用的获取token对应储存的值的数据方法。因为我们token对应的值中有好几个数据,有用户的id,微信的openid,session_key。我们需要哪个字段的值,就传入这个key就可以了。

    /**
     * 获取token对应的某个数据
     * @param $key
     */
    protected static function getCurrentTokenValue($key)
    {
        //取出token
        $token = request()->header('token');
        //到缓存中获取
        $value = Cache::get($token);
        //如果没有这条数据
        if (empty($value)) {
            //抛出异常
            throw new TokenException();
        }  
    }
  1. 如果缓存中有值,我们因为在存储的时候是将数组序列话存储的。那么我这里就将数据反序列化一下。当然,我还加入了一点容错机制,因为我是直接使用tp5提供的文件缓存,所以只能存储字符串。如果今后我换了redis或者别的缓存驱动,可以存储数组或者对象什么的。就不用序列化了。所以我在反序列化之前,先判断下是不是数组,如果不是数组那就反序列化
    protected static function getCurrentTokenValue($key)
    {
        //取出token
        $token = request()->header('token');
        $value = Cache::get($token);
        if (empty($value)) {
            throw new TokenException();
        }
        //判断是否是以数组形式储存的
        if (!is_array($value)) {
            $value = unserialize($value);
        }
    }
  1. 最后一步就很简单了,根据传入的key值,去找$value中有没有,如果有就返回,如果没有就抛出异常
    下面是完整代码
    /**
     * 获取token对应的某个数据
     * @param $key
     * @return mixed
     * @throws TokenException
     * @throws Exception
     */
    protected static function getCurrentTokenValue($key)
    {
        //取出token
        $token = request()->header('token');
        $value = Cache::get($token);
        if (empty($value)) {
            throw new TokenException();
        }
        //判断是否是以数组形式储存的
        if (!is_array($value)) {
            $value = unserialize($value);
        }
        if (array_key_exists($key, $value)) {
            return $value[$key];
        } else {
            throw new Exception('尝试请求的参数并不存在');
        }
    }

好了 写好这个通用的获取token对应的值的方法之后呢,我们就需要想想我们到底需要这个token中的什么数据呢?既然是地址添加接口,那么就一定绕不开用户id,不然怎么知道是谁的地址呢?那么我们就需要写一个获取当前用户id的方法

    /**
     * 获取当前用户的id
     * @return mixed
     */
    public static function getCurrentId()
    {
        $id = self::getCurrentTokenValue('id');
        return $id;
    }

方法非常简单啊,就是之前的通用方法的具体实现。这样我们在控制器中使用这个getCurrenId方法就可以拿到用户的id了

三:接受地址数据,过滤地址数据

这里的接受数据过滤数据,其实主要是关于安全方面的考虑。因为我们要将用户发来的数据直接存入数据库是非常危险的。现在我们将接收数据,验证数据,过滤数据。总结到一起。方便今后使用。这里介绍的也是一个非常通用的方法。
大家一定知道在接收数据后我们都会使用验证器来验证这些数据。

AddressValidate::instance()->goCheck();

就像这样,就是我之前写过的独立验证器。验证如果不符合规则就会抛出异常。那么验证通过的数据就一定安全吗?我们现在来看看我的验证规则


验证规则

因为用户id我们从token对应的值中取了,所以我们不会从客户端传递的数据中获取用户id也不会信任用户传递的数据的。

那么我如果验证通过我就去取得用户传递的所有数据这样安全吗?不安全。一个不起眼的小细节就在这里。首先我们的确不从客户端取用户id,也不取验证用户id。但是我们不能保证客户端不会传用户id呢。如果他传了所有的地址数据,通过验证,还多传了一个用户id怎么办。所以我们不能全盘接收用户传递的数据。可是我也不想一个字段一个字段的去接收。那么就要写个通用的方法

这个通用的方法我就写到验证器的基类里中的goCheck方法中,因为每次接收用户数据,都会去验证的,也就会去调用goCheck方法,那么我们就写在这里面让它功能更强大.

先说下这个过滤的思路。我们在写验证器的时候是根据我们要接收哪些字段我们就会验证哪些字段对吧。那么我们要过滤出来的字段也就是我们验证规则数组中的key,不知道大家注意到了没有


验证规则中的key

那么我们只需将$rule中的key值作为过滤依据,验证规则中有的我们就保留,验证规则之外的不保留

    /**
     * 根据验证器来获取客户端传递的信息
     * @param $arrays 传入接收的客户端所有数据
     * @return array
     * @throws Exception
     */
    protected function getDataByRule($arrays)
    {
        //如果传递过来的数据中包含user_id这个字段的话,那就十有八九这人是个黑帽子了
        if (array_key_exists('user_id', $arrays)) {
            throw new Exception('本接口不支持你这个user_id的参数,别想了兄弟');
        }
        $newArr = [];
        //遍历验证规则数组
        foreach ($this->rule as $key => $value) {
            //规则中有的key值,对应客户端的所有数据中的key值。保存在$newArrr中
            $newArr[$key] = $arrays[$key];
        }
        return $newArr;
    }

这个方法写好之后,我么说过要将它写到goCheck方法中,让验证数据的同时,过滤了数据。我们就在验证成功后调用过滤方法。不清楚这个验证器的使用的可以回顾一下

    /**
     * 获取传递参数,并验证
     * @return bool|array
     * @throws ParameterException
     */
    public function goCheck()
    {
        //接收参数
        $request = Request::instance();
        //通过param方法获取到所有的参数
        $params = $request->param();
        //由哪个对象来调用goCheck方法,就是由哪个对象来调用check方法,将接收的所有参数传递进去
        $result = $this->batch()->check($params);
        if (!$result) {
            //如果结果为false,调用getError方法获取错误信息
            $error = $this->getError();
            //抛出参数错误异常
            throw new ParameterException(['msg' => $error]);
        } else {
            //调用获取过滤参数的方法,返回给控制器
            return $this->getDataByRule($params);
        }
    }

那么现在,只需要在控制器中拿个变量接收下goCheck方法的返回值就行了

$data = AddressValidate::instance()->goCheck();
四:判断是新增还是修改,保存数据

由于新增地址和修改地址的逻辑很类似 完全可以写在一个接口中。
之前通过token获取到对应的用户id,我们还需要为程序的健壮性,再去查询下这个用户是否是存在,是不是被我们删除了或者怎么了。
所有在控制器中调用模型上的getUserById方法

  //验证用户是否存在
  $user = User::getUserById($id);

模型方法也很简单

    /**
     * 通过用户id,查询用户
     * @param $id
     * @return bool|null|static
     */
    public static function getUserById($id)
    {
        $result = self::get($id);
        if (empty($result)) {
            return false;
        } else {
            //用户存在,返回数据模型对象
            return $result;
        }
    }

顺便在User模型中将地址UserAddress关联起来(关联就不详细介绍了。有时间单独写)

    //一对一关联,外键在外
   public function address()
    {
        return $this->hasOne('UserAddress','user_id','id');
    }

好了,现在我们通过判断用户是否存在顺便也取到了用户的数据模型对象。也关联好了地址表
下一步我们就判断地址是否存在,然后保存数据就好了

        //判断用户是新增还是修改
        $address = $user->address();
        //返回关联的对象hasOne对象
        if (empty($address)) {
            //调用hasOne对象上的保存(新增)当前关联数据对象方法
            $result = $address->save($data);
        } else {
            //address属性中保存的是UserAddress数据对象,是model的子类。调用model上的save方法
            $result = $user->address->save($data);
        }
        //如果保存成功
        if (!empty($result)) {
            //返回一个操作成功的对象,对象里包含成功的信息
            return new SuccessMessage();
        } else {
            //抛出异常
            throw new Exception('保存地址失败');
        }
细节

值得注意的是这里,新增和修改的save方法是不同的。我也将$user->address()和$user->address打印出来看了。带括号的是hasOne对象也就我们关联的时候返回的那个对象。不带括号的是model对象,就是我创建的UserAddress类的对象。

我的理解是,当关联地址表中没有这个用户的数据,返回的hasOne中的data属性就会是个空数组。来判断数据是否存在。我们就可以调用关联对象也就是(hasOne)对象上的save方法。这里是创建一个关联上user的数据对象的意思。
当返回的关联对象含有数据,那么使用user模型上的address属性,address属性上存储的其实就是UserAddress模型对象,我们打印出来也印证了这个观点。那么既然是模型对象model的子类,那么直接使用model中最常用的save方法,将数据存入,就可以完成修改了。

这里比较绕哈,虽然可以用模型直接新增,传入用户id的方法完成这个业务,但是这个方法我之前没有怎么用过,还是给大家介绍下一个新的思路。

下面我附上完整的控制器代码

   /**
     * 添加或者更新地址
     * @url http://local.jxshop.com/api/v1/address/add
     * @url http://local.jxshop.com/api/v1/address/update
     * @http GET
     *
     */
    public function createOrUpdateAddress()
    {
        $data = AddressValidate::instance()->goCheck();
        //验证token真实性
        $id = TokenService::getCurrentId();
        //验证用户是否存在
        $user = User::getUserById($id);
        if (!$user) {
            throw new UserException();
        }
        //判断用户是新增还是修改
        $address = $user->address();
        //返回关联的对象hasOne对象
        if (empty($address)) {
            //调用hasOne对象上的保存(新增)当前关联数据对象方法
            $result = $address->save($data);
        } else {
            //address属性中保存的是UserAddress数据对象,是model的子类。调用model上的save方法
            $result = $user->address->save($data);
        }
        //如果保存成功
        if (!empty($result)) {
            //返回一个操作成功的对象,对象里包含成功的信息
            return new SuccessMessage();
        } else {
            //抛出异常
            throw new Exception('保存地址失败');
        }
    }

那么今天的非开放接口就介绍到这里了。有哪些不对的地方,希望大神能够指正,我共同学习。

以上

相关文章
预约按摩小程序开发,为什么很多上门按摩平台根本招聘不到优秀技师?
上门按摩平台面临招不到优秀技师的问题,主要原因是平台众多,技师选择多样。为解决此问题,平台可引入技师等级制度,根据订单数量和好评率划分高、低等级技师。高等级技师可享受70%-90%的高提成及首页推荐,这不仅能激励技师的积极性,还能帮助平台筛选出优质技师,提升服务质量和口碑,形成良性循环。
|
25天前
|
人工智能 小程序
【一步步开发AI运动小程序】十五、AI运动识别中,如何判断人体站位的远近?
【云智AI运动识别小程序插件】提供人体、运动及姿态检测的AI能力,无需后台支持,具有快速、体验好、易集成等特点。本文介绍如何利用插件判断人体与摄像头的远近,确保人体图像在帧内的比例适中,以优化识别效果。通过`whole`检测规则,分别实现人体过近和过远的判断,并给出相应示例代码。
|
6天前
|
开发框架 小程序 前端开发
圈子社交app前端+后端源码,uniapp社交兴趣圈子开发,框架php圈子小程序安装搭建
本文介绍了圈子社交APP的源码获取、分析与定制,PHP实现的圈子框架设计及代码编写,以及圈子小程序的安装搭建。涵盖环境配置、数据库设计、前后端开发与接口对接等内容,确保平台的安全性、性能和功能完整性。通过详细指导,帮助开发者快速搭建稳定可靠的圈子社交平台。
66 17
|
24天前
|
人工智能 小程序 UED
【一步步开发AI运动小程序】十六、AI运动识别中,如何判断人体站位?
【云智AI运动识别小程序插件】提供人体、运动及姿态检测的AI能力,本地引擎无需后台支持,具备快速、体验好、易集成等优势。本文介绍如何利用插件的`camera-view`功能,通过检测人体站位视角(前、后、左、右),确保运动时的最佳识别率和用户体验。代码示例展示了如何实现视角检查,确保用户正或背对摄像头,为后续运动检测打下基础。
|
19天前
|
人工智能 小程序 API
【一步步开发AI运动小程序】十七、如何识别用户上传视频中的人体、运动、动作、姿态?
【云智AI运动识别小程序插件】提供人体、运动、姿态检测的AI能力,支持本地原生识别,无需后台服务,具有速度快、体验好、易集成等优点。本文介绍如何使用该插件实现用户上传视频的运动识别,包括视频解码抽帧和人体识别的实现方法。
|
2月前
|
人工智能 小程序 IDE
【一步步开发AI运动小程序】九、姿态辅助调试桌面工具的使用
随着AI技术的发展,各大厂商推出的AI运动APP如“乐动力”、“天天跳绳”等,使云上运动会、线上健身等概念大热。本文将指导你如何利用“云智AI运动识别小程序插件”,在微信小程序中实现类似功能,包括工具搭建、服务启动及数据回传等步骤,助力开发者高效开发AI运动小程序。
【一步步开发AI运动小程序】九、姿态辅助调试桌面工具的使用
|
30天前
|
人工智能 小程序 API
【一步步开发AI运动小程序】十三、自定义一个运动分析器,实现计时计数02
本文介绍如何利用“云智AI运动识别小程序插件”开发AI运动小程序,详细解析了俯卧撑动作的检测规则构建与执行流程,涵盖卧撑和撑卧两个姿态的识别规则,以及如何通过继承`sports.SportBase`类实现运动分析器,适用于小程序开发者。
|
30天前
|
人工智能 小程序 API
【一步步开发AI运动小程序】十二、自定义一个运动分析器,实现计时计数01
随着AI技术的发展,AI运动APP如雨后春笋般涌现,如“乐动力”、“天天跳绳”等,推动了云上运动会、线上健身等热潮。本文将指导你从零开始开发一个AI运动小程序,利用“云智AI运动识别小程序插件”,介绍运动识别原理、计量方式及运动分析器基类的使用,帮助你在小程序中实现运动计时和计数功能。下篇将继续探讨运动姿态检测规则的编写。
|
2月前
|
人工智能 JSON 小程序
【一步步开发AI运动小程序】八、利用body-calc进行姿态识别
随着AI技术的发展,各大厂商推出的如“乐动力”、“天天跳绳”等AI运动APP,使得云上运动会、线上健身等概念风靡一时。本文将引导读者从零开始开发一个AI运动小程序,使用“云智AI运动识别小程序插件”,介绍人体姿态检测规则、规则编写语法、执行检测规则运算及姿态检测实战等内容,助力开发者轻松实现AI运动小程序的开发。
【一步步开发AI运动小程序】八、利用body-calc进行姿态识别
|
20天前
|
人工智能 小程序 数据处理
uni-app开发AI康复锻炼小程序,帮助肢体受伤患者康复!
近期,多家康复机构咨询AI运动识别插件是否适用于肢力运动受限患者的康复锻炼。本文介绍该插件在康复锻炼中的应用场景,包括康复运动指导、运动记录、恢复程度记录及过程监测。插件集成了人体检测、姿态识别等功能,支持微信小程序平台,使用便捷,安全可靠,帮助康复治疗更加高效精准。