Flash XSS 漏洞详解 根治的好办法

本文的目的是深层次的分析Flash的ExternalInterface的XSS漏洞,并提出比较有效的解决方案。

首先,我们看看什么情况下,会出现XSS。

第一种情况:

     把flashvars传入的参数(或者其他能被别人控制的方式)当ExternalInterface.call的第一个参数

package
{
        import flash.display.Sprite;
        import flash.external.ExternalInterface;
       
        public class XSSTest extends Sprite
       {
               public function XSSTest()
              {
                      var jsFunction:String = loaderInfo.parameters.jsFunction;
                      var param:String = "abc";
                     ExternalInterface.call(jsFunction, param);
              }
       }
}

     注意,这里通过flashvars传递了一个参数,是js的函数。这种方式比较常见,swf可以做成通用,放到不同的业务中使用,每次只需要传入对应的js函数即可。但是,这里就存在漏洞了。

     在浏览器中,构造url: XSSTest.swf?jsFunction=alert(/XSS/),访问swf,并以get参数的形式传入flashvars,结果,造成了

甚至更狠一点,jsFunction=function(){alert(1);alert(2);}。。。

当然,这么恶作剧alert一下,貌似对小白用户没什么损失,但如果在function内调用这个域名的CGI,就能带来很大的惊喜了~~~因为这里能获取到对应的cookie,时间有限,具体攻击的方式,这里不多说。

第二种情况:

     把flashvars传入的参数(或者其他能被别人控制的方式)当ExternalInterface.call的第二和第三个参数

 这次,我们使用这段代码:

   public function XSSTest()
   {
          var param:String = loaderInfo.parameters.param;
          ExternalInterface.call("console.log", param);
   }

     这个方式也许没有这么简单进行XSS,但对于黑客来说,还是有办法的。

在IE8下调试模式下,我们可以看到ExternalInterface的代码:

正常情况下,Flash player会生成这样的代码:

try { __flash__toXML(console.log("good" )) ; } catch (e) { "<undefined/>"; }

对比自己写的as代码和生成的这段js代码,可以猜测,Flash player是以一种简单的拼接字符串的方式实现的。

稍稍做个小把戏,结果就可以注入代码执行了。

是不是很神奇?怎么做到的呢?为什么url稍稍变化可以达到这样呢。我们看看现在的js代码:

try { __flash__toXML(console.log("\\" ));alert(/XSS/);}catch(e){} //")) ; } catch (e) { "<undefined/>"; }

正好跟原来的双引号对上了,结果,最后的catch也被替换了。。。也就是说,黑客可以写自己的函数了,想怎么执行都可以了。。。

至于为什么这里双引号对上了,可以简单猜测flash遇到字符串中有双引号的时候,只是简单的以  \"  方式打印成js代码,但如果用户再恶意拼一个\,就负负得正了。。。

(这里__flash__toXML的代码并不是关键点了,所以将在文章最后再列出)

第三种情况:

     没有对swf Object的id没有过滤

     页面加载Flash,我们需要设定Object或者embed的id,否则ExternalInterface会失效。而这个地方,也会被黑客利用。

     我们看看实际执行的代码:

try { document.getElementById("XSSTest" ).SetReturnValue(__flash__toXML(alert( null)) ); } catch (e) { document.getElementById("XSSTest" ).SetReturnValue("<undefined/>"); }     

     看到这里,应该发现跟上边说的第二种情况很类似,黑客可以通过修改了Object id,恶意闭合双引号,达到目的。

接下来,简洁的总结一下怎么防XSS。

对于第一和第三种情况,我们应该对字符进行过滤,例如用以下的两个函数:

               public static function checkJsFunctionValid(functionName:String):Boolean
              {
                      var reg:RegExp = /^[a-zA-Z0-9_\.]+$/;
                      return reg.test(functionName);
              }          
    
               public static function checkObjectIdValid():Boolean
              {
                      if (ExternalInterface.available)
                     {
                            var objectId:String = ExternalInterface.objectID;
                            if (!objectId || (objectId == objectId.replace(/[^0-9a-zA-Z_]/g , "")))
                                   return true;
                            else
                                   return false;
                     }
                      return true;
              }
 

对于第二种情况,我们应该尽量避免这样跟js传递数据,但如果实在无法避免。可以用这样的方式转义字符串:

