Greasy Fork is available in English.

GMscrobber

记录在线音乐到 last.fm

This script should not be not be installed directly. It is a library for other scripts to include with the meta directive // @require https://update.greasyfork.org/scripts/7806/183861/GMscrobber.js

var log = function(){},GM_log,////unsafeWindow.console.log,//
	getVal = GM_getValue,
	setVal = GM_setValue,
	delVal = GM_deleteValue,
	xhr = GM_xmlhttpRequest,
	rmc = GM_registerMenuCommand,

	_md5 = hex_md5;
  
//simple scrobbler for userscript
var Scrobbler = function(){
	var apikey = "4472aff22b680a870bfd583f99644a03",
		secret = "cbc5528721f63b839720633d7c1258d2",
		apiurl = "http://ws.audioscrobbler.com/2.0/",	
		scrate = .9,
		tokenreg = /[?&]token=(.+?)(&|#|$)/,
		_timer, _shift;
    
  /**
   * Scrobbler
   * @constructor
   * @param {Object} info 该页面 scrobbler 信息
   * @param {Number} info.type 1: 手动调用scrobble记录歌曲; 
           0(缺省): 在调用 nowplaying 后根据播放时间自动调用 scrobble 记录歌曲
   * @param {String} info.name 该 scrobbler 名字, 会显示在 greasemonkey 菜单中
   * @param {Function} info.ready 回调. scrobbler sessionid 取得后会调用此函数.
   * @param {Number} [info.scrate] 自动记录的百分比, info.type == 1 时无效
   */
	var fn = function(info){
		this.type = info.type;
		this.name = info.name || "";
		this.ready = info.ready || function(){};
		this.scrate = info.scrate || scrate;
		this.init();
	};
	fn.prototype = {
		init: function(){
			var sk = getVal("session"),
        token = document.location.search.match(tokenreg),
        that = this;
      
      token = token && token[1];
			
			log(sk + "\n" + token + "\n" + document.location.href);
			if(sk){
				rmc("停止记录" + that.name, that.delSession);
				that.username = sk.split("/")[0];
				that.sk = sk.split("/")[1];
				setTimeout(function(){that.ready()}, 0);
			}else if(token){
				that.sk = "wait";
				that.ajax({method:"auth.getSession", _sig:"", token: token},
					function(d){
					log(JSON.stringify(d))
						if(d.session && d.session.key){
							that.sk = d.session.key;
							that.username = d.session.name;
							setVal("session",that.username + "/" + that.sk);
							rmc("停止记录" + that.name, that.delSession);
							that.ready();
						}
					}, true);
			}else{
				rmc("开始记录" + that.name, fn.redirect);
			}
      
      this.listeners = {};
		},
		
		getSession: function(){
			return this.sk;
		},
		delSession: function(){
			delVal("session");
			document.location = document.location.href.replace(tokenreg, "");
		},
    /**
     * 定期检查页面歌曲信息变化. 将歌曲信息获取函数传给此函数, 即可自动完成歌曲的记录.
     * @param {Function} getSongInfo 各页面脚本的歌曲信息获取函数.
        应该返回歌曲信息: {title: '', artist: '', duration: 0, playTime: '', album: ''}
     * @param {Object} opts 
     * @param {Nunber} opts.checktime 定时器周期, 毫秒
     */
    setSongInfoFN: (function(){
      var checkTime;
      var fn = function(getSongInfo, opts){
        opts = opts || {};
        checkTime = opts.checktime || 2000;
        
        var info = {}, that = this;
        
        setInterval(function(){
          try{
            that.getSongInfo = getSongInfo;
            info = getSongInfo();
            infoChecker.call(that, info);
          }catch(e){
            log(e.stack);
          }
        }, checkTime);
      };
      var oldSong = {};
      var infoChecker = function(song){
        if(song.title && song.artist && song.duration){
          if(song.title != oldSong.title || song.artist != oldSong.artist){
            this.nowPlaying(song);
          }else{
            //log(this.state)
            if(song.playTime != oldSong.playTime){
              if(song.playTime <= Math.ceil(checkTime / 1000) && (Date.now() / 1000 - this.timestamp > song.duration)){
                log(song.title + ' repeating.');
                //单曲重复
                this.nowPlaying(song);
              }else{
                this.state != 'play' && this.play(song.playTime + this.info.offset);
              }
            }else{
              this.state == 'play' && this.pause();
            }
          }
          oldSong = uso.clone(song);
        }
      };
      return fn;
    })(),
		
	//song's command
  
    /**
     * 向 last.fm 发送正在播放请求
     * @param {Object} song 歌曲信息
     * @param {String} song.title 曲名
     * @param {String} song.artist 歌手(多个歌手用 & 连接)
     * @param {String} song.duration 时长. 单位: 秒
     * @param {String} [song.album] 专辑名
     * @param {String} [song.playTime] 开始播放时的时间
     */
		nowPlaying: function(song){
			var that = this;
			this.song = song; //song: {title: "", artist: "", duration: "", album: ""}
			this.timestamp = Math.floor(new Date().getTime()/1000);
			this.info = {iscrobble: false, offset: 0};
			this.play(song.playTime || 0);
			//log(JSON.stringify(song));
			log(song.title + " now playing");
			this.ajax({
				method: "track.updateNowPlaying", 
				track: song.title,
				artist: song.artist,
				duration: song.duration,
				album: song.album,//
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d));
			},
			true);
      
      //typeof meta != 'undefined' && that.record(song, 'nowplaying');
      this.fire('nowplaying');
      lyr.call(this);
		},
    
    //
    record: function(song, type){
      var query = uso.clone(song), path;
      query.source = document.location.host;
      query.version = meta.version;
      query.username = this.username;
      query = uso.paramSerialize(query);
      if(type == 'nowplaying'){
        path = '/nowplaying?';
      }else if(type == 'scrobble'){
        path = '/scrobble?';
      }else{
        return false;
      }
      xhr({
        method: 'GET',
        url: meta.namespace.replace('\/', '') + path + query
      });
    },
    
    /**
     * 向 last.fm 发送正在记录请求
     * @param {Object} [song] 歌曲信息. 当nowPlaying中的歌曲信息不全时, 应在此补全. last.fm 的播放记录以此为准
     */
		scrobble: function(song){
			var that = this;
			song = song || this.song;
			this.ajax({
				method: "track.scrobble", 
				track: song.title,
				artist: song.artist,
				album: song.album,
				timestamp: this.timestamp,
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d));
				that.info.iscrobble = true;
				log(song.artist + "'s " + song.title + " / " + song.album + " scrobbled..");
			},
			true);
      //typeof meta != 'undefined' && that.record(song, 'scrobble');
      this.fire('scrobble');
		},
		/** love
     * @param {Object} [song] 歌曲信息. 事实上你可以听得是一首歌, love 的却是另一首
     */
    love: function(song){
			song = song || this.song;			
			this.ajax({
				method: "track.love", 
				track: song.title,
				artist: song.artist,
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d))
			},
			true);
      this.fire('love');
		},
		unlove: function(song){
			song = song || this.song;
			this.ajax({
				method: "track.unlove", 
				track: song.title,
				artist: song.artist,
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d))
			},
			true);
      this.fire('unlove');
		},
    ban: function(song){
      song = song || this.song;
      this.ajax({
				method: "track.ban", 
				track: song.title,
				artist: song.artist,
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d))
			},
			true);
      this.fire('ban');
    },
    unban: function(song){
      song = song || this.song;
      this.ajax({
				method: "track.unban", 
				track: song.title,
				artist: song.artist,
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d))
			},
			true);
      this.fire('unban');
    },
		getInfo: function(song, callback){
			var that = this;
			song = song || this.song;
			this.ajax({
				method: "track.getInfo", 
				track: song.title,
				artist: song.artist,
				username: this.username,
				_sig:""
			},
			function(d){
				//log(JSON.stringify(d));
				var n, t;
				if(d.track){
					n = d.track.userplaycount ? d.track.userplaycount : 0;
					if(d.track.userloved == "1"){
						t = "1";
					}else if(d.track.userloved === "0"){
						t = "0";
					}
				}else{
					n = 0;
					t = "0";
				}
				typeof callback == "function" && callback({islove: t, len: n});
			},
			true);
		},
		
	//play control
    /**
     * 开始播放. 从所有停止状态开始播放, 都应该调用此函数
     * @param {Number} [realPlayTime] 播放时校正. 
        当一首歌暂停次数太多的时候, 记录的播放时间可能会有误差, 传入 realPlayTime 即可校正播放的时间
     */
		play: function(realPlayTime){
			var that = this, rpt = realPlayTime, rt;
			this.state = "play";
      this.fire(this.state);
			
			if(!rpt){
				rpt = Math.floor(new Date().getTime()/1000) - this.timestamp;
			}
      rt = (Math.min(that.song.duration*this.scrate, 240) - rpt)*1000;//remain time
			if(!this.type && !this.info.iscrobble){
				clearTimeout(_timer);
        log('will scrobbler in: ' + rt/1000 + ' seconds')
				_timer = setTimeout(function(){that.scrobble()}, rt);
			}
		},
		pause: function(){
			this.type || clearTimeout(_timer);
			this.state = "pause";
      this.fire(this.state);
		},
		buffer: function(){
			this.type || clearTimeout(_timer);
			this.state = "buffer";
      this.fire(this.state);
		},
		stop: function(){
			this.type || clearTimeout(_timer);
			this.state = "stop";
      this.fire(this.state);
		},
		seek: function(offset){
      this.state = "seek";
      this.fire(this.state, offset);
			this.info.offset += offset;
      log('seek, offset: ' + offset + ', totle offset: ' + this.info.offset);
		},
		
		ajax: function(params, callback, auth){
			if(this.sk){
				params.sk = this.sk;
			}else{
				delete params.sk;
			}
			fn.ajax(params, callback, auth);
		},
    
    on: function(event, handler){
      this.listeners[event] = this.listeners[event] || [];
      this.listeners[event].push(handler);
      return this;
    },
    off: function(event, handler){
      var listeners = this.listeners[event] || [];
      if(handler){
        for(var i = 0, l = listeners.length; i < l; i++){
          if(handler == listeners[i]){
            delete listeners[i];
          }
        }
      }else{
        delete this.listeners[event];
      }
      return this;
    },
    fire: function(event){
      var listeners = this.listeners[event] || [];
      var args = [].slice.call(arguments);
      args.shift();
      
      for(var i = 0, l = listeners.length; i < l; i++){
        listeners[i] && listeners[i].apply(this, args);
      }
      return this;
    }
	};
	
	fn.redirect = function(){
		document.location = "http://www.last.fm/api/auth/?api_key=" + apikey + "&cb=" + encodeURIComponent(document.location.href.replace(/^https/,'http'));
	};
	fn.ajax = function(params, callback, auth){
		var method = "POST",
			headers = {"Content-Type": "application/x-www-form-urlencoded"},
			url = apiurl + "?format=json",
			data = "";
		if(!auth){
			method = "GET";
			headers = {};
			url = url + "&" + fn.paramsInit(params);
			data = "";
		}else{
			data = fn.paramsInit(params, true);
		}
		xhr({
			method: method,
			headers: headers,
			url: url,
			data: data,
			onload: function(d){
				var res = JSON.parse(d.responseText);
				//log(JSON.stringify(d));
				res.error && log(JSON.stringify(d));
				if(res.error == "9"){
					//delVal(fn.name);
					//fn.redirect();
				}
				callback(res);
			},
			onerror: function(e){
				//alert(params.method + "failed");
				log("[ error ] " + params.method + " request failed.. " + JSON.stringify(e));
			}
		});
	};
	fn.paramsInit = function(params){
		var keys = [], str1 = "", str2 = "", flag;
		if(typeof params._sig != "undefined"){
			delete params._sig;
			flag = true;
		}else{
			flag = end;
		}
		params.api_key = apikey;
    for(var key in params){
			if(params[key]){
				keys.push(key + params[key]);
			}
		}
		str1 = uso.paramSerialize(params);
		keys.sort();
		str2 = keys.join("") + secret;
		//log("str2: " + str2);
		//log("str1: " + str1);
		if(flag){
			return str1 + "&api_sig=" + _md5(str2);
		}else{
			return str1;
		}
	};
	return fn;
}();

