接到策划需求,需要将移动端开发的游戏移植到steam平台。奈何策划水平有点次,信誓旦旦的说什么steam支付是使用dlc支付的,使用什么dlc币(窝里个大草),马的,不懂就别瞎咧咧,净给老子挖坑。
查了一圈资料,在steam后台配置若干steam相关dlc相关,就是没有相关与dlc相关支付接口,这时候才意识到公司策划真不靠谱,还老是逼逼开发,搞得一肚子气,马上就摔键盘走人,但是这才是给我挖的第一个坑。
踩了坑之后发现,dlc配置完成之后,是在steam商店里面起作用的,不是在游戏商店起作用,和游戏内购完全没有关系,steam里面的内购又叫做小额支付,相关文档在这里
在接入小额支付时候,需要一个关键参数key,文档解释key是Steamworks Web API 发行商验证密钥。对的,你没猜错,这是他们给我挖的第二个坑。当时注明要发行商密钥,可是偏偏他们给的还是用户密钥,接口调试不通,让他们确认的时候,还是一口咬定给我的是发行商密钥,mmp想打人。
好的,抱怨完了,接下来我们开始接入sdk,steam的文档很怪,明明都是中文可就是看不懂,网上翻了一遍又一遍资料终于给摸索出来了。

服务端接入

下面是相关服务器接入相关代码(go开发的,大家可进行相关参考,感觉坑的是请求Header参数构造以及下订单时参数的构造,这些在steam文档里面是没有的)

const (
	SANDBOX_URL          = "https://partner.steam-api.com/ISteamMicroTxnSandbox/"
	PRODUCT_URL          = "https://partner.steam-api.com/ISteamMicroTxn/"
	KEY                  = "--------------------MY KEY----------------------"      //这里替换成自己的发行商密钥
)

type userInfoResp struct {
	Response respInfo `json:"response"`
}
type userInfoParams struct {
	State    string `json:"state"`
	Country  string `json:"country"`
	Currency string `json:"currency"`
	Status   string `json:"status"`
}
type respInfo struct {
	Result string         `json:"result"`
	Params userInfoParams `json:"params"`
}

type steamOrderInfo struct {
	Response steamOrderResponse `json:"response"`
}
type steamOrderParams struct {
	Orderid string `json:"orderid"`
	Transid string `json:"transid"`
}
type steamOrderResponse struct {
	Result   string           `json:"result"`
	Params   steamOrderParams `json:"params"`
	ErrorObj ErrorObj         `json:"error"`
}

type ErrorObj struct {
	ErrorCode string `json:"errorcode"`
	ErrorDesc string `json:"errordesc"`
}

func getUrl(isSandbox bool) string {
	if isSandbox {
		return SANDBOX_URL
	} else {
		return PRODUCT_URL
	}
}

func steamPostReq(httpUrl string, data io.Reader) (*http.Response, error) {
	req, err := http.NewRequest("POST", httpUrl, data)
	req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
	req.Header.Set("Accept", "application/json")
	req.Header.Set("Accept-Language", "en_US")
	client := &http.Client{}
	resp, err := client.Do(req)
	return resp, err
}

// 获取用户所需的相关国家地区等信息
func GetUserInfo(write http.ResponseWriter, request *http.Request) {
	body := utils.GetRequestMap(request, "getUserInfo")
	appid := body["appid"].(string)
	steamid := body["steamid"].(string)
	isSandbox := body["isSandbox"].(bool)
	log.Info("Steam UserInfo Body %v", body)
	userInfo := getSteamUserInfo(appid, steamid, isSandbox)
	log.Info("Steam UserInfo UserInfo %v", userInfo)
	utils.ResultSuccess(write, userInfo)
}

