加密就不用说了,看上一篇2.X加密的方式,怎么弄都可以。的保证解密规则就行;

现在重点说3.X解密:

在新的3.X引擎中官方整合了大部分获取资源的方法,最终合成一个getdata;

可以从源码,和堆栈调用中看到:

CCFileUtils.cpp:

Data FileUtils::getDataFromFile(const std::string& filename)
{
return getData(filename, false);
}

getDataFromFile目前只调用getData(filename,false);

Data getData(const std::string& filename, bool forString)

这个函数是一个非类成员静态函数。

forString是用来标识是否是一个文本文件,如果是那么buffer需要多一个字节。

这个其实不重要,因为我们处理的最终buffer是获取完全的

所以直接改代码:

static Data getData(const std::string& filename, bool forString)
{
if (filename.empty())
{
return Data::Null;
} Data ret;
unsigned char* buffer = nullptr;
size_t size = ;
size_t readsize;
const char* mode = nullptr; if (forString)
mode = "rt";
else
mode = "rb"; std::string lastname = FileUtils::getInstance()->fullPathForFilename(filename);
lastname = lastname.substr(lastname.length()-5,lastname.length()); do
{
// Read the file from hardware
std::string fullPath = FileUtils::getInstance()->fullPathForFilename(filename);
FILE *fp = fopen(fullPath.c_str(), mode);
CC_BREAK_IF(!fp);
fseek(fp,,SEEK_END);
size = ftell(fp);
fseek(fp,,SEEK_SET); if (forString)
{
buffer = (unsigned char*)malloc(sizeof(unsigned char) * (size + ));
buffer[size] = '\0';
}
else
{
buffer = (unsigned char*)malloc(sizeof(unsigned char) * size);
} readsize = fread(buffer, sizeof(unsigned char), size, fp);
fclose(fp); if (forString && readsize < size)
{
buffer[readsize] = '\0';
}
} while (); if (nullptr == buffer || == readsize)
{
std::string msg = "Get data from file(";
msg.append(filename).append(") failed!");
CCLOG("%s", msg.c_str());
}
else
{
if(lastname == "_jm.d")
{
for (int i = 0; i<readsize; i++) {
buffer[i]=MD5(buffer[i]);
}
buffer[readsize]=buffer[readsize]-MD5size;
}
ret.fastSet(buffer, readsize);
} return ret;
}

红色代码是我们自定义加密的文件解密,不管你用什么加密,或者修改地址扰码,只要保证加密解密格式相同;

OK,解密就算完成了,但是注意还有资源类型需要判断,在2.X中,处理了EImageFormat的判断,可以定义资源类型

但是还是建议还是不要去大改源码,3.2以上版本已经非常简洁强大了

3.X中整合了Format后,也有资源类型另外一种方式定义_fileType = detectFormat(unpackedData, unpackedLen);

我们解密后的资源类型在3.X中是行不通的,是Format::UNKOWN;

那么就在不大改的情况下重载函数方式解决这个问题:

重载initWithImageData,在CCImage.h中CCImage.cpp中

.h添加:

/*
jmflag 加密标识
*/
bool initWithImageData(const unsigned char * data, ssize_t dataLen,bool jmflag);

.cpp添加:

bool Image::initWithImageData(const unsigned char * data, ssize_t dataLen,bool jmflag)
{
bool ret = false; do
{
CC_BREAK_IF(! data || dataLen <= ); unsigned char* unpackedData = nullptr;
ssize_t unpackedLen = ; //detecgt and unzip the compress file
if (ZipUtils::isCCZBuffer(data, dataLen))
{
unpackedLen = ZipUtils::inflateCCZBuffer(data, dataLen, &unpackedData);
}
else if (ZipUtils::isGZipBuffer(data, dataLen))
{
unpackedLen = ZipUtils::inflateMemory(const_cast<unsigned char*>(data), dataLen, &unpackedData);
}
else
{
unpackedData = const_cast<unsigned char*>(data);
unpackedLen = dataLen;
} if(jmflag == true)
{
_fileType=Format::PNG;
} ret = initWithPngData(unpackedData, unpackedLen); if(unpackedData != data)
{
free(unpackedData);
}
} while (); return ret;
}

