/* 硬声 注册登录之后,设置密码,把手机密码填到yingshengAccount里 多账户换行或者@隔开,格式:account=13888888888&password=xxxxxxxx 一天至少三次 */ const jsname = '硬声' const $ = new Env(jsname); var CryptoJS = require("crypto-js"); let envSplitor = ['\n','@'] let httpResult, httpReq, httpResp let userCookie = ($.isNode() ? process.env.yingshengAccount : $.getdata('yingshengAccount')) || ''; let userList = [] let userIdx = 0 let userCount = 0 let TASK_WAIT_TIME = 1000 let param_version = '2.3.1' let param_platform = 'ios' let salt = 'lw0270iBJzxXdJLRtePEENsauRzkHSqm' let AES_method = 'AES' let AES_mode = 'CBC' let AES_padding = 'Pkcs7' let AES_key = 'q09cRVOPCnfJzt7p' let AES_IV = 'cnry8k3o4WdCGU1T' /////////////////////////////////////////////////////////////////// class UserInfo { constructor(str) { this.index = ++userIdx this.name = this.index this.valid = false this.params = $.str2json(str) this.auth = '' this.user_id = '' this.device_id = $.randomString(32) this.userIdList = [] } calculateSign_yingsheng(param) { param['Authorization'] = this.auth param['platform'] = 'h5' param['timestamp'] = Math.floor(Date.now()/1000) param['sign'] = SHA1Encrypt(salt + SHA1Encrypt($.json2str(param,'&')) + this.auth) return param; } calculateSign_ysapi(param) { param['Authorization'] = this.auth param['platform'] = param_platform param['timestamp'] = Math.floor(Date.now()/1000) param['sign'] = SHA1Encrypt(AES_IV + AES_key + SHA1Encrypt($.json2str(param,'&',true)+AES_IV+AES_key) + this.auth) param['version'] = param_version return param; } async login() { try { let pwd = EncryptCrypto(AES_method,AES_mode,AES_padding,this.params.password,AES_key,AES_IV) let param = {'account': this.params.account, 'password': pwd, 'device_id': this.device_id} let url = `https://ysapi.elecfans.com/api/sso/accountLogin` let body = $.json2str(param,'&',true) param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { this.valid = true this.name = result.data.mobile this.coins = result.data.coins this.auth = result.data.Authorization this.user_id = result.data.user_id console.log(`登录成功`) console.log(`手机:${this.name}`) console.log(`硬币:${this.coins}`) } else { console.log(`登录失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async getInfo() { try { let param = {'user_id':this.user_id} let url = `https://ysapi.elecfans.com/api/member/getInfo?${$.json2str(param,'&')}` let body = `` param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('get',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { this.coins = result.data.coins console.log(`硬币:${this.coins}`) } else { console.log(`查询账户失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async getSignStatus() { try { let param = {'date':''} let url = `https://yingsheng.elecfans.com/ysapi/wapi/activity/signin/data?${$.json2str(param,'&')}` let body = `` param = this.calculateSign_yingsheng(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('get',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { if(result.data.data.today_is_sign==1) { console.log(`今日已签到`) } else { console.log(`今日未签到`) await $.wait(TASK_WAIT_TIME); await this.signin(); } } else { console.log(`查询签到状态失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async signin() { try { let param = {'date':''} let url = `https://yingsheng.elecfans.com/ysapi/wapi/activity/signin/signin` let body = JSON.stringify(param) param = this.calculateSign_yingsheng(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { console.log(`签到成功,获得${result.data.coins}硬币`) } else { console.log(`签到失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } //com_status: 10,11 -- 未完成, 12 -- 待领取, 13 -- 已完成 async getTaskList() { try { let param = {} let url = `https://yingsheng.elecfans.com/ysapi/wapi/activity/task/dailyList` let body = `` param = this.calculateSign_yingsheng(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('get',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { await $.wait(TASK_WAIT_TIME); await this.recommendList(); for(let key in result.data) { let task = result.data[key] if(task.title.includes('发布作品') || task.title.includes('作品播放量')) continue; if(task.title.indexOf('观看作品') > -1) { for(let idx in task.step) { if(task.step[idx].com_status == 12) { await $.wait(TASK_WAIT_TIME); await this.receiveCoin(task); break; } else if(task.step[idx].com_status != 13) { await $.wait(TASK_WAIT_TIME); await this.viewVideoAdd(idx); await $.wait(TASK_WAIT_TIME); await this.receiveCoin(task); break; } } } else { for(let idx in task.step) { let step = task.step[idx] if(step.com_status == 10) { await $.wait(TASK_WAIT_TIME); await this.taskReceive(task); } if(step.com_status == 12) { await $.wait(TASK_WAIT_TIME); await this.receiveCoin(task); } else if(step.com_status != 13) { let num = step.condition<25 ? (step.condition - step.finish_progress) : 1 let getReward = true; for(let i=0; i -1) { let rndIdx = Math.floor(Math.random()*2000) + 8000 await $.wait(TASK_WAIT_TIME); await this.thumbsup(rndIdx); await $.wait(TASK_WAIT_TIME); await this.thumbsup(rndIdx); } else if(task.title.indexOf('观看直播') > -1) { await $.wait(TASK_WAIT_TIME); await this.finishLive(); } else if(task.title.indexOf('关注') > -1) { let uid = this.userIdList[i] ? this.userIdList[i] : Math.floor(Math.random()*100000) + 4900000 await $.wait(TASK_WAIT_TIME); await this.doFollow(uid,1); await $.wait(TASK_WAIT_TIME); await this.doFollow(uid,2); } else { getReward = false; } } if(getReward) { await $.wait(TASK_WAIT_TIME); await this.receiveCoin(task); } } break; } } } } else { console.log(`查询任务列表失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async taskReceive(task) { try { let param = {'type':task.type} let url = `https://yingsheng.elecfans.com/ysapi/wapi/activity/task/receive` let body = JSON.stringify(param) param = this.calculateSign_yingsheng(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { console.log(`开始任务[${task.title}]成功`) } else { console.log(`开始任务[${task.title}]失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async receiveCoin(task) { try { let param = {'type':task.type} let url = `https://yingsheng.elecfans.com/ysapi/wapi/activity/task/receiveCoin` let body = JSON.stringify(param) param = this.calculateSign_yingsheng(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { console.log(`领取任务[${task.title}]奖励获得${result.data.coins}硬币`) } else { console.log(`领取任务[${task.title}]奖励失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async thumbsup(video_id) { try { let param = {'video_id':video_id} let url = `https://ysapi.elecfans.com/api/video/publish/thumbsup` let body = $.json2str(param,'&') param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { console.log(`${result.data.msg}: video_id=${video_id}`) } else { console.log(`点赞[video_id=${video_id}]失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async viewVideoAdd(step) { try { let param = {'step':step} let url = `https://ysapi.elecfans.com/api/activity/task/viewVideo/add` let body = $.json2str(param,'&') param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { console.log(`刷新看视频进度[step=${step}]成功`) } else { console.log(`刷新看视频进度[step=${step}]失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async finishLive() { try { let param = {} let url = `https://ysapi.elecfans.com/api/activity/task/live/finish` let body = `` param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { console.log(`完成观看直播任务成功`) } else { console.log(`完成观看直播任务失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async recommendList() { try { let param = {} let url = `https://ysapi.elecfans.com/api/live/recommendList` let body = `` param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('get',urlObject) let result = httpResult; if(!result) return //console.log(result) if(result.code==0) { for(let item of result.data.data) { this.userIdList.push(item.user_id) } } else { console.log(`获取推荐列表失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async doFollow(user_id,type) { try { let param = {'user_id':user_id, 'type':type} let url = `https://ysapi.elecfans.com/api/member/follow` let body = $.json2str(param,'&') param = this.calculateSign_ysapi(param) let urlObject = populateUrlObject(url,param,body) await httpRequest('post',urlObject) let result = httpResult; if(!result) return //console.log(result) let str = type==1 ? '关注' : '取消关注' if(result.code==0) { console.log(`${str}用户[${user_id}]成功`) } else { console.log(`${str}用户[${user_id}]失败: ${result.message}`) } } catch(e) { console.log(e) } finally {} } async userTask() { console.log(`\n============= 账号[${this.index}] =============`) await this.login(); if(!this.valid) return; console.log(`----------- 签到 -----------`) await $.wait(TASK_WAIT_TIME); await this.getSignStatus(); console.log(`----------- 任务 -----------`) await $.wait(TASK_WAIT_TIME); await this.getTaskList(); return; console.log(`----------- 积分 -----------`) await $.wait(TASK_WAIT_TIME); await this.getInfo(); } } !(async () => { if (typeof $request !== "undefined") { await GetRewrite() }else { //let nowtime = $.time('yyyy-MM-dd hh:mm:ss'); //console.log(nowtime); if(!(await checkEnv())) return; for(let user of userList) { await user.userTask(); } } })() .catch((e) => console.log(e)) .finally(() => $.done()) /////////////////////////////////////////////////////////////////// async function GetRewrite() { } async function checkEnv() { if(userCookie) { let splitor = envSplitor[0]; for(let sp of envSplitor) { if(userCookie.indexOf(sp) > -1) { splitor = sp; break; } } for(let userCookies of userCookie.split(splitor)) { if(userCookies) userList.push(new UserInfo(userCookies)) } userCount = userList.length } else { console.log('未找到CK') return; } console.log(`共找到${userCount}个账号`) return true } //////////////////////////////////////////////////////////////////// function populateUrlObject(url,param,body=''){ let host = url.replace('//','/').split('/')[1] let urlObject = { url: url, headers: { 'Host': host, 'Connection': 'keep-alive', 'Authorization': param.Authorization, 'timestamp': param.timestamp, 'sign': param.sign, 'platform': param.platform, 'version': param.version, }, timeout: 5000, } if(body) { urlObject.body = body urlObject.headers['Content-Type'] = 'application/x-www-form-urlencoded' if(host=='yingsheng.elecfans.com') urlObject.headers['Content-Type'] = 'application/json;charset=utf-8' urlObject.headers['Content-Length'] = urlObject.body ? urlObject.body.length : 0 } return urlObject; } async function httpRequest(method,url) { httpResult = null, httpReq = null, httpResp = null; return new Promise((resolve) => { $.send(method, url, async (err, req, resp) => { try { httpReq = req; httpResp = resp; if (err) { console.log(`${method}请求失败`); console.log(JSON.stringify(err)); } else { if(resp.body) { if(typeof resp.body == "object") { httpResult = resp.body; } else { try { httpResult = JSON.parse(resp.body); } catch (e) {} } } } } catch (e) { console.log(e); } finally { resolve(); } }); }); } //////////////////////////////////////////////////////////////////// function EncryptCrypto(method,mode,padding,message,key,iv) { return CryptoJS[method].encrypt( CryptoJS.enc.Utf8.parse(message), CryptoJS.enc.Utf8.parse(key), {mode:CryptoJS.mode[mode], padding:CryptoJS.pad[padding], iv:CryptoJS.enc.Utf8.parse(iv)} ).ciphertext.toString(CryptoJS.enc.Base64); } function DecryptCrypto(method,mode,padding,message,key,iv) { return CryptoJS[method].decrypt( {ciphertext: CryptoJS.enc.Base64.parse(message)}, CryptoJS.enc.Utf8.parse(key), {mode:CryptoJS.mode[mode], padding:CryptoJS.pad[padding], iv:CryptoJS.enc.Utf8.parse(iv)} ).toString(CryptoJS.enc.Utf8); } var Base64={_keyStr:"ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/=",encode:function(e){var t="";var n,r,i,s,o,u,a;var f=0;e=Base64._utf8_encode(e);while(f>2;o=(n&3)<<4|r>>4;u=(r&15)<<2|i>>6;a=i&63;if(isNaN(r)){u=a=64}else if(isNaN(i)){a=64}t=t+this._keyStr.charAt(s)+this._keyStr.charAt(o)+this._keyStr.charAt(u)+this._keyStr.charAt(a)}return t},decode:function(e){var t="";var n,r,i;var s,o,u,a;var f=0;e=e.replace(/[^A-Za-z0-9+/=]/g,"");while(f>4;r=(o&15)<<4|u>>2;i=(u&3)<<6|a;t=t+String.fromCharCode(n);if(u!=64){t=t+String.fromCharCode(r)}if(a!=64){t=t+String.fromCharCode(i)}}t=Base64._utf8_decode(t);return t},_utf8_encode:function(e){e=e.replace(/rn/g,"n");var t="";for(var n=0;n127&&r<2048){t+=String.fromCharCode(r>>6|192);t+=String.fromCharCode(r&63|128)}else{t+=String.fromCharCode(r>>12|224);t+=String.fromCharCode(r>>6&63|128);t+=String.fromCharCode(r&63|128)}}return t},_utf8_decode:function(e){var t="";var n=0;var r=c1=c2=0;while(n191&&r<224){c2=e.charCodeAt(n+1);t+=String.fromCharCode((r&31)<<6|c2&63);n+=2}else{c2=e.charCodeAt(n+1);c3=e.charCodeAt(n+2);t+=String.fromCharCode((r&15)<<12|(c2&63)<<6|c3&63);n+=3}}return t}} function MD5Encrypt(a){function b(a,b){return a<>>32-b}function c(a,b){var c,d,e,f,g;return e=2147483648&a,f=2147483648&b,c=1073741824&a,d=1073741824&b,g=(1073741823&a)+(1073741823&b),c&d?2147483648^g^e^f:c|d?1073741824&g?3221225472^g^e^f:1073741824^g^e^f:g^e^f}function d(a,b,c){return a&b|~a&c}function e(a,b,c){return a&c|b&~c}function f(a,b,c){return a^b^c}function g(a,b,c){return b^(a|~c)}function h(a,e,f,g,h,i,j){return a=c(a,c(c(d(e,f,g),h),j)),c(b(a,i),e)}function i(a,d,f,g,h,i,j){return a=c(a,c(c(e(d,f,g),h),j)),c(b(a,i),d)}function j(a,d,e,g,h,i,j){return a=c(a,c(c(f(d,e,g),h),j)),c(b(a,i),d)}function k(a,d,e,f,h,i,j){return a=c(a,c(c(g(d,e,f),h),j)),c(b(a,i),d)}function l(a){for(var b,c=a.length,d=c+8,e=(d-d%64)/64,f=16*(e+1),g=new Array(f-1),h=0,i=0;c>i;)b=(i-i%4)/4,h=i%4*8,g[b]=g[b]|a.charCodeAt(i)<>>29,g}function m(a){var b,c,d="",e="";for(c=0;3>=c;c++)b=a>>>8*c&255,e="0"+b.toString(16),d+=e.substr(e.length-2,2);return d}function n(a){a=a.replace(/\r\n/g,"\n");for(var b="",c=0;cd?b+=String.fromCharCode(d):d>127&&2048>d?(b+=String.fromCharCode(d>>6|192),b+=String.fromCharCode(63&d|128)):(b+=String.fromCharCode(d>>12|224),b+=String.fromCharCode(d>>6&63|128),b+=String.fromCharCode(63&d|128))}return b}var o,p,q,r,s,t,u,v,w,x=[],y=7,z=12,A=17,B=22,C=5,D=9,E=14,F=20,G=4,H=11,I=16,J=23,K=6,L=10,M=15,N=21;for(a=n(a),x=l(a),t=1732584193,u=4023233417,v=2562383102,w=271733878,o=0;o=0;j--)str+=sHEXChars.charAt((num>>(j*4))&0x0F);return str;}function AlignSHA1(sIn){var nblk=((sIn.length+8)>>6)+1,blks=new Array(nblk*16);for(var i=0;i>2]|=sIn.charCodeAt(i)<<(24-(i&3)*8);blks[i>>2]|=0x80<<(24-(i&3)*8);blks[nblk*16-1]=sIn.length*8;return blks;}function rol(num,cnt){return(num<>>(32-cnt));}function ft(t,b,c,d){if(t<20)return(b&c)|((~b)&d);if(t<40)return b^c^d;if(t<60)return(b&c)|(b&d)|(c&d);return b^c^d;}function kt(t){return(t<20)?1518500249:(t<40)?1859775393:(t<60)?-1894007588:-899497514;}var x=AlignSHA1(msg);var w=new Array(80);var a=1732584193;var b=-271733879;var c=-1732584194;var d=271733878;var e=-1009589776;for(var i=0;i -1 && process.exit(0); return new class { constructor(name,env) { this.name = name this.notifyStr = '' this.startTime = (new Date).getTime() Object.assign(this,env) console.log(`${this.name} 开始运行:\n`) } isNode() { return "undefined" != typeof module && !!module.exports } isQuanX() { return "undefined" != typeof $task } isSurge() { return "undefined" != typeof $httpClient && "undefined" == typeof $loon } isLoon() { return "undefined" != typeof $loon } getdata(t) { let e = this.getval(t); if (/^@/.test(t)) { const[, s, i] = /^@(.*?)\.(.*?)$/.exec(t), r = s ? this.getval(s) : ""; if (r) try { const t = JSON.parse(r); e = t ? this.lodash_get(t, i, "") : e } catch (t) { e = "" } } return e } setdata(t, e) { let s = !1; if (/^@/.test(e)) { const[, i, r] = /^@(.*?)\.(.*?)$/.exec(e), o = this.getval(i), h = i ? "null" === o ? null : o || "{}" : "{}"; try { const e = JSON.parse(h); this.lodash_set(e, r, t), s = this.setval(JSON.stringify(e), i) } catch (e) { const o = {}; this.lodash_set(o, r, t), s = this.setval(JSON.stringify(o), i) } } elses = this.setval(t, e); return s } getval(t) { return this.isSurge() || this.isLoon() ? $persistentStore.read(t) : this.isQuanX() ? $prefs.valueForKey(t) : this.isNode() ? (this.data = this.loaddata(), this.data[t]) : this.data && this.data[t] || null } setval(t, e) { return this.isSurge() || this.isLoon() ? $persistentStore.write(t, e) : this.isQuanX() ? $prefs.setValueForKey(t, e) : this.isNode() ? (this.data = this.loaddata(), this.data[e] = t, this.writedata(), !0) : this.data && this.data[e] || null } send(m, t, e = (() => {})) { if(m != 'get' && m != 'post' && m != 'put' && m != 'delete') { console.log(`无效的http方法:${m}`); return; } if(m == 'get' && t.headers) { delete t.headers["Content-Type"]; delete t.headers["Content-Length"]; } else if(t.body && t.headers) { if(!t.headers["Content-Type"]) t.headers["Content-Type"] = "application/x-www-form-urlencoded"; } if(this.isSurge() || this.isLoon()) { if(this.isSurge() && this.isNeedRewrite) { t.headers = t.headers || {}; Object.assign(t.headers, {"X-Surge-Skip-Scripting": !1}); } let conf = { method: m, url: t.url, headers: t.headers, timeout: t.timeout, data: t.body }; if(m == 'get') delete conf.data $axios(conf).then(t => { const { status: i, request: q, headers: r, data: o } = t; e(null, q, { statusCode: i, headers: r, body: o }); }).catch(err => console.log(err)) } else if (this.isQuanX()) { t.method = m.toUpperCase(), this.isNeedRewrite && (t.opts = t.opts || {}, Object.assign(t.opts, { hints: !1 })), $task.fetch(t).then(t => { const { statusCode: i, request: q, headers: r, body: o } = t; e(null, q, { statusCode: i, headers: r, body: o }) }, t => e(t)) } else if (this.isNode()) { this.got = this.got ? this.got : require("got"); const { url: s, ...i } = t; this.instance = this.got.extend({ followRedirect: false }); this.instance[m](s, i).then(t => { const { statusCode: i, request: q, headers: r, body: o } = t; e(null, q, { statusCode: i, headers: r, body: o }) }, t => { const { message: s, response: i } = t; e(s, i, i && i.body) }) } } time(t) { let e = { "M+": (new Date).getMonth() + 1, "d+": (new Date).getDate(), "h+": (new Date).getHours(), "m+": (new Date).getMinutes(), "s+": (new Date).getSeconds(), "q+": Math.floor(((new Date).getMonth() + 3) / 3), S: (new Date).getMilliseconds() }; /(y+)/.test(t) && (t = t.replace(RegExp.$1, ((new Date).getFullYear() + "").substr(4 - RegExp.$1.length))); for (let s in e) new RegExp("(" + s + ")").test(t) && (t = t.replace(RegExp.$1, 1 == RegExp.$1.length ? e[s] : ("00" + e[s]).substr(("" + e[s]).length))); return t } async showmsg() { if(!this.notifyStr) return; let notifyBody = this.name + " 运行通知\n\n" + this.notifyStr if($.isNode()){ var notify = require('./sendNotify'); console.log('\n============== 推送 ==============') await notify.sendNotify(this.name, notifyBody); } else { this.msg(notifyBody); } } logAndNotify(str) { console.log(str) this.notifyStr += str this.notifyStr += '\n' } msg(e = t, s = "", i = "", r) { const o = t => { if (!t) return t; if ("string" == typeof t) return this.isLoon() ? t : this.isQuanX() ? { "open-url": t } : this.isSurge() ? { url: t } : void 0; if ("object" == typeof t) { if (this.isLoon()) { let e = t.openUrl || t.url || t["open-url"], s = t.mediaUrl || t["media-url"]; return { openUrl: e, mediaUrl: s } } if (this.isQuanX()) { let e = t["open-url"] || t.url || t.openUrl, s = t["media-url"] || t.mediaUrl; return { "open-url": e, "media-url": s } } if (this.isSurge()) { let e = t.url || t.openUrl || t["open-url"]; return { url: e } } } }; this.isMute || (this.isSurge() || this.isLoon() ? $notification.post(e, s, i, o(r)) : this.isQuanX() && $notify(e, s, i, o(r))); let h = ["", "============== 系统通知 =============="]; h.push(e), s && h.push(s), i && h.push(i), console.log(h.join("\n")) } getMin(a,b){ return ((anumStr.length) ? (length-numStr.length) : 0 let retStr = '' for(let i=0; i setTimeout(e, t)) } done(t = {}) { const e = (new Date).getTime(), s = (e - this.startTime) / 1e3; console.log(`\n${this.name} 运行结束,共运行了 ${s} 秒!`) if(this.isSurge() || this.isQuanX() || this.isLoon()) $done(t) } }(name,env) }