黑客派自动签到脚本

本贴最后更新于 1795 天前,其中的信息可能已经时移世异

一个已知问题:脚本会长时间卡住(大概率).多次运行脚本仍无法继续.
问题原因:由于 Invoke-WebRequest 命令使用的是 IE 内核进行 DOM 解析,获取网页之后,会进行 DOM 解析,才能进一步使用 ParsedHtml 方法中的 getElementsByClassName 等网页方法,脚本就卡在了这个地方.
具体原因未知,可能是该站完全放弃了 IE,未进行兼容导致的.

*.ps1 文件

<#

.Synopsis
   自动签到的PowerShell脚本
.DESCRIPTION
   自动签到
.EXAMPLE
   自动签到程序.ps1 [-Task]
   自动签到程序.ps1 [-DelUser] <动态值>
.EXAMPLE
   自动签到程序.ps1 [-WebName] <动态值> [-UserName] <String> [-PassWord] <String>

.INPUTS
   到此 cmdlet 的输入(如果有)
.OUTPUTS
   来自此 cmdlet 的输出(如果有)
.NOTES
   一般注释
.COMPONENT
   此 cmdlet 所属的组件
.ROLE
   此 cmdlet 所属的角色
.FUNCTIONALITY
   最准确描述此 cmdlet 的功能
#>
# 该脚本声明必须在开头,其他函数及命令之前 
# 用于手动添加账号密码
[CmdletBinding(DefaultParameterSetName)] # 设置默认参数组为空
Param(

    [Parameter(ParameterSetName="adds",Position=1,Mandatory=$true)]
    [ValidateNotNullOrEmpty()]
    [string]$UserName,

    [Parameter(ParameterSetName="adds",Position=2,Mandatory=$true)]
    [ValidateNotNullOrEmpty()]
    [string]$PassWord,

    [Parameter(ParameterSetName="Task")]
    [switch]$Task,

    [Parameter(DontShow,ParameterSetName="None")]
    [switch]$None
)

#动态参数
DynamicParam
{
    
    $filepath="$($PSScriptRoot)\cookie.user"
    if (-not (Test-Path -Path $filepath))
    {
        New-Item -ItemType "file" -Path $filepath | Out-Null
    }
    #主
    $PD = New-Object -Type System.Management.Automation.RuntimeDefinedParameterDictionary
    $modName=(Get-ChildItem -Path $($PSScriptRoot) -Name -Include 签到-*.psm1) -replace '签到-','' -replace '.psm1',''
    #$modName=$mods 
    #当有模块时,动态添加模块列表
    if ($modName.Count){
        
        #容器
        $a = New-Object -TypeName System.Collections.ObjectModel.Collection[System.Attribute]
        #属性
        $b = New-Object System.Management.Automation.ParameterAttribute
        $b.Mandatory = $true
        $b.ParameterSetName='adds'
        $b.Position=0
        $b.HelpMessage='可选参数:'+$modName
        $a.Add($b)
        #参数值
        $c=New-Object System.Management.Automation.ValidateSetAttribute($modName)
        $a.Add($c)

        $d = New-Object -TypeName System.Management.Automation.RuntimeDefinedParameter('WebName',[string], $a)
        $PD.Add('WebName', $d)
    }
    #账号删除功能
    [PSObject].Assembly.GetType('System.Management.Automation.TypeAccelerators')::Add('SA', [System.Collections.ArrayList])
    $pp=(Get-Content $filepath) | Convertfrom-Json
    $ar = New-Object -TypeName SA
    foreach ($s in $pp)
    {

        $ar.Add($s.WebName+":"+$s.UserName)|Out-Null
    }
    if ($ar.Count) {

        #容器
        $a = New-Object -TypeName System.Collections.ObjectModel.Collection[System.Attribute]
        #属性
        $b = New-Object System.Management.Automation.ParameterAttribute
        $b.Mandatory = $true
        $b.ParameterSetName='Deleted'
        $a.Add($b)

        #参数值
        $c=New-Object System.Management.Automation.ValidateSetAttribute($ar)
        $a.Add($c)

        $d = New-Object -TypeName System.Management.Automation.RuntimeDefinedParameter('DelUser',[string], $a)
        $PD.Add('DelUser',$d)
    }
    return $PD
}