/**
 * 歌词查询
 * 歌词 API 来源于 (@solos)[https://github.com/solos] 的(歌词迷)[http://api.geci.me/en/latest/index.html]
 */
var lyr = function(){
  var lrc;
  var fn = function(){
    var title = this.song.title
      , artist = this.song.artist
      , album = this.song.album
      , startTime = this.song.playTime || 0
      , that = this
      ;
    
    log('lyric for ' + artist + '\'s ' + title + ' / ' + album + ' is getting..');
    
    var t1 = Date.now();
    lrc && lrc.stop();
    
    xhr({
      method: 'GET',
      url: 'http://geci.me/api/lyric/' + title + '/' + artist,
      onload: function (res){
        //log(res.responseText);
        var lyrSrc = '';
        res = JSON.parse(res.responseText);
        
        if(res.count){
          lyrSrc = res.result[0].lrc;
          log('lyrics link: ' + lyrSrc);
          xhr({
            method: 'GET',
            url: lyrSrc,
            //overrideMimeType: 'text/plain; charset=gb2312',
            onload: function(d){
              var txt = d.responseText;
              GM_log(txt);
              
              if(typeof Lrc != 'undefined'){
                lrc = (new Lrc(txt, function(txt, extra){
                  
                  txt && lrcOut.call(this, txt, extra);
                  //that.getSongInfo && lrcOut('player time: ' + that.getSongInfo().playTime)
                  //lrcOut(lrc.lrc.split('\n')[extra.originLineNum])
                }));
                //提前1秒显示歌词
                that.state === 'play' && lrc.play(startTime * 1000 + Date.now() - t1 + 1000);
              }
            },
            onerror: function(){
              lrcOut('some error occured..');
            }
          });
        }else{
          lrcOut('no lyrics for ' + title);
        }
      },
      onerror: function (e){
        lrcOut('搜索歌词失败! ' + JSON.stringify(e));
      }
    });
    
    this.off('pause', pause).off('play', pause).off('buffer', pause).
      on('pause', pause).on('play', pause).on('buffer', pause).
      off('seek', seek).on('seek', seek);
  };
  
  function pause(){
    log('pause')
    lrc && lrc.pauseToggle();
  }
  
  function seek(offset){
    lrc && lrc.seek(-offset * 1000);
  }

  return fn;
}();

