一道xss题目--intigriti-0422-XSS-Challenge-Write-up

news2024/11/16 0:17:24

目录

进入挑战

 js代码

代码分析

构造payload

​编辑

结果


进入挑战

Intigriti April Challenge题目地址

打开题目后,找到对应页面的js代码,寻找一下我们用户可控的点

 js代码

<!DOCTYPE html>
<html lang="en"><head>
<meta http-equiv="content-type" content="text/html; charset=UTF-8">
  <meta charset="utf-8">
  <meta name="robots" content="noindex">
  <meta name="viewport" content="width=device-width, initial-scale=1.0">
  <meta http-equiv="X-UA-Compatible" content="ie=edge">
  <meta http-equiv="Content-Security-Policy" content="default-src 'none'; style-src 'self' 'unsafe-inline'; img-src 'self' data:; font-src 'self'; script-src 'self' 'unsafe-inline';">

  <title>Window Maker</title>
  <!-- downloaded from https://unpkg.com/xp.css@0.3.0/dist/XP.css -->
  <link rel="stylesheet" href="Window%20Maker_files/XP.css">
  <style>
    body {
      margin: 0;
      padding: 0;
      background-image: url("bg.jpg");
      background-position: center;
      background-repeat: no-repeat;
      font-size: 18px;
      min-height: 100vh;
    }
    main {
      max-width: 750px;
      margin: 100px auto;
    }
    .window-body {
      margin: 16px;
      font-size: 14px;
    }
    label {
      font-size: 13px;
    }

    @media (min-height: 980px), (min-width: 1900px) {
      body {
        background-size: 100% 100%;
      }
    }
  </style>
</head>