func getSteamUserInfo(appid, steamid string, isSandbox bool) *userInfoParams {
	url := getUrl(isSandbox) + "GetUserInfo/v2/?key=%s&appid=%s&steamid=%s"
	url = fmt.Sprintf(url, KEY, appid, steamid)
	client := &http.Client{}
	resp, err := client.Get(url)
	defer resp.Body.Close()
	body, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		log.Info("GetUserInfo err:%v", err)
		return nil
	}
	info := new(userInfoResp)
	err = json.Unmarshal(body, &info)
	if err != nil {
		log.Info("Get UserInfo Unmarshal err:", err)
		return nil
	}
	return &info.Response.Params
}

// InitTxn 用户请求下订单
func InitTxn(write http.ResponseWriter, request *http.Request) {
	txnInfo := utils.GetRequestMap(request, "initTxn")
	steamId, _ := txnInfo["steamid"].(string)
	appid, _ := txnInfo["appid"].(string)
	language := txnInfo["language"].(string)
	currency := txnInfo["currency"].(string)
	itemId, _ := txnInfo["itemid"].(string)
	amount, _ := txnInfo["amount"].(string)
	userId, _ := utils.CovToLong(txnInfo["userId"])
	description := txnInfo["description"].(string)
	userIdStr := strconv.FormatInt(userId, 10)
	orderId := userIdStr[len(userIdStr)-6:] + strconv.FormatInt(time.Now().Unix(), 10)
	isSandbox := txnInfo["isSandbox"].(bool)
	httpUrl := getUrl(isSandbox) + "InitTxn/v3/"

	dataStr := fmt.Sprintf(
		"key=%v"+
			"&orderid=%v"+
			"&steamid=%v"+
			"&appid=%v"+
			"&itemcount=%v"+
			"&language=%v"+
			"¤cy=%v"+
			"&itemid[0]=%v"+
			"&qty[0]=%v"+
			"&amount[0]=%v"+
			"&description[0]=%v"+
			"&usersession=%v",
		KEY, orderId, steamId, appid, 1, language, currency, itemId, "1", amount, description, "client")
	data := strings.NewReader(dataStr)
	resp, err := steamPostReq(httpUrl, data)
	defer resp.Body.Close()
	content, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		log.Info("Steam Purchase Fail:%v", err.Error())
	}
	steamOrderInfo := new(steamOrderInfo)
	err = json.Unmarshal(content, &steamOrderInfo)
	log.Info("Steam Purchase back:%v", string(content))
	utils.ResultSuccess(write, steamOrderInfo)
}

// FinalizeTxn 验证订单
func FinalizeTxn(write http.ResponseWriter, request *http.Request) {
	txnInfo := utils.GetRequestMap(request, "finalizeTxn")
	isSandbox := txnInfo["isSandbox"].(bool)
	orderid := txnInfo["orderid"].(string)
	appid := txnInfo["appid"].(string)
	httpUrl := getUrl(isSandbox) + "FinalizeTxn/v2/"
	dataStr := fmt.Sprintf("key=%v"+
		"&orderid=%v"+
		"&appid=%v", KEY, orderid, appid)
	data := strings.NewReader(dataStr)
	resp, err := steamPostReq(httpUrl, data)
	defer resp.Body.Close()
	content, err := ioutil.ReadAll(resp.Body)
	if err != nil {
		log.Info("Steam Purchase Confirm back Fail:%v", err.Error())
	}
	log.Info("Steam Purchase Confirm back:%v", string(content))
	steamOrderInfo := new(steamOrderInfo)
	err = json.Unmarshal(content, &steamOrderInfo)
	if err != nil {
		log.Error("Steam Purchase Confirm back response error :%v", err)
		return
	}
        //TODO 执行相关入库操作
        // ...
	utils.ResultSuccess(write, steamOrderInfo)
}

客户端的接入

客户端执行步骤Window>PackageManager>点击+>Add package from git URL,然后填入相关地址Add即可(这个地址可能版本有变动,具体的以文档为准)https://github.com/rlabrecque/Steamworks.NET/releases

Unity 接入rtmp unity 接入steam_客户端