//重新此方法可实现自己歌词输出
var lrcOut = function(txt){
  unsafeWindow.console && unsafeWindow.console.log(txt);
  //GM_log(txt);
};

//userscript 自动更新工具
var uso = {
	//usersctipt meta 解析工具
	metaParse: function(metadataBlock) {
	  var headers = {};
	  var line, name, prefix, header, key, value, _t;

		var lines = metadataBlock.split(/\n/).filter(function(line){return /\/\/ @/.test(line)});
		lines.forEach(function(line) {
		  _t = line.match(/\/\/ @(\S+)\s*(.*)/);
      name = _t[1];
      value = _t[2];

		  switch (name) {
			case "licence":
			  name = "license";
			  break;
		  }

		  _t = name.split(/:/).reverse();
      key = _t[0];
      prefix = _t[1];

		  if (prefix) {
			if (!headers[prefix]) 
			  headers[prefix] = new Object;
			header = headers[prefix];
		  } else
			header = headers;

		  if (header[key] && !(header[key] instanceof Array))
			header[key] = new Array(header[key]);

		  if (header[key] instanceof Array)
			header[key].push(value);
		  else
			header[key] = value;
		});

		headers["licence"] = headers["license"];

	  return headers;
	},
	/**
   * 自动升级工具
   * @param {String} ver 当前版本号
   * @param {String} id userscript.org 上的编号
   * @param {Function} cb 检测结果回调
   */
  check: function(ver, id, cb){
		var that = this, self = arguments.callee, flag = false;
		xhr({
		  method:"GET",
		  url:"https://userscripts.org/scripts/source/" + id + ".meta.js",
		  headers:{
			"Accept":"text/javascript; charset=UTF-8"
		  },
		  overrideMimeType:"application/javascript; charset=UTF-8",
		  onload:function(response) {
			var meta = that.metaParse(response.responseText),
				ver0 = meta.version, r;
				
			if(that.verCompare(ver, ver0) < 0){
				flag = true;
        if(meta.initiative == 'true' || meta.initiative == 'yes'){
          alert([
              meta.name + " ver" + ver0, "",
              meta.changelog].join("\n    "));
          document.location = "http://userscripts.org/scripts/source/" + id + ".user.js";
        }else{
          rmc("更新" + meta.name + " " + ver + " 至 " + ver0, function(){
            r = confirm([
              meta.name + " ver: " + ver0, "",
              "更新说明: " + meta.changelog, "",
              "是否更新?"].join("\n    "));
            if(r){
              document.location = "http://userscripts.org/scripts/source/" + id + ".user.js";
            }
          });
        }
			}
			typeof cb == "function" && cb(flag);
		  },
		  onerror: function(e){
			log("check version failed; \n" + JSON.stringify(e));
		  }
		});
	},
	verCompare: function(ver0, ver1){
		var a0 = ver0.split("."), a1 = ver1.split("."),
			len = Math.max(a0.length, a1.length);
		if(ver0 == ver1){
			return 0;
		}
		for(var i = 0; i < len; i++){
			if(a0[i] < a1[i] || typeof a0[i] == "undefined"){
				return -1;//ver0 < ver1
			}else if(a0[i] != a1[i]){
        break;
      }
		}
		return 1;
	},
  paramSerialize: function(params){
    var str = '';
    for(var key in params){
			if(params[key]){
				str += encodeURIComponent(key) + "=" + encodeURIComponent(params[key]) + "&";
			}
		}
    str = str.replace(/&$/, "");
    return str;
  },
  clone: function(obj){
    if(obj == null || typeof(obj) != 'object'){ return obj }
    var temp = obj.constructor(); // changed
    for(var key in obj){ temp[key] = arguments.callee(obj[key]) }
    return temp;
  },
  //str hh:mm:ss
  timeParse: function(str) {
    var ts = str.trim().match(/(?:(\d+):)?(\d\d?):(\d\d?)/);
    return (ts[1] || 0) * 3600 + ts[2] * 60 + ts[3] * 1 || 0;
  },
  
  //函数切面
  //前面的函数返回值传入 breakCheck 判断, breakCheck 返回值为真时不执行后面的函数
  beforeFn: function (oriFn, fn, breakCheck) {
    return function() {
      var ret = fn.apply(this, arguments);
      if(breakCheck && breakCheck.call(this, ret)){
        return ret;
      }
      return oriFn.apply(this, arguments);
    };
  },

  afterFn: function (oriFn, fn, breakCheck) {
    return function() {
      var ret = oriFn.apply(this, arguments);
      if(breakCheck && breakCheck.call(this, ret)){
        return ret;
      }
      fn.apply(this, arguments);
      return ret;
    }
  }
};