begin
{
    $T=1
    #cookie存储文件位置
    $filepath="$($PSScriptRoot)\cookie.user"
    if (-not (Test-Path -Path $filepath))
    {
        New-Item -ItemType "file" -Path $filepath | Out-Null
    }
    #签到结果记录路径文件位置
    $LogPath="{0}\Log\" -f $PSScriptRoot
    if (-not (Test-Path -Path $LogPath))
    {
        New-Item -ItemType "directory" -Path $LogPath | Out-Null
    }
    $global:Log="{0}{1:yyyyMMdd}.log" -f $LogPath,(get-date)
    #可用签到模板
    $modName=(Get-ChildItem -Path $($PSScriptRoot) -Name -Include 签到-*.psm1) -replace '签到-','' -replace '.psm1',''
    #所有用户信息
    [array]$pp=(Get-Content $filepath) | Convertfrom-Json
    #添加计划任务
    if ($task)
    {
        # 任务操作   程序 ,工作目录,参数
        $AA = New-ScheduledTaskAction -Execute "PowerShell"  -WorkingDirectory "$($PSScriptRoot)" -Argument "$($Script:MyInvocation.InvocationName)"
        #$BB = New-ScheduledTaskAction -Execute "PowerShell"  -WorkingDirectory "$($PSScriptRoot)" -Argument "-WindowStyle Hidden -Command `"& { Get-WinEvent -FilterHashtable @{LogName='论坛签到';StartTime=(Get-Date -Hour 0 -Minute 0 -Second 1)} | sort  TimeCreated -Descending| SELECT-Object TimeCreated,id,leveldisplayname,ProviderName,message | Out-GridView -Wait -Title 今日签到记录}`""
        $tem=Get-ScheduledTask -TaskName '论坛自动签到任务' -ErrorAction SilentlyContinue -WarningAction SilentlyContinue
        if ($tem)
        {
            if (($tem.Actions.Arguments -eq $AA.Arguments ) -and ($tem.Actions.WorkingDirectory -eq $AA.WorkingDirectory))
            {
                Write-Warning '该计划任务已存在'
            }
        }
        else
        {
            # 触发器    早上6点到晚上23点
            $T = New-ScheduledTaskTrigger -Daily -At 6:0:0  -RandomDelay 17:0:0          
            # 创建者  当前用户名
            $P = New-ScheduledTaskPrincipal "$env:userdomain\$env:username" 
            # 设置集   各种相关设置
            $S = New-ScheduledTaskSettingsSet -AllowStartIfOnBatteries -WakeToRun -Priority 4 -DontStopIfGoingOnBatteries -DontStopOnIdleEnd -RunOnlyIfNetworkAvailable -MultipleInstances IgnoreNew -StartWhenAvailable -Compatibility win8  -ExecutionTimeLimit 0   -DisallowHardTerminate       
            # 新任务   最后是任务描述  DisallowStartIfOnBatteries
            $D = New-ScheduledTask -Action $AA -Principal $P -Trigger $T -Settings $S  -Description "论坛自动签到脚本"
            # 注册任务     
            Register-ScheduledTask -TaskName "论坛自动签到任务" -InputObject $D |Out-Null
            Write-Host '计划任务已经添加'
        }
        return
    }
    #删除账号
    if ($PSBoundParameters.DelUser)
    {
        $DU=$PSBoundParameters.DelUser -split ":"
        Remove-Item $filepath
        foreach ($s in $pp)
        {
            if (-not(($s.WebName -eq $DU[0]) -and ($s.UserName -eq $DU[1])))
            {
                ConvertTo-Json -Compress -InputObject $s |  Out-File -append $filepath
            }
        }
        return
    }
    "签到脚本开始运行"
    #设置UserAgent
    enum 浏览器
    {
        Chrome
        FireFox
        InternetExplorer
        Opera
        Safari
    }
    function Get-UA ([浏览器]$ua)
    {
        return [Microsoft.PowerShell.Commands.PSUserAgent]::[浏览器]$ua
    }
    $global:UserAgents=Get-UA('Chrome')

    #写入到文件
    function global:Write-file ($Account){
        ConvertTo-Json -Compress -InputObject $Account |  Out-File -append $filepath
    }

    #写入到日志文件
    function global:Write-Log ($Account){
        Out-File -append $Log -InputObject $Account.ToString()
    }

    function global:HttpRest ($S)
    {
        Start-Sleep -Seconds $T
        $b=try{
            Invoke-RestMethod  @s
        }catch{
            $_.Exception.Response
        }
        return $b
    }

    function global:HttpWeb ($S)
    {
        Start-Sleep -Seconds $T
        $b=try{
            Invoke-WebRequest  @s
        }catch{
            $_.Exception.Response
        }
        return $b
    }
    #进行url编码
    function global:UrlEncode ([string]$pa)
    {
        return [uri]::EscapeUriString($pa)
    }

    #添加cookie
    function global:AddCookie($Tok,$WebHost)
    {
        # 创建cookie 数据包
        $Loadion = New-Object Microsoft.PowerShell.Commands.WebRequestSession
        $y=($Tok| Get-Member -MemberType NoteProperty ).Name
        # 循环添加所有 cookie
        foreach ($tk in $y)
        {
            $cookie = New-Object System.Net.Cookie($tk,$Tok.$tk,"",$WebHost)
            $Loadion.Cookies.Add($cookie)
        }
        return $Loadion
    }
    #转换cookie为散列表
    function global:CookieToHashTable ([Microsoft.PowerShell.Commands.WebRequestSession]$cookie,$uri)
    {
        $d=@{}
        $i=$cookie.Cookies.GetCookieHeader($uri)
        $i -split '; '|%{
            $temp=$_ -split '='
            $d.Add($temp[0],$temp[1])
        }
        return $d
    }
    #计算MD5
    function global:MD5
    {
         param
        (
            [parameter(mandatory=$true)]
            [ValidateNotNullOrEmpty()]
            [String] $String
        )
        #希哈计算
        $StringBuilder = New-Object System.Text.StringBuilder
        [System.Security.Cryptography.HashAlgorithm]::Create("MD5").ComputeHash([System.Text.Encoding]::UTF8.GetBytes($String))|%{
            [Void]$StringBuilder.Append($_.ToString("x2"))
        }
        return $StringBuilder.ToString()
    }

    $color=@{
        ForegroundColor='Black'
        BackgroundColor='White'
        NoNewline=$true
    }

    Write-Host -NoNewline "当前可用签到模板: "
    foreach ($item in $modName)
    {
        Write-Host "$item" @color
        Write-Host " " -NoNewline
        import-module "$($PSScriptRoot)\签到-$($item).psm1" -Prefix $item -Force
    }
    Write-Host ''
    #$pp=(Get-Content $filepath) | Convertfrom-Json
    Write-Host -NoNewline "共有 "
    Write-Host -Object $pp.Count  @color
    Write-Host " 个账号`n"

    #通过命令行添加新账号
    if ($PSBoundParameters.Count -eq 3)
    {
        foreach ($item in $pp)
        { #通过遍历,如果输入的用户名已存在,且与网站都吻合,则提示该用户已存在
            if (($item.UserName -eq $PSBoundParameters.UserName) -and  ($item.WebName -eq $PSBoundParameters.WebName))        
            { 
                Write-Warning  "该用户已存在"
                return
            }
        }
        $NewA=&($PSBoundParameters.WebName+'Account')  $PSBoundParameters -news
        $NewA.LogIn()
        if ($NewA.Token)
        {
            "登陆成功"
        }elseif($NewA.Msg){
            $NewA.Msg
#|Format-Table -AutoSize
        }
        return
    }

    #-----------自动签到----------------
    "开始自动签到"
    foreach ($item in $pp)
    {
        #如果该账号缺少对应模块,则直接跳过
        if ($modName -notcontains $item.WebName) {continue}
        $Account=&($item.WebName+'Account') $item
        $Account.CheckIn()
        $Account.Result | Format-Table -AutoSize
        
    }
    Start-Sleep -Seconds 3;pause
}
<#
process
{
    
}
#>
end
{


"运行结束"
}

