Module: Rex::Powershell::Parser

Included in:
Function, Script
Defined in:
lib/rex/powershell/parser.rb

Constant Summary collapse

RESERVED_VARIABLE_NAMES =

Reserved special variables Acquired with: Get-Variable | Format-Table name, value -auto

[
  '$$',
  '$?',
  '$^',
  '$_',
  '$args',
  '$ConfirmPreference',
  '$ConsoleFileName',
  '$DebugPreference',
  '$Env',
  '$Error',
  '$ErrorActionPreference',
  '$ErrorView',
  '$ExecutionContext',
  '$false',
  '$FormatEnumerationLimit',
  '$HOME',
  '$Host',
  '$input',
  '$LASTEXITCODE',
  '$MaximumAliasCount',
  '$MaximumDriveCount',
  '$MaximumErrorCount',
  '$MaximumFunctionCount',
  '$MaximumHistoryCount',
  '$MaximumVariableCount',
  '$MyInvocation',
  '$NestedPromptLevel',
  '$null',
  '$OutputEncoding',
  '$PID',
  '$PROFILE',
  '$ProgressPreference',
  '$PSBoundParameters',
  '$PSCulture',
  '$PSEmailServer',
  '$PSHOME',
  '$PSSessionApplicationName',
  '$PSSessionConfigurationName',
  '$PSSessionOption',
  '$PSUICulture',
  '$PSVersionTable',
  '$PWD',
  '$ReportErrorShowExceptionClass',
  '$ReportErrorShowInnerException',
  '$ReportErrorShowSource',
  '$ReportErrorShowStackTrace',
  '$ShellId',
  '$StackTrace',
  '$true',
  '$VerbosePreference',
  '$WarningPreference',
  '$WhatIfPreference'
].map(&:downcase).freeze

Instance Method Summary collapse

Instance Method Details

#block_extract(idx) ⇒ String

Extract block of code inside brackets/parenthesis

Attempts to match the bracket at idx, handling nesting manually Once the balanced matching bracket is found, all script content between idx and the index of the matching bracket is returned

Parameters:

  • idx (Integer)

    index of opening bracket

Returns:

  • (String)

    content between matching brackets



134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
# File 'lib/rex/powershell/parser.rb', line 134

def block_extract(idx)
  fail ArgumentError unless idx

  if idx < 0 || idx >= code.length
    fail ArgumentError, 'Invalid index'
  end

  start = code[idx]
  stop = match_start(start)
  delims = scan_with_index(/#{Regexp.escape(start)}|#{Regexp.escape(stop)}/, code[idx + 1..-1])
  delims.map { |x| x[1] = x[1] + idx + 1 }
  c = 1
  sidx = nil
  # Go through delims till we balance, get idx
  while (c != 0) && (x = delims.shift)
    sidx = x[1]
    x[0] == stop ? c -= 1 : c += 1
  end

  code[idx..sidx]
end

#get_func(func_name, delete = false) ⇒ String

Extract a block of function code

Parameters:

  • func_name (String)

    function name

  • delete (Boolean) (defaults to: false)

    delete the function from the code

Returns:

  • (String)

    function block



163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
# File 'lib/rex/powershell/parser.rb', line 163

def get_func(func_name, delete = false)
  start = code.index(func_name)

  return nil unless start

  idx = code[start..-1].index('{') + start
  func_txt = block_extract(idx)

  if delete
    delete_code = code[0..idx]
    delete_code << code[(idx + func_txt.length)..-1]
    @code = delete_code
  end

  Function.new(func_name, func_txt)
end

#get_func_namesArray

Get function names from code

Returns:

  • (Array)

    function names



76
77
78
# File 'lib/rex/powershell/parser.rb', line 76

def get_func_names
  code.scan(/function\s([a-zA-Z\-\_0-9]+)/).uniq.flatten
end

#get_string_literalsArray

Attempt to find string literals in PSH expression

Returns:

  • (Array)

    string literals



84
85
86
# File 'lib/rex/powershell/parser.rb', line 84

def get_string_literals
  code.scan(/@"(.+?)"@|@'(.+?)'@/m)
end

#get_var_namesArray

Get variable names from code, removes reserved names from return

Returns:

  • (Array)

    variable names



67
68
69
70
# File 'lib/rex/powershell/parser.rb', line 67

def get_var_names
  our_vars = code.scan(/\$[a-zA-Z\-\_0-9]+/).uniq.flatten.map(&:strip)
  our_vars.select { |v| !RESERVED_VARIABLE_NAMES.include?(v.downcase) }
end

#match_start(char) ⇒ String

Return matching bracket type

Parameters:

  • char (String)

    opening bracket character

Returns:

  • (String)

    matching closing bracket



109
110
111
112
113
114
115
116
117
118
119
120
121
122
# File 'lib/rex/powershell/parser.rb', line 109

def match_start(char)
  case char
  when '{'
    '}'
  when '('
    ')'
  when '['
    ']'
  when '<'
    '>'
  else
    fail ArgumentError, 'Unknown starting bracket'
  end
end

#scan_with_index(str, source = code) ⇒ Array[String,Integer]

Scan code and return matches with index

Parameters:

  • str (String)

    string to match in code

  • source (String) (defaults to: code)

    source code to match, defaults to @code

Returns:

  • (Array[String,Integer])

    matched items with index



95
96
97
98
99
100
101
# File 'lib/rex/powershell/parser.rb', line 95

def scan_with_index(str, source = code)
  ::Enumerator.new do |y|
    source.scan(str) do
      y << ::Regexp.last_match
    end
  end.map { |m| [m.to_s, m.offset(0)[0]] }
end