另外在initWithImageFile函数中得修改调用方法

CCImage.cpp:

Data data = FileUtils::getInstance()->getDataFromFile(_filePath);

    if (!data.isNull())
{
std::string lastname = _filePath;
lastname = lastname.substr(lastname.length()-,lastname.length());
if(lastname=="_jm.d")
{
ret = initWithImageData(data.getBytes(), data.getSize(),true);
}
else
{
ret = initWithImageData(data.getBytes(), data.getSize());
}
}

OK,自定义类型的资源解密就搞定了,但这只处理了PNG图片类型的加密,其他图片类型资源可以按照这种方式处理。

好了,那么有同学问,那么JS,Lua脚本的解密呢?

看这里

ScirptingCore.cpp:

  // Check whether '.jsc' files exist to avoid outputing log which says 'couldn't find .jsc file'.
CCLOG("byteCodePath > %s",byteCodePath.c_str());
if (futil->isFileExist(byteCodePath))
{
Data data = futil->getDataFromFile(byteCodePath);
if (!data.isNull())
{
script = JS_DecodeScript(cx, data.getBytes(), static_cast<uint32_t>(data.getSize()), nullptr, nullptr);
}
}

Data data = futil->getDataFromFile(byteCodePath);

对于脚本语言的加载读取还是上面我们已经改过的getDataFromFile方法噢!

但是还有一点

script = JS_DecodeScript(cx, data.getBytes(), static_cast<uint32_t>(data.getSize()), nullptr, nullptr);

这样是直接拿不到script的,JS_DecodeScript只是处理.jsc的,那么怎么给script复制呢?

代码修改如下:

        //只解密scr下面目录文件
if(jmflag==true)
{
if (futil->isFileExist(jmfullPath))
{
Data data = futil->getDataFromFile(jmfullPath,true);
if (!data.isNull())
{
script = JS::Compile(cx, obj, options, (const char*)data.getBytes(), data.getSize());
}
}
}
else
{
script = JS::Compile(cx, obj, options, fullPath.c_str());
}

script = JS::Compile(cx, obj, options, (const char*)data.getBytes(), data.getSize());

用Compile的这个重载函数赋值就全部搞定了;Lua和JS的脚本代码解密也一样非常简单!

大功告成,再见!

PS一下广告时间:

我目前在录制cocos2dx-3.X系列的实战项目视频,可以完全来说是商业项目视频讲解,让你爱上更有趣的游戏开发方式;

9秒课堂会第一时间上线我们的实战视频!

可以让大家更快速的掌握cocos2dx-Js的游戏实战开发!现在用脚本做游戏已然成为一个大的趋势;

而且在手游游戏行业飞速发展的今天,如果不用一种快速上手的脚本语言做游戏开发岂不是会Out了

另外希望大家支持我们的游戏开发群:【41131516】

 
 
 
 
 
 
 
 