签到-黑客派.psm1

#网络地址

$WebName='黑客派'

[uri]$login_uri='https://hacpai.com/api/v2/login'
[uri]$login_OTH_uri='https://hacpai.com/api/v2/login/2fa'
[uri]$user_info_uri='https://hacpai.com/api/v2/user/{0}'
#[uri]$user_point_uri='https://hacpai.com/api/v2/notifications/point?p=1'
[uri]$user_checkin_uri='https://hacpai.com/activity/checkin'
$WebHost=$login_uri.Host

#网络连接的基类
class GetWeb:Hashtable
{
    [uri]$Uri
    [validateset("Get", "Post", "Head")]
    [string]$Method='Get'
    hidden [int] $TimeoutSec
    hidden [string] $UserAgent

    GetWeb ([uri]$uri) {
        [uri]$this.Uri=$uri
        $this.Method='Get'
        $this.TimeoutSec= 7
        $this.UserAgent=$global:UserAgents
    }
}

class Result
{
    [string]$WebName=$Script:WebName
    [string]$UserName=$UserName
    [int]$StatusCode=-1
    [string]$Status
    [int]$Sc=-1
    [string]$Msg
    [int]$Points=-1
    [int]$AllPoint=-1
    [int]$Now=-1
    [int]$Long=-1
    Result($UserName){}
    [string]ToString(){
        $g="时间: {0:HH:mm′ss″}`t网站: {1}`t用户: {2}`t积分: {3}({4})`t次数: {5}/{6}" -f (get-date),$this.WebName,$this.UserName,$this.AllPoint,$this.points,$this.now,$this.Long 
        return $g
    }
}