<body>
    <!-- downloaded from https://unpkg.com/mithril@2.0.4/mithril.js -->
    <script src="Window%20Maker_files/mithril.js"></script>
    <script>
      (function(){
        const App = {
          view: function() {
            return m("div", {class: "window"}, [
              m(TitleBar),
              m(WindowBody),
              m(StatusBar)
            ])
          }
        }

        const TitleBar = {
          view: function(vnode) {
            const options = vnode.attrs.options || ['min', 'max', 'close']
            const name = vnode.attrs.name || "Window Maker"
            return m("div", {class: "title-bar"}, [
              m("div", {class: "title-bar-text"}, String(name)),
              m("div", {class: "title-bar-controls"}, [
                options.includes('min') && m("button", {'aria-label': 'Minimize'}),
                options.includes('max') && m("button", {'aria-label': 'Maximize'}),
                options.includes('close') && m("button", {'aria-label': 'Close'}),
              ]),
            ])
          }
        }

        const WindowBody = {
          view: function() {
            return m("div", {class: "window-body"}, [
              m("p", [
                "Do you miss these looks and feels? We can help!",
                m("br"),
                "Window Maker is a website to help people build their own UI in 3 minutes!"
              ]),
              m("p", [
                m(InputWindowName),
                m(InputWindowContent),
                m("br"),
                m(InputToolbar),
                m("br"),
                m(InputStatusBar)
              ]),
              m("button", {onclick: function() {
                const windowName = document.querySelector('#win-name').value
                const windowContent = document.querySelector('#win-content').value
                const toolbar = Array.from(document.querySelectorAll('input[type=checkbox]:checked')).map(item => item.value)
                const showStatus = document.querySelector('#radio-yes').checked
                const config = {
                  'window-name': windowName,
                  'window-content': windowContent,
                  'window-toolbar': toolbar,
                  'window-statusbar': showStatus
                }
                const qs = m.buildQueryString({
                  config
                })
                window.location.search = '?' + qs
              }}, "generate")
            ])
          }
        } 

        const InputWindowName = {
          view: function(vnode) {
            return m("div", {class: "field-row-stacked"}, [
              m("label", {for: 'win-name' }, 'Window name'),
              m("input", {id: 'win-name', type: 'text' }),
            ])
          }
        }

        const InputWindowContent = {
          view: function(vnode) {
            return m("div", {class: "field-row-stacked"}, [
              m("label", {for: 'win-content' }, 'Window content(plaintext only)'),
              m("textarea", {id: 'win-content', rows: '8' }),
            ])
          }
        }

        const InputToolbar = {
          view: function(vnode) {
            return m("div", [
              m("div", {class: "field-row"}, [
                m("label", "Toolbar"),
              ]),
              m(Checkbox, { id: "toolbar-min", value: "min" }),
              m(Checkbox, { id: "toolbar-max", value: "max" }),
              m(Checkbox, { id: "toolbar-close", value: "close" }),
            ])
          }
        }

        const Checkbox = {
          view: function(vnode) {
            return m("div", {class: "field-row"}, [
              m("input", {id: String(vnode.attrs.id), type: 'checkbox', value: String(vnode.attrs.value) }),
              m("label", {for: String(vnode.attrs.id) }, String(vnode.attrs.value)),
            ])
          }
        }

        const InputStatusBar = {
          view: function() {
            return m("div", [
              m("div", {class: "field-row"}, [
                m("label", "Status bar"),
              ]),
              m(RadioButton, { id: "radio-yes", value: "Yes" }),
              m(RadioButton, { id: "radio-no", value: "No" }),
            ])
          }
        }

        const RadioButton = {
          view: function(vnode) {
            return m("div", {class: "field-row"}, [
              m("input", {id: String(vnode.attrs.id), type: 'radio', name: 'status-radio' }),
              m("label", {for: String(vnode.attrs.id) }, String(vnode.attrs.value)),
            ])
          }
        }

        const StatusBar = {
          view: function() {
            return m("div", {class: "status-bar"}, [
              m("p", {class: "status-bar-field"}, "Press F1 for help"),
              m("p", {class: "status-bar-field"}, "Powered by XP.css and Mithril.js"),
              m("p", {class: "status-bar-field"}, "CPU Usage: 32%"),
            ])
          }
        }

        const CustomizedApp = {
          view: function(vnode) {
            return m("div", {class: "window"}, [
              m(TitleBar, {name: vnode.attrs.name, options: vnode.attrs.options}),
              m("div", {class: "window-body"},[
                String(vnode.attrs.content)
              ]),
               vnode.attrs.status && m(StatusBar)
            ])
          }
        }

        function main() {
          const qs = m.parseQueryString(location.search)

          let appConfig = Object.create(null)
          appConfig["version"] = 1337
          appConfig["mode"] = "production"
          appConfig["window-name"] = "Window"
          appConfig["window-content"] = "default content"
          appConfig["window-toolbar"] = ["close"]
          appConfig["window-statusbar"] = false
          appConfig["customMode"] = false

          if (qs.config) {
            merge(appConfig, qs.config)
            appConfig["customMode"] = true
          }

          let devSettings = Object.create(null)
          devSettings["root"] = document.createElement('main')
          devSettings["isDebug"] = false
          devSettings["location"] = 'challenge-0422.intigriti.io'
          devSettings["isTestHostOrPort"] = false

          if (checkHost()) {
            devSettings["isTestHostOrPort"] = true
            merge(devSettings, qs.settings)
          }

          if (devSettings["isTestHostOrPort"] || devSettings["isDebug"]) {
            console.log('appConfig', appConfig)
            console.log('devSettings', devSettings)
          }

          if (!appConfig["customMode"]) {
            m.mount(devSettings.root, App)
          } else {
            m.mount(devSettings.root, {view: function() {
              return m(CustomizedApp, {
                name: appConfig["window-name"],
                content: appConfig["window-content"] ,
                options: appConfig["window-toolbar"],
                status: appConfig["window-statusbar"]
              })
            }})
          }

          document.body.appendChild(devSettings.root)
        }

        function checkHost() {
          const temp = location.host.split(':')
          const hostname = temp[0]
          const port = Number(temp[1]) || 443
          return hostname === 'localhost' || port === 8080
        }

        function isPrimitive(n) {
          return n === null || n === undefined || typeof n === 'string' || typeof n === 'boolean' || typeof n === 'number'
        }

        function merge(target, source) {
          let protectedKeys = ['__proto__', "mode", "version", "location", "src", "data", "m"]

          for(let key in source) {
            if (protectedKeys.includes(key)) continue

            if (isPrimitive(target[key])) {
              target[key] = sanitize(source[key])
            } else {
              merge(target[key], source[key])
            }
          }
        }
        function sanitize(data) {
          if (typeof data !== 'string') return data
          return data.replace(/[<>%&\$\s\\]/g, '_').replace(/script/gi, '_')
        }

        main()
      })()
    </script>

</body></html>

代码分析

在这里你输入什么接的就是什么,比如输入的是config然后接的就是qs.config,所以我们用户可控的点在这里

然后由于有merge所以我们就可以尝试控制appconfig这个变量,然后可以尝试原型链污染了。当然后也有别的merge,如下图,这里有qs.settings这个属性然后就可以尝试控制decSettings,但是我们的注入点在那个上面呢,不急接着往下看

再往下看一下,这里有一个关于decSettings的插入节点,在这里我们可以尝试进行dom破坏

然后我们想要触发document.body.appendChild(devSettings.root)的话就要进入这里

然后就要进入这里

构造payload

我们尝试构造一个原型对象原型对象有一个属性名为1的属性,然后temp[1]又是空,我们访问1的时候就可以取出1里面的内容。然后我们构造的时候需要满⾜对象类型为Array,且可被merge的参数,满⾜这样条件的只有

