Laravel 构建一个AES密码登录的接口,并在iOS下对接

iOS应用做登录的时候,密码还是采用密文来传输比较可靠,接下来描述一下需要做好这件事情,前后端各自应该做那些事情。
这里的加密解密采用Laravel自带的encrypt与decrypt方法,由于加密过程在iOS端实现,解密过程在服务端实现,所以我们要先了解Laravel的加解密过程,才好将iOS内的加密逻辑写好。

一、了解加解密的过程

先看加密流程,Laravel中的encrypt方法在项目目录下的Vendor\Laravel\Framework\Src\Illuminate\Encryption下的Encrypter.php与EncryptionServiceProvider.php来构成。

先看下EncryptionServiceProvider.php做了什么

EncryptionServiceProvider.php中的方法主要是作为一个验证器来给Encrypter.php注入本次加密的加密规则以及加密的秘钥。

//EncryptionServiceProvider.php

public function register()
    {
        //这里是读取config/app.php下的配置文件内容
        $this->app->singleton('encrypter', function ($app) {
            $config = $app->make('config')->get('app');

           //这里首先获取了配置文件中的key,判断一下key是否由'base64:'开头
            if (Str::startsWith($key = $this->key($config), 'base64:')) {
                //如果是'base64:'开头的话,会去掉'base64:',再进行base64_decode来获取原始的key
                $key = base64_decode(substr($key, 7));
            }
            return new Encrypter($key, $config['cipher']);
        });
    }

我这里配置的key是用artisan命令生成的,很方便,然后加密方法使用默认的'AES-256-CBC'方式加密

要注意的是,使用artisan命令生成的key值设置好后,最好在别处备份一份,一但被更改,很麻烦。

再来看看Encrypter.php中的加密方法实现过程吧

先看构造方法

//Encrypter.php

public function __construct($key, $cipher = 'AES-128-CBC')
    {
        //先将key转化为字符串
        $key = (string) $key;
        //判断加密的方式与秘钥是否都符合要求,不符合的情况下会抛出异常
        if (static::supported($key, $cipher)) {
            $this->key = $key;
            $this->cipher = $cipher;
        } else {
            throw new RuntimeException('The only supported ciphers are AES-128-CBC and AES-256-CBC with the correct key lengths.');
        }
    }

这里面调用了'supported'方法,是用来判断key的长度是否符合加密方式的要求的

 public static function supported($key, $cipher)
    {
        //这里采用8bit来计算长度,保证在任何系统下计算都是一样的
        $length = mb_strlen($key, '8bit');
         //查看秘钥长度是否与加密方式匹配
        return ($cipher === 'AES-128-CBC' && $length === 16) ||
               ($cipher === 'AES-256-CBC' && $length === 32);
    }

接下来就是encrypt的本体方法了

//

public function encrypt($value, $serialize = true)
    {
        //根据秘钥的长度随机生成一个向量值
        $iv = random_bytes(openssl_cipher_iv_length($this->cipher));
        //这里的openssl_encrypt加密方式有4个参数
        /*
          *openssl_encrypt($data, $method, $password, $options = 0, $iv = "")
          *$data:要加密的数据
          *$method:加密方式,这里是使用 'AES-256-CBC'加密
          *$password:加密使用的key值
          *$options:加密后返回的值是否需要base64编码,默认为0是返回经过base64编码的数据
          *$iv :随机向量
         */
        $value = \openssl_encrypt(
            $serialize ? serialize($value) : $value,
            $this->cipher, $this->key, 0, $iv
        );
        //加密不成功后抛出异常
        if ($value === false) {
            throw new EncryptException('Could not encrypt the data.');
        }
        //对随机向量和密文进行签名
        $mac = $this->hash($iv = base64_encode($iv), $value);
        //将随机向量、密文、签名生成一个数组,转为Json格式
        $json = json_encode(compact('iv', 'value', 'mac'));
        
        if (json_last_error() !== JSON_ERROR_NONE) {
            throw new EncryptException('Could not encrypt the data.');
        }
        //把json格式转换为base64位,用于传输
        return base64_encode($json);
    }

下面梳理一下加密的过程:
1.随机生成一个向量值
2.使用Openssl的加密方式加密,并将加密结果经过base64编码,形成密文返回(因为应用场景为加密密码,而且是跨平台加密解密,所以用不上序列化加密,会使用Crypt门面提供的encryptString方法以及decryptString方法,其实就是帮你把encrypt与decrypt的第二个参数写成了false而已emmmmm)
3.将向量进行base64编码后与密文组合,与Key进行SHA256签名
4.将向量、密文、签名组成数组,且将数组转为Json格式
5.将Json转为base64,返回。

二、封装iOS端的加密算法

1.首先建立一个NSString的加密分类,写一个AES-256-CBC的加密方法

//NSString+AES.m
size_t const kKeySize = kCCKeySizeAES256;
NSString *const kInitVector = @"16-Bytes--String";


// key跟后台协商一个即可,保持一致
NSString *const PSW_AES_KEY = @"YW14cFRXMVdhazVFVm0xYVJHZDVXa1JvYWs0eVZURT0=";
// 偏移向量
NSString *const AES_IV_PARAMETER = @"RGd5WkRoak4yVTE=";