class WebSession:GetWeb
{
    $WebSession

    WebSession ([uri]$uri,$WebSession):base($uri){
        $this.WebSession=$WebSession
    }

}

class PostAndWebSession:WebSession
{
    $WebSession
    PostAndWebSession ([uri]$uri,$WebSession):base($uri,$WebSession){
        $this.Method='Post'
    }
}

#用于登录的body
class login_Body
{
    [string]$userName
    [string]$userPassword
    $captcha
    login_Body ([string]$userName,[string]$MD5) {
        $this.userName=$userName
        $this.userPassword=$MD5
        $this.captcha=$NULL
    }
}

class Head:Hashtable
{
    [string]$Referer
    Head(){
        $this.Referer=$Script:user_checkin_uri
    }
}

#post加body
class PostAndBody:GetWeb
{
    $Body
    PostAndBody ([uri]$uri,$body):base($uri){
        $this.Method='Post'
        $this.Body=$body
    }
}

#发送两步验证的请求参数
class PostAndBodyAndWebSession:PostAndBody
{
    $WebSession
    PostAndBodyAndWebSession ([uri]$uri,$body,$WebSession):base($uri,$body){
        $this.WebSession=$WebSession
    }
}

#建立以用户为主的类,其中包含登录,签到等网络操作的方法
 class 账户:Hashtable
 {
    [string]$WebName=$Script:WebName
    [ValidateNotNullOrEmpty()]
    [string]$UserName
    [ValidateNotNullOrEmpty()]    
    [string]$MD5
    [PSCustomObject]$Token
    #输入方式添加
    账户 ([string]$username,[string]$userPassword){
        $this.WebName=$Script:WebName
        $this.UserName=$username
        $this.MD5=MD5($userPassword)
        #$this.Result=[Result]::new($UserName)
    }
    #从文件读取
    账户 ([PSCustomObject]$Account){
        $this.WebName=$Script:WebName
        $this.UserName=$Account.UserName
        $this.MD5=$Account.MD5
        $this.Token=$Account.Token
        $this.Result=[Result]::new($Account.UserName)
    }
    #登录
    LogIn()
    {
        $body=[login_Body]::new($this.UserName,$this.MD5)
        $login=[PostAndBody]::new($Script:login_uri,$body)
        
        $b=HttpRest $login

        if ($b.sc -ne $NULL)
        {
            switch ($b.sc)
            {
                0 
                {
                    $this.Token=@{symphony=$b.token}
                    Write-file -Account $this
                    return
                }
                10 
                {#两步验证
                    $r = new-object psobject -Property @{symphony=$b.token}
                    $this.OTH($r)
                    return
                }
                Default 
                {
                    $this.Sc=$b.sc
                    $this.Msg=$b.msg
                }
            }
        }elseif($b.StatusCode.value__ -ne $NULL){
            $this.StatusCode=$b.StatusCode.value__
            $this.Status=$b.StatusCode
            switch ($b.StatusCode.value__)
            {
                401 
                {
                    $this.Status='需要登录'
                    return
                }
                403 
                {
                    $this.Status='权限不足'
                    return
                }
            }
        }
    }

    #两步验证
    hidden OTH ($r) 
    {
        $Loadion=AddCookie -Tok $r -WebHost $Script:WebHost
        $OTH = Read-Host '两步验证码'
        $body=@{twofactorAuthCode=$OTH}
        $TOTH=[PostAndBodyAndWebSession]::new($Script:login_OTH_uri,$body,$Loadion)
        $c=HttpRest $TOTH

        if ($c.sc -ne $NULL)
        {
            switch ($c.sc)
            {
                0
                {
                    $this.Token=@{symphony=$c.token}
                    Write-file -Account $this
                    return
                }
                Default 
                {
                    $this.Sc=$c.sc
                    $this.Msg=$c.msg
                }
            }
        }elseif($c.StatusCode.value__ -ne $NULL){
            $this.StatusCode=$c.StatusCode.value__
            $this.Status=$c.StatusCode
            switch ($c.StatusCode.value__)
            {
                401 
                {
                    $this.Msg='需要登录'
                    return
                }
                403 
                {
                    $this.Msg='权限不足'
                    return
                }
            }
        }
    }

    #主要的签到操作流程
    CheckIn()
    {
        $this.Loadion=AddCookie -Tok $this.Token -WebHost $Script:WebHost
        $this.get_check_uri()
        $this.get_info()
        Write-Log $this.Result
    }

    #获取积分信息
    hidden get_info()
    {
        $e=[WebSession]::new($Script:user_info_uri -f $this.UserName,$this.Loadion)
        $q=HttpRest $e
        if ($q.sc -ne $NULL)
        {
            $this.Result.Sc=$q.sc
            $this.Result.Msg=$q.msg
            if ($q.sc -eq 0)
            {
                $this.Result.Long=$q.data.user.userLongestCheckinStreak
                $this.Result.Now=$q.data.user.userCurrentCheckinStreak
                $this.Result.AllPoint=$q.data.user.userPoint
                return
            }
        }elseif($q.StatusCode.value__ -ne $NULL){
            $this.Result.StatusCode=$q.StatusCode.value__
            $this.Result.Status=$q.StatusCode
            switch ($q.StatusCode.value__)
            {
                401 
                {
                    $this.Result.Msg='需要登录'
                    return
                }
                403 
                {
                    $this.Result.Msg='权限不足'
                    return
                }
            }
        }
    }

    #获取签到地址
    hidden get_check_uri()
    {
        $e=[WebSession]::new($Script:user_checkin_uri,$this.Loadion)
        $e2=$e.Clone()
        $s=HttpWeb $e 

        $this.Result.StatusCode=$s.StatusCode
        $this.Result.Status=$s.StatusDescription
        :ss switch ($s.StatusCode)
        {
            200
            {
                $re=$s.ParsedHtml.body.getElementsByClassName('btn green')[0].href
                if ($re)
                {
                    $e.Uri=$re
                    $e.Headers=[head]::new()
                    Break :ss
                }
            }
            401 
            {
                $this.Result.Msg='需要登录'
                return
            }
            403 
            {
                $this.Result.Msg='权限不足'
                return
            }
        }

        $s=HttpWeb $e
        $this.Result.StatusCode=$s.StatusCode
        $this.Result.Status=$s.StatusDescription
        switch ($s.StatusCode)
        {
            200 
            {
                $re=$s.ParsedHtml.body.getElementsByTagName('code')[0].innerText
                if ($re)
                {
                    $this.Result.Points=$re
                    return
                }
            }
            401 
            {
                $this.Result.Msg='需要登录'
                return
            }
            403 
            {
                $this.Result.Msg='权限不足'
                return
            }
        }
    }
}

