给你的Qt软件加个授权

20 篇文章 3 订阅

写在前面

环境:

Win11 64位

VS2019 + Qt5.15.2

核心思路:

将授权相关信息加密保存到License.txt中,软件运行时获取并解密授权信息,判断是否在限制期限内即可。

加解密部分使用第三方openssl库进行,因此需要手动在项目中链接下openssl库,参考步骤如下。

链接openssl库

①官网下载openssl库安装包
官网链接:https://slproweb.com/products/Win32OpenSSL.html

下载自己当前操作系统位数对应的安装包即可,我当前系统为Win11 64位,因此下载以下安装包:
1

②双击安装。注意勾选添加到系统环境变量:
2
③打开安装目录如下:
3

④拷贝include文件夹到项目路径下:
4
⑤拷贝当前项目使用运行库对应的lib到项目路径下,这里使用MD Release版本中的动态库:
5
不会全部用到,可按需选择使用。

例如只用到以下三个库:
6
⑥在项目中链接这三个库,包含相应头文件即可使用:
7

示例

//AuthorizationManager.h
#pragma once
#include <cstring>
#include <fstream>
#include <iostream>
#include <string>
#include <sstream>

struct MyLicense
{
    int nAuthorizationStatus{ 1 };    //0: 未授权 1: 已授权
    std::string sFirstRunDate;    //首次运行日期
    int nAuthorizationDays{ -1 };        //授权天数

    std::string Serialize()
    {
        std::ostringstream os;
        os << nAuthorizationStatus << "\n" << sFirstRunDate << "\n" << nAuthorizationDays << "\n";
        return os.str();
    }

    void Deserialize(const std::string& str)
    {
        std::istringstream is(str);
        is >> nAuthorizationStatus;
        is.ignore();
        std::getline(is, sFirstRunDate);
        sFirstRunDate = sFirstRunDate.substr(0, sFirstRunDate.length());
        is >> nAuthorizationDays;
    }
};

class AuthorizationManager
{
public:
    static AuthorizationManager& Instance()
    {
        static AuthorizationManager instance;
        return instance;
    }

    AuthorizationManager(const AuthorizationManager&) = delete;
    AuthorizationManager& operator=(const AuthorizationManager&) = delete;

    std::string do_encrypt(const std::string& plaintext, const std::string& key, const std::string& iv);
    std::string do_decrypt(const std::string& ciphertext, const std::string& key, const std::string& iv);

    bool AuthorizationVerify();

private:
    AuthorizationManager() = default;

};
#include "AuthorizationManager.h"
#include <openssl/evp.h>
#include <openssl/aes.h>
#include <vector>
#include <QDate>

std::string AuthorizationManager::do_encrypt(const std::string& plaintext, const std::string& key, const std::string& iv)
{
	EVP_CIPHER_CTX* ctx;
	std::vector<unsigned char> ciphertext(plaintext.size() + EVP_MAX_BLOCK_LENGTH);
	int len;
	int ciphertext_len;

	if (!(ctx = EVP_CIPHER_CTX_new()))
	{
		return "";
	}

	if (1 != EVP_EncryptInit_ex(ctx, EVP_aes_256_cbc(), NULL, (const unsigned char*)key.c_str(), (const unsigned char*)iv.c_str()))
	{
		return "";
	}

	if (1 != EVP_EncryptUpdate(ctx, ciphertext.data(), &len, (const unsigned char*)plaintext.c_str(), plaintext.size()))
	{
		return "";
	}
	ciphertext_len = len;

	if (1 != EVP_EncryptFinal_ex(ctx, ciphertext.data() + len, &len))
	{
		return "";
	}
	ciphertext_len += len;

	EVP_CIPHER_CTX_free(ctx);

	return std::string((char*)ciphertext.data(), ciphertext_len);
}

std::string AuthorizationManager::do_decrypt(const std::string& ciphertext, const std::string& key, const std::string& iv)
{
	EVP_CIPHER_CTX* ctx;
	std::vector<unsigned char> plaintext(ciphertext.size());
	int len;
	int plaintext_len;

	if (!(ctx = EVP_CIPHER_CTX_new()))
	{
		return "";
	}

	if (1 != EVP_DecryptInit_ex(ctx, EVP_aes_256_cbc(), NULL, (const unsigned char*)key.c_str(), (const unsigned char*)iv.c_str()))
	{
		return "";
	}

	if (1 != EVP_DecryptUpdate(ctx, plaintext.data(), &len, (const unsigned char*)ciphertext.c_str(), ciphertext.size()))
	{
		return "";
	}
	plaintext_len = len;

	if (1 != EVP_DecryptFinal_ex(ctx, plaintext.data() + len, &len))
	{
		return "";
	}
	plaintext_len += len;

	EVP_CIPHER_CTX_free(ctx);

	return std::string((char*)plaintext.data(), plaintext_len);
}