我们尝试构造下面代码,这里不用__proto__是因为被过滤掉了

appConfig["window-toolbar"][constructor][prototype]['1']=8080

现在我们进入了if (checkHost())我们接下来就可以尝试对settings进行修改,构造下面代码

settings[root][ownerDocument][body][children][1][outerHTML]
[1]=%3Csvg%20onload%3Dalert(1)%3E

setting为什么下面不直接覆盖root,是因为后面root会被main会被覆盖且后面有过滤,如下图

然后我们需要绕过一下。绕过的方法就看root了,root下面有很多方法,我们想办法找到decSetting.root上面的元素,如下面

然后找到ownerDocument

 然后找到body

然后找到]children[1]

然后找到这里[outerHTML][1]

一层一层往下取值最后插入这样在经过isprimitive时就会被判断是一个数组就不会进入isprimitive

结果

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2071902.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

阵列信号处理2_阵列信号最优处理常用准则(CSDN_20240825)

目录 最小均方误差&#xff08;Minimum Square Error&#xff0c;MSE&#xff09;准则 最大信噪比&#xff08;Maximum Signal Noise Ratio&#xff0c;MSNR&#xff09;准则 极大似然&#xff08;Maximum Likehood, ML&#xff09;准则 最小方差无损响应&#xff08;Minim…

移动应用平台,企业移动门户就选WorkPlus

随着移动设备的普及和移动办公的兴起&#xff0c;企业需要一个高效可靠的移动应用平台来加强内部沟通、协作和信息管理。在众多的移动应用平台中&#xff0c;WorkPlus作为一款专为企业打造的移动门户&#xff0c;凭借其稳定性、功能丰富和易用性成为了企业移动门户的首选。 一、…

【LVGL-下拉列表部件 lv_dropdown】

LVGL-下拉列表部件 lv_dropdown ■ LVGL-下拉列表部件 lv_dropdown■ 下拉列表部件的组成■ 添加选项■ 获取当前选中的选项■ 设置列表展开方向■ 设置下拉列表图标■ 设置列表常显文本■ 打开、开闭下拉列表■ 下拉列表部件的 API 函数 ■ LVGL-下拉列表部件 lv_dropdown ■ …

STM32学习记录-05 -3-TIM输入捕获

1 输入捕获简介 IC&#xff08;Input Capture&#xff09;输入捕获 输入捕获模式下&#xff0c;当通道输入引脚出现指定电平跳变时&#xff0c;当前CNT的值将被锁存到CCR中&#xff0c;可用于测量PWM波形的频率、占空比、脉冲间隔、电平持续时间等参数 每个高级定时器和通用…

FME批处理:WorkspaceRunner转换器

1、创建一个待处理工程 注意&#xff1a; Reader设置&#xff1a; Writer设置&#xff1a; 2、创建另外一个**批处理工程**&#xff0c;加入workspaceRunner并设置参数 workspaceRunner设置 directory and file pathnames设置

认知杂谈23

今天分享 有人说的一段争议性的话 I I 《忙碌不停&#xff0c;成长迷失》 现代生活啊&#xff0c;就跟一场一直转个不停的旋转木马似的。你每天都被各种小事儿缠得死死的&#xff0c;从大清早一睁开眼&#xff0c;一直到晚上要睡觉了&#xff0c;几乎就找不到一点能停下来喘…

BC156 牛牛的数组匹配(c语言)

1./描述 //牛牛刚学会数组不久&#xff0c;他拿到两个数组 a 和 b&#xff0c;询问 b 的哪一段连续子数组之和与数组 a 之和最接近。 //如果有多个子数组之和同样接近&#xff0c;输出起始点最靠左的数组。 //输入描述&#xff1a; //第一行输入两个正整数 n 和 m &#xff0c;…

NLP从零开始------12. 关于前十一章补充(英文分词)

相较于基础篇章&#xff0c;这一部分相较于基础篇减少了很多算法推导&#xff0c;多了很多代码实现。 1.英文词规范化 英文词规范化一般分为标准化缩写,大小写相互转化&#xff0c;动词目态转化等。 1.1 大小写折叠 大小写折叠( casefolding) 是将所有的英文大写字母转化成小…

开发高质量PDF应用的不二选择:PdfiumViewer库详细解析

1. PdfiumViewer库简介 PdfiumViewer是一款基于谷歌开源PDF渲染引擎PDFium的.NET库&#xff0c;主要用于在Windows应用程序中显示和处理PDF文档。PdfiumViewer提供了多种API和控件&#xff0c;使得开发者可以轻松地将PDF文档嵌入到其应用程序中。同时&#xff0c;PdfiumViewer…