function Account ()
{
    param
    (
        $Account,
        [switch] $news
    )

    if ($news){
        $f=[账户]::new($Account.UserName,$Account.PassWord)
    }else{
        $f=[账户]::new($Account)
    }
    return $f
}

Export-ModuleMember -Function Account
  • 黑客派

    黑客派是 B3log 开源社区的线上论坛,这里主要汇聚了程序员和设计师。HacPai 分别取 Hacker / Painter 的头三个字母组成,源自《黑客与画家》。

    358 引用 • 4754 回帖 • 510 关注
  • 签到
    38 引用 • 383 回帖 • 10 关注
2 操作
YxxXlv0COaxl 在 2019-05-27 20:14:45 更新了该帖
YxxXlv0COaxl 在 2019-05-26 13:53:58 更新了该帖

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • RESTful

    一种软件架构设计风格而不是标准,提供了一组设计原则和约束条件,主要用于客户端和服务器交互类的软件。基于这个风格设计的软件可以更简洁,更有层次,更易于实现缓存等机制。

    30 引用 • 114 回帖
  • 京东

    京东是中国最大的自营式电商企业,2015 年第一季度在中国自营式 B2C 电商市场的占有率为 56.3%。2014 年 5 月,京东在美国纳斯达克证券交易所正式挂牌上市(股票代码:JD),是中国第一个成功赴美上市的大型综合型电商平台,与腾讯、百度等中国互联网巨头共同跻身全球前十大互联网公司排行榜。

    14 引用 • 102 回帖 • 404 关注
  • jsoup

    jsoup 是一款 Java 的 HTML 解析器,可直接解析某个 URL 地址、HTML 文本内容。它提供了一套非常省力的 API,可通过 DOM,CSS 以及类似于 jQuery 的操作方法来取出和操作数据。

    6 引用 • 1 回帖 • 462 关注
  • GitHub

    GitHub 于 2008 年上线,目前,除了 Git 代码仓库托管及基本的 Web 管理界面以外,还提供了订阅、讨论组、文本渲染、在线文件编辑器、协作图谱(报表)、代码片段分享(Gist)等功能。正因为这些功能所提供的便利,又经过长期的积累,GitHub 的用户活跃度很高,在开源世界里享有深远的声望,并形成了社交化编程文化(Social Coding)。

    207 引用 • 2031 回帖
  • 微软

    微软是一家美国跨国科技公司,也是世界 PC 软件开发的先导,由比尔·盖茨与保罗·艾伦创办于 1975 年,公司总部设立在华盛顿州的雷德蒙德(Redmond,邻近西雅图)。以研发、制造、授权和提供广泛的电脑软件服务业务为主。

    8 引用 • 44 回帖
  • Java

    Java 是一种可以撰写跨平台应用软件的面向对象的程序设计语言,是由 Sun Microsystems 公司于 1995 年 5 月推出的。Java 技术具有卓越的通用性、高效性、平台移植性和安全性。

    3168 引用 • 8207 回帖
  • Openfire

    Openfire 是开源的、基于可拓展通讯和表示协议 (XMPP)、采用 Java 编程语言开发的实时协作服务器。Openfire 的效率很高,单台服务器可支持上万并发用户。

    6 引用 • 7 回帖 • 89 关注
  • Sillot

    Sillot (汐洛)孵化自思源笔记,致力于服务智慧新彖乄,具有彖乄驱动、极致优雅、开发者友好的特点
    Github 地址:https://github.com/Hi-Windom/Sillot

    15 引用 • 6 回帖 • 28 关注
  • 智能合约

    智能合约(Smart contract)是一种旨在以信息化方式传播、验证或执行合同的计算机协议。智能合约允许在没有第三方的情况下进行可信交易,这些交易可追踪且不可逆转。智能合约概念于 1994 年由 Nick Szabo 首次提出。

    1 引用 • 11 回帖 • 7 关注
  • TGIF

    Thank God It's Friday! 感谢老天,总算到星期五啦!

    284 引用 • 4481 回帖 • 654 关注
  • 游戏

    沉迷游戏伤身,强撸灰飞烟灭。

    169 引用 • 799 回帖
  • ngrok

    ngrok 是一个反向代理,通过在公共的端点和本地运行的 Web 服务器之间建立一个安全的通道。

    7 引用 • 63 回帖 • 598 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    51 引用 • 37 回帖
  • Maven

    Maven 是基于项目对象模型(POM)、通过一小段描述信息来管理项目的构建、报告和文档的软件项目管理工具。

    185 引用 • 318 回帖 • 344 关注
  • 开源

    Open Source, Open Mind, Open Sight, Open Future!

    396 引用 • 3416 回帖
  • JSON

    JSON (JavaScript Object Notation)是一种轻量级的数据交换格式。易于人类阅读和编写。同时也易于机器解析和生成。

    51 引用 • 190 回帖
  • Ngui

    Ngui 是一个 GUI 的排版显示引擎和跨平台的 GUI 应用程序开发框架,基于
    Node.js / OpenGL。目标是在此基础上开发 GUI 应用程序可拥有开发 WEB 应用般简单与速度同时兼顾 Native 应用程序的性能与体验。

    7 引用 • 9 回帖 • 345 关注
  • Windows

    Microsoft Windows 是美国微软公司研发的一套操作系统,它问世于 1985 年,起初仅仅是 Microsoft-DOS 模拟环境,后续的系统版本由于微软不断的更新升级,不但易用,也慢慢的成为家家户户人们最喜爱的操作系统。

    215 引用 • 462 回帖
  • 友情链接

    确认过眼神后的灵魂连接,站在链在!

    24 引用 • 373 回帖
  • 安全

    安全永远都不是一个小问题。

    189 引用 • 813 回帖
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    25 引用 • 191 回帖 • 21 关注
  • React

    React 是 Facebook 开源的一个用于构建 UI 的 JavaScript 库。

    192 引用 • 291 回帖 • 443 关注
  • Swagger

    Swagger 是一款非常流行的 API 开发工具,它遵循 OpenAPI Specification(这是一种通用的、和编程语言无关的 API 描述规范)。Swagger 贯穿整个 API 生命周期,如 API 的设计、编写文档、测试和部署。

    26 引用 • 35 回帖 • 13 关注
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    129 引用 • 793 回帖 • 2 关注
  • RYMCU

    RYMCU 致力于打造一个即严谨又活泼、专业又不失有趣,为数百万人服务的开源嵌入式知识学习交流平台。

    4 引用 • 6 回帖 • 40 关注
  • Sandbox

    如果帖子标签含有 Sandbox ,则该帖子会被视为“测试帖”,主要用于测试社区功能,排查 bug 等,该标签下内容不定期进行清理。

    370 引用 • 1215 回帖 • 582 关注
  • Vditor

    Vditor 是一款浏览器端的 Markdown 编辑器,支持所见即所得、即时渲染(类似 Typora)和分屏预览模式。它使用 TypeScript 实现,支持原生 JavaScript、Vue、React 和 Angular。

    313 引用 • 1667 回帖 • 1 关注