bool AuthorizationManager::AuthorizationVerify()
{
	//密钥对,妥善保管
	std::string key = "98765432100123456789987654321001";
	std::string iv = "9876543210012345";

	//构造授权
	//MyLicense original;
	//original.nAuthorizationStatus = 1;
	//original.sFirstRunDate = "2024-04-08";
	//original.nAuthorizationDays = 90;
	//std::string serialized_str = original.Serialize();
	//std::string ciphertext = AuthorizationManager::Instance().do_encrypt(serialized_str, key, iv);
	//std::ofstream outfile("License.txt", std::ios::binary);
	//outfile << ciphertext;
	//outfile.close();

	//获取授权
	std::ifstream infile("License.txt", std::ios::binary);
	std::stringstream ss;
	ss << infile.rdbuf();
	std::string content = ss.str();
	std::string decryptedtext = AuthorizationManager::Instance().do_decrypt(content, key, iv);
	MyLicense restored;
	restored.Deserialize(decryptedtext);

	//授权校验
	bool bRet = false;
	if (restored.nAuthorizationStatus == 0)
	{
		//首次运行授权
		MyLicense original;
		original.nAuthorizationStatus = 1;

		QDate curDate = QDate::currentDate();
		std::string sCurDate = curDate.toString("yyyy-MM-dd").toStdString();
		original.sFirstRunDate = sCurDate;

		original.nAuthorizationDays = restored.nAuthorizationDays;
		std::string serialized_str = original.Serialize();
		std::string ciphertext = AuthorizationManager::Instance().do_encrypt(serialized_str, key, iv);
		std::ofstream outfile("License.txt", std::ios::binary);
		outfile << ciphertext;
		outfile.close();
		bRet = true;
	}
	else
	{
		QDate curDate = QDate::currentDate();
		QDate firstRunDate = QDate::fromString(QString::fromStdString(restored.sFirstRunDate), "yyyy-MM-dd");

		QDate endDate = firstRunDate.addDays(restored.nAuthorizationDays);

		if (curDate < firstRunDate)
		{
			bRet = false;
		}
		else if (curDate > endDate)
		{
			bRet = false;
		}
		else
		{
			bRet = true;
		}
	}

	return bRet;
}

//main.cpp
#include "AuthorizationWidget.h"
#include <QtWidgets/QApplication>
#include "AuthorizationManager.h"

int main(int argc, char* argv[])
{
	QApplication a(argc, argv);

	//授权检查
	if (!AuthorizationManager::Instance().AuthorizationVerify())
	{
		//未授权,退出软件
		return false;
	}

	AuthorizationWidget w;
	w.show();
	return a.exec();
}

  • 10
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
QT软件是一种跨平台的C++应用程序开发框架,它提供了一系列用户界面组件和工具,可以开发各种类型的桌面应用程序。权限管理是QT软件开发过程中的一个重要组成部分,它可以确保用户安全和数据保护。 在QT软件中,基本的用户权限分为三种:普通用户、管理员和超级管理员。普通用户只能使用系统中的一些基本的功能,不能修改、删除或添加数据。管理员可以对软件进行一些配置和修改操作,但不能对系统做出大的改变。而超级管理员拥有最高的权限,可以对系统的所有文件和数据进行操作,例如修改、删除、添加等。 QT软件中的用户权限管理可以通过多种方式来实现。一种常用的方法是使用Access Control Lists(ACLs)来管理用户对数据的访问。ACLs是一种常见的访问控制机制,可以对文件或目录设置不同的访问权限。QT软件也可以使用访问控制列表来限制用户的权限,例如限制用户在添加、删除或修改数据时需要管理员或超级管理员的授权。 除了ACLs,QT软件还可以使用访问控制矩阵来管理用户权限。访问控制矩阵是一个二维矩阵,可以将权限和用户之间的关系进行可视化,方便管理员和开发人员识别和管理不同用户的权限。 在QT软件开发过程中,用户权限管理是必要的,因为它可以确保软件的安全性和数据保护。通过使用访问控制机制和矩阵,可以简化管理员和开发人员对用户权限的管理。同时,QT软件还提供了一些额外的安全函数,如加密算法、安全通讯协议等,以加强软件的安全保护能力。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值