- (NSData *)AES128operation:(CCOperation)operation data:(NSData *)data key:(NSString *)key iv:(NSString *)iv {
    
    char keyPtr[kKeySize + 1];  //kCCKeySizeAES128是加密位数 可以替换成256位的
    bzero(keyPtr, sizeof(keyPtr));
    [key getCString:keyPtr maxLength:sizeof(keyPtr) encoding:NSUTF8StringEncoding];
    // IV
    char ivPtr[kKeySize + 1];
    bzero(ivPtr, sizeof(ivPtr));
    [iv getCString:ivPtr maxLength:sizeof(ivPtr) encoding:NSUTF8StringEncoding];
    
    size_t bufferSize = [data length] + kCCBlockSizeAES128;
    void *buffer = malloc(bufferSize);
    size_t numBytesEncrypted = 0;
    
    // 设置加密参数
    /**
     这里设置的参数ios默认为CBC加密方式,如果需要其他加密方式如ECB,在kCCOptionPKCS7Padding这个参数后边加上kCCOptionECBMode,即kCCOptionPKCS7Padding | kCCOptionECBMode,但是记得修改上边的偏移量,因为只有CBC模式有偏移量之说
     
     */
    CCCryptorStatus cryptorStatus = CCCrypt(operation,kCCAlgorithmAES , kCCOptionPKCS7Padding,
                                            keyPtr, kKeySize,
                                            ivPtr,
                                            [data bytes], [data length],
                                            buffer, bufferSize,
                                            &numBytesEncrypted);
    
    if(cryptorStatus == kCCSuccess) {
        NSLog(@"Success");
        return [NSData dataWithBytesNoCopy:buffer length:numBytesEncrypted];
        
    } else {
        NSLog(@"Error");
    }
    
    free(buffer);
    return nil;
}

然后还要加上Base64的编码解码方法:

- (NSString *)encodeBase64String{
    //先将string转换成data
    NSData *data = [self dataUsingEncoding:NSUTF8StringEncoding];
    
    NSData *base64Data = [data base64EncodedDataWithOptions:0];
    
    NSString *baseString = [[NSString alloc]initWithData:base64Data encoding:NSUTF8StringEncoding];
    
    return baseString;
}

- (NSString *)dencodeBase64String{
    
    NSData *data = [[NSData alloc]initWithBase64EncodedString:self options:NSDataBase64DecodingIgnoreUnknownCharacters];
    
    NSString *string = [[NSString alloc]initWithData:data encoding:NSUTF8StringEncoding];
    
    return string;
}

+ (NSString *)encodeBase64String:(NSString *)string{
    return  [string encodeBase64String];
}

+ (NSString *)dencodeBase64String:(NSString *)base64String{
    return [base64String dencodeBase64String];
}

最后写一个HASH加密的方法,这里是使用了SHA256加密,并且输出小写字母十六进制位

+ (NSString *)hmac:(NSString *)plaintext withKey:(NSString *)key{
    const char *cKey  = [key cStringUsingEncoding:NSASCIIStringEncoding];
    const char *cData = [plaintext cStringUsingEncoding:NSASCIIStringEncoding];
    unsigned char cHMAC[CC_SHA256_DIGEST_LENGTH];
    CCHmac(kCCHmacAlgSHA256, cKey, strlen(cKey), cData, strlen(cData), cHMAC);
    NSData *HMACData = [NSData dataWithBytes:cHMAC length:sizeof(cHMAC)];
    const unsigned char *buffer = (const unsigned char *)[HMACData bytes];
    NSMutableString *HMAC = [NSMutableString stringWithCapacity:HMACData.length * 2];
    
    NSMutableString *test = [NSMutableString stringWithCapacity:HMACData.length * 2];
    for (int i = 0; i < HMACData.length; ++i){
        [HMAC appendFormat:@"%02x", buffer[i]];
        [test appendFormat:@"%d",buffer[i]];
    }
    
    return HMAC;
}

2.主要的方法配置好了,接下来按照上面总结的Laravel加密规则来写就行了

- (NSString *)laravelAES256CBCPassword{
    //对照Laravel中的encrypt方式进行加密签名
    //设置向量
    NSString *iv = AES_IV_PARAMETER;
    
    //1.对明文进行AES加密,得到Base64后的密文
    NSString *aesPassword = [self aci_encryptWithAES];
    //2.对向量、进行Base64编码
    NSString *base64Iv = [iv encodeBase64String];
    //3.对base64向量与密文进行hash签名
    NSString *mac = [NSString hmac:[base64Iv stringByAppendingString:aesPassword] withKey:[NSString dencodeBase64String:PSW_AES_KEY]];
    
    //4.将Base64后的向量、密文、签名组成字典
    NSDictionary *dict = @{@"iv":base64Iv,@"value":aesPassword,@"mac":mac};
    //5.将字典进行Json格式化
    NSString *jsonDict = [dict mj_JSONString];
    //6.将Json进行Base64
    NSString *base64JsonDict = [jsonDict encodeBase64String];
    
    return base64JsonDict;
}

到这里,iOS端的加密工作就做完了,使用时直接调用分类的“laravelAES256CBCPassword”方法就OK

三、构建密文登录的接口

到了这一步就很简单了,按部就班在验证控制器内写上一个解密、验证的方法即可

 /*
     * 密码登录
     * */
    public  function  pwdLogin(PwdLoginRequest $pwdLoginRequest){

        $password = Crypt::decrypt($pwdLoginRequest['password'],false);
        
        $phone = $pwdLoginRequest['phone'];

        if (Auth::attempt(['phone' => $phone, 'password' => $password])) {
            $user_info = Users::where('phone', $phone)->first()->toarray();
            return $this->success(compact('user_info'));
        }else{
            return $this->failed('用户名或密码错误');
        }
    }
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,222评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,455评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,720评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,568评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,696评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,879评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,028评论 3 409
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,773评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,220评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,550评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,697评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,360评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,002评论 3 315
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,782评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,010评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,433评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,587评论 2 350

推荐阅读更多精彩内容