【Cocos2d-x 3.X 资源及脚本解密】的更多相关文章

  1. 小白读iOS冗余资源扫描脚本

    随着公司项目的不断功能迭代,项目的体积越来越大,各种瘦身策略迫在眉睫.由于平时使用Linux高级命令和 shell脚本的机会不多,之前学的知识一下子想起来很难.所有趁着这次看脚本,重新温习一下. 本文 ...

  2. iOS项目冗余资源扫描脚本

    iOS项目冗余资源扫描脚本 随着iOS项目的版本不断迭代,app中冗余文件会越来越多,app size也持续增加,是时候需要对app冗余资源进行检测,对app进行瘦身. 使用方法: 1. 运行环境为m ...

  3. cocos2dx资源和脚本加密quick-lua3.3final

    一.资源加密 版本号:Quick-Cocos2d-x 3.3 Final 调试工具:xCode 工程创建的时候选择的拷贝源码. 项目结构如图: 这个功能七月大神在很早之前就已经实现了,但是在3.3版本 ...

  4. 使用KRPano资源分析工具解密被加密的XML

    软件交流群:571171251(软件免费版本在群内提供) krpano技术交流群:551278936(软件免费版本在群内提供) 最新博客地址:blog.turenlong.com 限时下载地址:htt ...

  5. cocos2d&amp;cocos2dx学习资源

    汇总一下自己学习Cocos2d和cocos2dx认为比較好的一些资源: 书籍: <iPhone&iPad cocos2d游戏开发实战> Steffen Itterheim < ...

  6. 通过apache的mod_status 统计占资源的脚本

    apache的mod_status模块,提供了对apache运行时的一些统计信息,对apache的管理员来说很有意义. 一.加载apache的mod_status模块 各种系统下,加载apache模块 ...

  7. 提取Unity游戏资源和脚本

    UnityStudio UnityStudio可以直接在自己的软件上查看图片.shader.文本.还能直接播放音频.甚至还能看场景Hierarchy视图的树状结构.强烈推荐用UnityStudio. ...

  8. 【COCOS2DX-LUA 脚本开发之十二】Hybrid模式-利用AssetsManager实现在线更新脚本文件lua、js、图片等资源(免去平台审核周期)

    本站文章均为李华明Himi原创,转载务必在明显处注明:(作者新浪微博:@李华明Himi) 转载自[黑米GameDev街区] 原文链接: http://www.himigame.com/iphone-c ...

  9. 利用AssetsManager实现在线更新脚本文件lua、js、图片等资源(免去平台审核周期)

    转自:http://www.himigame.com/iphone-cocos2dx/1354.html 首先说明一个问题: 为什么要在线更新资源和脚本文件!? 对于此问题,那要说的太多了,简单概括, ...

随机推荐

  1. 使用 Python 进行稳定可靠的文件操作

    程序需要更新文件.虽然大部分程序员知道在执行I/O的时候会发生不可预期的事情,但是我经常看到一些异常幼稚的代码.在本文中,我想要分享一些如何在Python代码中改善I/O可靠性的见解. 考虑下述Pyt ...

  2. SQLite主键自增需要设置为integer PRIMARY KEY

    按照正常的SQL语句,创建一个数据表,并设置主键是这样的语句: ), EventType )) 但使用这种办法,在SQLite中创建的的数据表,如果使用Insert语句插入记录,如下语句: INSER ...

  3. 【Leetcode】【Medium】Palindrome Partitioning

    Given a string s, partition s such that every substring of the partition is a palindrome. Return all ...

  4. 大熊君说说JS与设计模式之------单例模式Singleton()

    一,总体概要 1,笔者浅谈 顾名思义单例模式并不难理解,是产生一个类的唯一实例,在我们实际开发中也会使用到这种模式,它属于创建模式的一种,基于JS语言本身的语法特征, 对象直接量“{}”,也可以作为单 ...

  5. JS自动化

    写在前面 记得当时刷笔试题的时候有了解过“前端工程化”,无非就是用自动化工具帮助开发人员完成一些小细节,提高工作效率之类的,不过当时可没想到可维护性这么远 构建项目 -> 模块化开发 -> ...

  6. CoreCLR中超过3万行代码的gc.cpp文件的来源

    在CoreCLR的开源代码中,GC的主要实现代码gc.cpp文件大小竟然有1.17MB,打开文件一看,竟然有35490行!第一次见到如此多行的单个代码文件. github都不让直接查看:https:/ ...

  7. Nim教程【三】

    这是国内第一个关于Nim的系列教程 (至少我百度和必应是没有找到类似的教程) 先说废话 有人说 Golang的编译器/工具链也学互联网行业跟风拿用户当测试,简直一点素质没有. 还有人说 Go社区的风气 ...

  8. GUI 快捷键的实现思路

      思路: 前提快捷键操作不可重复,即一个快捷键对应一个控件的动作 一个窗体保持一份快捷键的map映射 在相应的消息中获取快捷键列表如键盘消息 在控件类对象中定义一个默认的响应行为,比如Button按 ...

  9. Android开发学习总结——Android开发的一些相关概念

    一.什么是3G.4G 1995年问世的第一代模拟制式手机(1G)只能进行语音通话. 1996到1997年出现的第二代GSM.CDMA等数字制式手机(2G)便增加了接收数据的功能 Ÿ 3G指的是第三代移 ...

  10. jenkins2 pipeline高级

    jenkins2 pipeline里groovy的高级用法.翻译自:https://github.com/jenkinsci/pipeline-plugin/blob/master/TUTORIAL. ...