在sdk引入项目后,你会发现,在项目的Assets目录的同级目录下,会生成一个steam_appid.txt文件,里面只存了一个appid,这个appid是一个默认值480,当你的app申请下来时,替换成自己的appid即可。需要注意的是项目平台需要切到PC平台,然后下载steam客户端,登录账号,才能正常进入编辑器,此时你打开steam会发现游戏正在运行,那就是指的你的unity编辑器。

开发过程中移动端和pc端会经常切换,而steam提供的宏定义DISABLESTEAMWORKS是关闭steam功能的,这个比较怪,也即是说默认情况下steam平台功能是开启的,再来回切平台时候要注意点。

在客户端点击支付之前,需要访问服务器接口GetUserInfo,获取接下来下订单所需的参数(country: ISO 3166-1-alpha-2 国家代码、currency:价格的 ISO 4217 货币代码。)这个推荐游戏启动时候就调用并存在本地。

接下来就是下订单所需的另外的参数

language

public string GetCurrentGameLanguage()
    {
        return SteamApps.GetCurrentGameLanguage();
    }

通过C#调用,但是返回过来的不是文档上要求的 ISO 639-1 语言代码。为此续写一个转换方法,然后将转换之后的数值传递给服务器即可。

local languages = {
    ["arabic"] = "ar",
    ["bulgarian"] = "bg",
    ["schinese"] = "zh-CN",
    ["tchinese"] = "zh-TW",
    ["czech"] = "cs",
    ["danish"] = "da",
    ["dutch"] = "nl",
    ["english"] = "en",
    ["finnish"] = "fi",
    ["french"] = "fr",
    ["german"] = "de",
    ["greek"] = "el",
    ["hungarian"] = "hu",
    ["italian"] = "it",
    ["japanese"] = "ja",
    ["koreana"] = "ko",
    ["norwegian"] = "no",
    ["polish"] = "pl",
    ["portuguese"] = "pt",
    ["brazilian"] = "pt-BR",
    ["romanian"] = "ro",
    ["russian"] = "ru",
    ["spanish"] = "es",
    ["latam"] = "es-419",
    ["swedish"] = "sv",
    ["thai"] = "th",
    ["turkish"] = "tr",
    ["ukrainian"] = "uk",
    ["vietnamese"] = "vn",
}

function SteamUtil.GetWebAPI(key)
    return languages[key]
end

和其他的支付不同(如谷歌支付,苹果支付),其他的支付平台能够提供在不同地区商品的价格以及货币符号类型,在steam平台需要自己获取到地区然后根据自己的配置,将价格传递给服务器。

好的,现在如果你准备好了各种参数,然后打包,上传至steam,再在steam平台下载游戏,如果一切正常的话,调用服务器的下订单接口,成功的话会出来一个steam支付弹窗,如下:(打码主要是怕公司策划看到,毕竟程序的地位最低,测试都能给我们骂哭)

Unity 接入rtmp unity 接入steam_Unity 接入rtmp_02


如果此时关闭支付弹窗,或者点击支付完成,在C#端会收到一个回调(首先回调要正确加入)

private Action<uint, ulong> OnPurchaseCallback;
    private void OnEnable()
    {
        m_MicroTxnAuthorizationResponse =
            Steamworks.Callback<MicroTxnAuthorizationResponse_t>.Create(OnMicroTxnAuthorizationResponse);
    }

    private void OnMicroTxnAuthorizationResponse(MicroTxnAuthorizationResponse_t pCallback)
    {
        Debug.Log("[" + MicroTxnAuthorizationResponse_t.k_iCallback + " - MicroTxnAuthorizationResponse] - " +
                  pCallback.m_unAppID + " -- " + pCallback.m_ulOrderID + " -- " + pCallback.m_bAuthorized);

        OnPurchaseCallback?.Invoke(pCallback.m_unAppID, pCallback.m_ulOrderID);
    }

然后在回调中调用服务器的支付验证接口,验证正常的话就下发奖励。
至此,steam支付接入完成,再次咒骂一下垃圾策划!!!