/*
 * A JavaScript implementation of the RSA Data Security, Inc. MD5 Message
 * Digest Algorithm, as defined in RFC 1321.
 * Version 2.2 Copyright (C) Paul Johnston 1999 - 2009
 * Other contributors: Greg Holt, Andrew Kepert, Ydnar, Lostinet
 * Distributed under the BSD License
 * See http://pajhome.org.uk/crypt/md5 for more info.
 */
var hexcase=0;function hex_md5(a){return rstr2hex(rstr_md5(str2rstr_utf8(a)))}function hex_hmac_md5(a,b){return rstr2hex(rstr_hmac_md5(str2rstr_utf8(a),str2rstr_utf8(b)))}function md5_vm_test(){return hex_md5("abc").toLowerCase()=="900150983cd24fb0d6963f7d28e17f72"}function rstr_md5(a){return binl2rstr(binl_md5(rstr2binl(a),a.length*8))}function rstr_hmac_md5(c,f){var e=rstr2binl(c);if(e.length>16){e=binl_md5(e,c.length*8)}var a=Array(16),d=Array(16);for(var b=0;b<16;b++){a[b]=e[b]^909522486;d[b]=e[b]^1549556828}var g=binl_md5(a.concat(rstr2binl(f)),512+f.length*8);return binl2rstr(binl_md5(d.concat(g),512+128))}function rstr2hex(c){try{hexcase}catch(g){hexcase=0}var f=hexcase?"0123456789ABCDEF":"0123456789abcdef";var b="";var a;for(var d=0;d<c.length;d++){a=c.charCodeAt(d);b+=f.charAt((a>>>4)&15)+f.charAt(a&15)}return b}function str2rstr_utf8(c){var b="";var d=-1;var a,e;while(++d<c.length){a=c.charCodeAt(d);e=d+1<c.length?c.charCodeAt(d+1):0;if(55296<=a&&a<=56319&&56320<=e&&e<=57343){a=65536+((a&1023)<<10)+(e&1023);d++}if(a<=127){b+=String.fromCharCode(a)}else{if(a<=2047){b+=String.fromCharCode(192|((a>>>6)&31),128|(a&63))}else{if(a<=65535){b+=String.fromCharCode(224|((a>>>12)&15),128|((a>>>6)&63),128|(a&63))}else{if(a<=2097151){b+=String.fromCharCode(240|((a>>>18)&7),128|((a>>>12)&63),128|((a>>>6)&63),128|(a&63))}}}}}return b}function rstr2binl(b){var a=Array(b.length>>2);for(var c=0;c<a.length;c++){a[c]=0}for(var c=0;c<b.length*8;c+=8){a[c>>5]|=(b.charCodeAt(c/8)&255)<<(c%32)}return a}function binl2rstr(b){var a="";for(var c=0;c<b.length*32;c+=8){a+=String.fromCharCode((b[c>>5]>>>(c%32))&255)}return a}function binl_md5(p,k){p[k>>5]|=128<<((k)%32);p[(((k+64)>>>9)<<4)+14]=k;var o=1732584193;var n=-271733879;var m=-1732584194;var l=271733878;for(var g=0;g<p.length;g+=16){var j=o;var h=n;var f=m;var e=l;o=md5_ff(o,n,m,l,p[g+0],7,-680876936);l=md5_ff(l,o,n,m,p[g+1],12,-389564586);m=md5_ff(m,l,o,n,p[g+2],17,606105819);n=md5_ff(n,m,l,o,p[g+3],22,-1044525330);o=md5_ff(o,n,m,l,p[g+4],7,-176418897);l=md5_ff(l,o,n,m,p[g+5],12,1200080426);m=md5_ff(m,l,o,n,p[g+6],17,-1473231341);n=md5_ff(n,m,l,o,p[g+7],22,-45705983);o=md5_ff(o,n,m,l,p[g+8],7,1770035416);l=md5_ff(l,o,n,m,p[g+9],12,-1958414417);m=md5_ff(m,l,o,n,p[g+10],17,-42063);n=md5_ff(n,m,l,o,p[g+11],22,-1990404162);o=md5_ff(o,n,m,l,p[g+12],7,1804603682);l=md5_ff(l,o,n,m,p[g+13],12,-40341101);m=md5_ff(m,l,o,n,p[g+14],17,-1502002290);n=md5_ff(n,m,l,o,p[g+15],22,1236535329);o=md5_gg(o,n,m,l,p[g+1],5,-165796510);l=md5_gg(l,o,n,m,p[g+6],9,-1069501632);m=md5_gg(m,l,o,n,p[g+11],14,643717713);n=md5_gg(n,m,l,o,p[g+0],20,-373897302);o=md5_gg(o,n,m,l,p[g+5],5,-701558691);l=md5_gg(l,o,n,m,p[g+10],9,38016083);m=md5_gg(m,l,o,n,p[g+15],14,-660478335);n=md5_gg(n,m,l,o,p[g+4],20,-405537848);o=md5_gg(o,n,m,l,p[g+9],5,568446438);l=md5_gg(l,o,n,m,p[g+14],9,-1019803690);m=md5_gg(m,l,o,n,p[g+3],14,-187363961);n=md5_gg(n,m,l,o,p[g+8],20,1163531501);o=md5_gg(o,n,m,l,p[g+13],5,-1444681467);l=md5_gg(l,o,n,m,p[g+2],9,-51403784);m=md5_gg(m,l,o,n,p[g+7],14,1735328473);n=md5_gg(n,m,l,o,p[g+12],20,-1926607734);o=md5_hh(o,n,m,l,p[g+5],4,-378558);l=md5_hh(l,o,n,m,p[g+8],11,-2022574463);m=md5_hh(m,l,o,n,p[g+11],16,1839030562);n=md5_hh(n,m,l,o,p[g+14],23,-35309556);o=md5_hh(o,n,m,l,p[g+1],4,-1530992060);l=md5_hh(l,o,n,m,p[g+4],11,1272893353);m=md5_hh(m,l,o,n,p[g+7],16,-155497632);n=md5_hh(n,m,l,o,p[g+10],23,-1094730640);o=md5_hh(o,n,m,l,p[g+13],4,681279174);l=md5_hh(l,o,n,m,p[g+0],11,-358537222);m=md5_hh(m,l,o,n,p[g+3],16,-722521979);n=md5_hh(n,m,l,o,p[g+6],23,76029189);o=md5_hh(o,n,m,l,p[g+9],4,-640364487);l=md5_hh(l,o,n,m,p[g+12],11,-421815835);m=md5_hh(m,l,o,n,p[g+15],16,530742520);n=md5_hh(n,m,l,o,p[g+2],23,-995338651);o=md5_ii(o,n,m,l,p[g+0],6,-198630844);l=md5_ii(l,o,n,m,p[g+7],10,1126891415);m=md5_ii(m,l,o,n,p[g+14],15,-1416354905);n=md5_ii(n,m,l,o,p[g+5],21,-57434055);o=md5_ii(o,n,m,l,p[g+12],6,1700485571);l=md5_ii(l,o,n,m,p[g+3],10,-1894986606);m=md5_ii(m,l,o,n,p[g+10],15,-1051523);n=md5_ii(n,m,l,o,p[g+1],21,-2054922799);o=md5_ii(o,n,m,l,p[g+8],6,1873313359);l=md5_ii(l,o,n,m,p[g+15],10,-30611744);m=md5_ii(m,l,o,n,p[g+6],15,-1560198380);n=md5_ii(n,m,l,o,p[g+13],21,1309151649);o=md5_ii(o,n,m,l,p[g+4],6,-145523070);l=md5_ii(l,o,n,m,p[g+11],10,-1120210379);m=md5_ii(m,l,o,n,p[g+2],15,718787259);n=md5_ii(n,m,l,o,p[g+9],21,-343485551);o=safe_add(o,j);n=safe_add(n,h);m=safe_add(m,f);l=safe_add(l,e)}return Array(o,n,m,l)}function md5_cmn(h,e,d,c,g,f){return safe_add(bit_rol(safe_add(safe_add(e,h),safe_add(c,f)),g),d)}function md5_ff(g,f,k,j,e,i,h){return md5_cmn((f&k)|((~f)&j),g,f,e,i,h)}function md5_gg(g,f,k,j,e,i,h){return md5_cmn((f&j)|(k&(~j)),g,f,e,i,h)}function md5_hh(g,f,k,j,e,i,h){return md5_cmn(f^k^j,g,f,e,i,h)}function md5_ii(g,f,k,j,e,i,h){return md5_cmn(k^(f|(~j)),g,f,e,i,h)}function safe_add(a,d){var c=(a&65535)+(d&65535);var b=(a>>16)+(d>>16)+(c>>16);return(b<<16)|(c&65535)}function bit_rol(a,b){return(a<<b)|(a>>>(32-b))};