str.replace( /[\"\\]/g , function(d:String, b:*, c:*){ return '\\' + d.charCodeAt(0).toString(8); });

简单解释一下,这里把双引号和反斜杠这样比较敏感的字符,替换为转义表示。再输出成js代码时,正好又还原回去了。

例如:

\\\"\\\"})));}catch(e){alert(/xss/);}// 变成了     \134\42\134\42})));}catch(e){alert(/xss/);}//

附带额外的在IE8 开发工具中抓获到的代码:

function __flash__arrayToXML(obj) {
         var s = "<array>" ;
         for (var i=0; i<obj.length; i++) {
                s += "<property id=\"" + i + "\">" + __flash__toXML(obj[i]) + "</property>";
        }
         return s+"</array>" ;
}
function __flash__argumentsToXML(obj,index) {
         var s = "<arguments>" ;
         for (var i=index; i<obj.length; i++) {
                s += __flash__toXML(obj[i]);
        }
         return s+"</arguments>" ;
}
function __flash__objectToXML(obj) {
         var s = "<object>" ;
         for (var prop in obj) {
                s += "<property id=\"" + prop + "\">" + __flash__toXML(obj[prop]) + "</property>" ;
        }
         return s+"</object>" ;
}
function __flash__escapeXML(s) {
         return s.replace(/&/g, "&amp;" ).replace(/</g, "&lt;").replace(/>/g, "&gt;" ).replace(/"/g, "&quot; ").replace(/'/g, "&apos;");
}
function __flash__toXML(value) {
   var type = typeof(value);
         if (type == "string" ) {
                 return "<string>" + __flash__escapeXML(value) + "</string>";
        } else if (type == "undefined") {
        return "<undefined/>" ;
        } else if (type == "number") {
        return "<number>" + value + "</number>";
        } else if (value == null) {
        return "<null/>" ;
        } else if (type == "boolean") {
        return value ? "<true/>" : "<false/>";
        } else if (value instanceof Date) {
        return "<date>" + value.getTime() + "</date>";
   } else if (value instanceof Array) {
       return __flash__arrayToXML(value);
   } else if (type == "object") {
       return __flash__objectToXML(value);
   } else {
            return "<null/>" ; //???
        }
}
function __flash__addCallback(instance, name) {
  instance[name] = function () {
    return eval(instance.CallFunction("<invoke name=\"" +name+"\" returntype=\"javascript\">" + __flash__argumentsToXML(arguments,0) + "</invoke>" ));
  }
}
function __flash__removeCallback(instance, name) {
  instance[name] = null;
}
  

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏Coding01

浅谈 Laravel Collections

这两天看了两本书《Laravel Collections Unraveled》和 《Refactoring to Collections》。

1002
来自专栏H2Cloud

python的解释器spython介绍

Python解释器spython介绍 简介   出于个人爱好和某种需求,我再16年对python的解释器产生了浓厚兴趣,并且下定决心重新实现一个版本。我个人再游...

3355
来自专栏Kevin-ZhangCG

二十三种设计模式总结

定义:Ensure a class has only one instance, and provide a global point of access to...

782
来自专栏腾讯Bugly的专栏

那些年,我们一起写过的“单例模式”

本文来自:“天天P图攻城狮”公众号(ttpic_dev) 题记 度娘上对设计模式(Design pattern)的定义是:“一套被反复使用、多数人知晓的、经过分...

4024
来自专栏Web 开发

JavaScript Cookbook 2nd 之 Function

昨晚翻了一下,虽然都是一些旧知识,不过深入下去对照着其他资料一起看,还是能发现一些有意思的地方。

690
来自专栏技术墨客

Spring核心——设计模式与IoC 原

“Spring”——每一个Javaer开发者都绕不开的字眼,从21世纪第一个十年国内异常活跃的SSH框架,到现在以Spring Boot作为入口粘合了各种应用。...

701
来自专栏liulun

Nim教程【五】

这是国内第一个关于Nim的系列教程 先说废话 业内的人认为能够直接操作系统硬件的语言才称得上系统级的编程语言 常见的系统级编程语言有:汇编、C、C++、D、GO...

1878
来自专栏pangguoming

完全掌握Android Data Binding

编辑推荐:稀土掘金,这是一个针对技术开发者的一个应用,你可以在掘金上获取最新最优质的技术干货,不仅仅是Android知识、前端、后端以至于产品和设计都有涉猎,想...

4877
来自专栏微信终端开发团队的专栏

ART 下的方法内联策略及其对 Android 热修复方案的影响分析

Android N 后对内联的新发现,似乎再一次认证了“ 热补丁不是请客吃饭 ”这句话。本文主要介绍了 ART 下的方法内联策略及其对 Android 热修复方...

1820
来自专栏后端沉思录

回调函数

什么是回调函数,上面的问题说的是不是很空洞,不是太形象,下面是知乎上的一位网友给的答案:

722

扫码关注云+社区