利用 OCR 和强大的 GPT-4o 迷你模型对收据进行信息提取

在本文中&#xff0c;我将向您展示如何从收据中提取信息&#xff0c;并提供收据的简单图像。首先&#xff0c;我们将利用 OCR 从收据中提取信息。然后&#xff0c;此信息将发送到 GPT-4o 迷你模型进行信息提取。我在这个项目中的目标是开发一个应用程序&#xff0c;只需拍摄收据…

【排序算法】八大排序(下)(c语言实现)(附源码)

&#x1f31f;&#x1f31f;作者主页&#xff1a;ephemerals__ &#x1f31f;&#x1f31f;所属专栏&#xff1a;算法 目录 前言 测试数据和交换函数 五、堆排序 六、快速排序 1.hoare版本 2.挖坑法 3.lomoto版本 4.快速排序的非递归实现 5.快速排序性能总结 七、归…

手机mkv转换mp4:轻松实现视频格式兼容

如今手机已成为我们日常生活中不可或缺的伴侣&#xff0c;而视频文件则是我们享受娱乐、获取信息的重要来源。然而&#xff0c;由于不同设备和平台对视频格式的支持各有不同&#xff0c;我们有时会遇到无法在手机上播放某些视频文件的问题。 mkv是一种常见的视频格式&#xff…

android使用YOLOV8数据返回到JAVA方法(JAVA)

一、下载扩展文件(最耗时,所以放第一步) 1.opencv下载 1)官网:Releases - OpenCV 2)下载最新版本的android包 2.NCNN下载 1)NCNN下载地址(20220420版本):https://github.com/Tencent/ncnn/releases/download/20220420/ncnn-20220420-android-vulkan.zip 3.在你的…

倍内菲新品发布揭示宠物营养新纪元,引领行业保驾护航

2024年8月21日&#xff0c;伴随着第26届亚洲宠物展览会的揭幕&#xff0c;宠物主粮领军品牌倍内菲在展会首日举行了一场意义深远的新品发布会&#xff0c;重磅推出两款革命性新品——鲜肉烘焙系列与至护烘焙系列&#xff0c;不仅是对宠物进阶营养需求的深刻洞察&#xff0c;更是…

【信创】统信UOS打包工具介绍与使用教程

原文链接&#xff1a;【信创】统信UOS打包工具介绍与使用教程 Hello&#xff0c;大家好啊&#xff01;今天给大家带来一篇关于统信UOS桌面操作系统上的UOS打包工具介绍与使用的文章。UOS打包工具是一款专为统信UOS系统开发的应用程序打包工具&#xff0c;旨在帮助开发者轻松创建…

Vue.js:解锁前端开发的快速入门之旅

标题&#xff1a;《Vue.js&#xff1a;解锁前端开发的快速入门之旅》 在日新月异的Web开发领域中&#xff0c;Vue.js以其简洁、灵活和高效的特点&#xff0c;迅速成为前端开发者们的宠儿。对于初学者而言&#xff0c;Vue.js不仅是一个易于上手的框架&#xff0c;更是开启现代前…

python动画:颜色(color)能接受的[manim_colors]

Manim_colors指的是Manim动画引擎中全局命名空间中包含的一组颜色。这些颜色构成了Manim默认的颜色空间。通过使用manim_colors&#xff0c;动画师和创作者可以轻松地访问和应用各种颜色到他们的动画中&#xff0c;而无需单独定义它们。这个特性简化了动画制作的过程&#xff0…

张宇线代9讲啃不动,换李永乐来得及吗?

快9月了&#xff0c;很多同学在后台留言说&#xff0c;跟宇哥的线代&#xff0c;但是啃不动&#xff0c;接下来该怎么办&#xff0c;换李永乐来的急吗&#xff1f; 让我来认真分析一下&#xff01; 张宇线代9讲为什么这么难啃&#xff1f; 25版张宇线代改版&#xff0c;线代的…

二叉树【2.5】代码专项

目录 醍醐灌顶——node* root 和node* &root作为参数的区别 return value&#xff1a; 写一个前序遍历的&#xff08;使用指针&#xff09; 中序遍历&#xff0c;只改动了preorder&#xff0c;只调换了一行 后序 层序&#xff08;使用bfs&#xff09;&#xff0c;新建队…

如何保证支付服务和交易服务订单状态一致?

消息传给消费者&#xff0c;消费者自己弄丢 业务幂等 所有的业务都应该保证幂等性&#xff0c; 如何保障业务幂等性 非幂等业务表单重复提交&#xff0c;在进入表单之前生成唯一标识&#xff0c;未token&#xff0c;携带token进行请求&#xff0c;执行表单提交&#xff0c;把…