Skip to content

Commit

Permalink
linter: new inspection for php functions aliases (#1219)
Browse files Browse the repository at this point in the history
  • Loading branch information
Hidanio authored Dec 11, 2024
1 parent 2d18e46 commit 069d518
Show file tree
Hide file tree
Showing 10 changed files with 237 additions and 33 deletions.
23 changes: 22 additions & 1 deletion docs/checkers_doc.md
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

| Total checks | Checks enabled by default | Disabled checks by default | Autofixable checks |
| ------------ | ------------------------- | -------------------------- | ------------------ |
| 105 | 87 | 18 | 14 |
| 106 | 88 | 18 | 15 |

## Table of contents
- Enabled by default
Expand Down Expand Up @@ -64,6 +64,7 @@
- [`oldStyleConstructor` checker](#oldstyleconstructor-checker)
- [`paramClobber` checker](#paramclobber-checker)
- [`parentConstructor` checker](#parentconstructor-checker)
- [`phpAliases` checker (autofixable)](#phpaliases-checker)
- [`precedence` checker](#precedence-checker)
- [`printf` checker](#printf-checker)
- [`redundantGlobal` checker](#redundantglobal-checker)
Expand Down Expand Up @@ -1294,6 +1295,26 @@ class Foo extends Bar {
<p><br></p>


### `phpAliases` checker

> Auto fix available
#### Description

Report php aliases functions.

#### Non-compliant code:
```php
join("", []);
```

#### Compliant code:
```php
implode("", []);
```
<p><br></p>


### `precedence` checker

#### Description
Expand Down
153 changes: 125 additions & 28 deletions src/ir/phpcore/phpcore.go
Original file line number Diff line number Diff line change
Expand Up @@ -9,49 +9,146 @@ func ResolveAlias(function ir.Node) ir.Node {
if !ok {
return function
}
alias, ok := funcAliases[nm.Value]
alias, ok := FuncAliases[nm.Value]
if ok {
return alias
}
return function
}

func ResolveAliasName(n *ir.Name) *ir.Name {
alias, ok := funcAliases[n.Value]
alias, ok := FuncAliases[n.Value]
if ok {
return alias
}
return n
}

var funcAliases = map[string]*ir.Name{
var FuncAliases = map[string]*ir.Name{
// See https://www.php.net/manual/ru/aliases.php

`doubleval`: {Value: `floatval`},

`ini_alter`: {Value: `ini_set`},
`is_integer`: {Value: `is_int`},
`is_long`: {Value: `is_int`},
`is_real`: {Value: `is_float`},
`is_double`: {Value: `is_float`},

`join`: {Value: `implode`},
`chop`: {Value: `rtrim`},
`strchr`: {Value: `strstr`},
`pos`: {Value: `current`},
`key_exists`: {Value: `array_key_exists`},
`sizeof`: {Value: `count`},

`checkdnsrr`: {Value: `dns_check_record`},
`getmxrr`: {Value: `dns_get_mx`},

`is_writeable`: {Value: `is_writable`},
`diskfreespace`: {Value: `disk_free_space`},
`close`: {Value: `closedir`},
`fputs`: {Value: `fwrite`},
`magic_quotes_runtime`: {Value: `set_magic_quotes_runtime`},
`show_source`: {Value: `highlight_file`},

"chop": {Value: "rtrim"},
"close": {Value: "closedir"},
"com_get": {Value: "com_propget"},
"com_propset": {Value: "com_propput"},
"com_set": {Value: "com_propput"},
"die": {Value: "exit"},
"diskfreespace": {Value: "disk_free_space"},
"doubleval": {Value: "floatval"},
"fputs": {Value: "fwrite"},
"gzputs": {Value: "gzwrite"},
"i18n_convert": {Value: "mb_convert_encoding"},
"i18n_discover_encoding": {Value: "mb_detect_encoding"},
"i18n_http_input": {Value: "mb_http_input"},
"i18n_http_output": {Value: "mb_http_output"},
"i18n_internal_encoding": {Value: "mb_internal_encoding"},
"i18n_ja_jp_hantozen": {Value: "mb_convert_kana"},
"i18n_mime_header_decode": {Value: "mb_decode_mimeheader"},
"i18n_mime_header_encode": {Value: "mb_encode_mimeheader"},
"imap_create": {Value: "imap_createmailbox"},
"imap_fetchtext": {Value: "imap_body"},
"imap_getmailboxes": {Value: "imap_list_full"},
"imap_getsubscribed": {Value: "imap_lsub_full"},
"imap_header": {Value: "imap_headerinfo"},
"imap_listmailbox": {Value: "imap_list"},
"imap_listsubscribed": {Value: "imap_lsub"},
"imap_rename": {Value: "imap_renamemailbox"},
"imap_scan": {Value: "imap_listscan"},
"imap_scanmailbox": {Value: "imap_listscan"},
"ini_alter": {Value: "ini_set"},
"is_double": {Value: "is_float"},
"is_integer": {Value: "is_int"},
"is_long": {Value: "is_int"},
"is_real": {Value: "is_float"},
"is_writeable": {Value: "is_writable"},
"join": {Value: "implode"},
"key_exists": {Value: "array_key_exists"},
"ldap_close": {Value: "ldap_unbind"},
"mbstrcut": {Value: "mb_strcut"},
"mbstrlen": {Value: "mb_strlen"},
"mbstrpos": {Value: "mb_strpos"},
"mbstrrpos": {Value: "mb_strrpos"},
"mbsubstr": {Value: "mb_substr"},
"mysql": {Value: "mysql_db_query"},
"mysql_createdb": {Value: "mysql_create_db"},
"mysql_db_name": {Value: "mysql_result"},
"mysql_dbname": {Value: "mysql_result"},
"mysql_dropdb": {Value: "mysql_drop_db"},
"mysql_fieldflags": {Value: "mysql_field_flags"},
"mysql_fieldlen": {Value: "mysql_field_len"},
"mysql_fieldname": {Value: "mysql_field_name"},
"mysql_fieldtable": {Value: "mysql_field_table"},
"mysql_fieldtype": {Value: "mysql_field_type"},
"mysql_freeresult": {Value: "mysql_free_result"},
"mysql_listdbs": {Value: "mysql_list_dbs"},
"mysql_listfields": {Value: "mysql_list_fields"},
"mysql_listtables": {Value: "mysql_list_tables"},
"mysql_numfields": {Value: "mysql_num_fields"},
"mysql_numrows": {Value: "mysql_num_rows"},
"mysql_selectdb": {Value: "mysql_select_db"},
"mysql_tablename": {Value: "mysql_result"},
"ociassignelem": {Value: "OCICollection::assignElem"},
"ocibindbyname": {Value: "oci_bind_by_name"},
"ocicancel": {Value: "oci_cancel"},
"ocicloselob": {Value: "OCILob::close"},
"ocicollappend": {Value: "OCICollection::append"},
"ocicollassign": {Value: "OCICollection::assign"},
"ocicollmax": {Value: "OCICollection::max"},
"ocicollsize": {Value: "OCICollection::size"},
"ocicolltrim": {Value: "OCICollection::trim"},
"ocicolumnisnull": {Value: "oci_field_is_null"},
"ocicolumnname": {Value: "oci_field_name"},
"ocicolumnprecision": {Value: "oci_field_precision"},
"ocicolumnscale": {Value: "oci_field_scale"},
"ocicolumnsize": {Value: "oci_field_size"},
"ocicolumntype": {Value: "oci_field_type"},
"ocicolumntyperaw": {Value: "oci_field_type_raw"},
"ocicommit": {Value: "oci_commit"},
"ocidefinebyname": {Value: "oci_define_by_name"},
"ocierror": {Value: "oci_error"},
"ociexecute": {Value: "oci_execute"},
"ocifetch": {Value: "oci_fetch"},
"ocifetchinto": {Value: "oci_fetch_object"},
"ocifetchstatement": {Value: "oci_fetch_all"},
"ocifreecollection": {Value: "OCICollection::free"},
"ocifreecursor": {Value: "oci_free_statement"},
"ocifreedesc": {Value: "oci_free_descriptor"},
"ocifreestatement": {Value: "oci_free_statement"},
"ocigetelem": {Value: "OCICollection::getElem"},
"ociinternaldebug": {Value: "oci_internal_debug"},
"ociloadlob": {Value: "OCILob::load"},
"ocilogon": {Value: "oci_connect"},
"ocinewcollection": {Value: "oci_new_collection"},
"ocinewcursor": {Value: "oci_new_cursor"},
"ocinewdescriptor": {Value: "oci_new_descriptor"},
"ocinlogon": {Value: "oci_new_connect"},
"ocinumcols": {Value: "oci_num_fields"},
"ociparse": {Value: "oci_parse"},
"ocipasswordchange": {Value: "oci_password_change"},
"ociplogon": {Value: "oci_pconnect"},
"ociresult": {Value: "oci_result"},
"ocirollback": {Value: "oci_rollback"},
"ocisavelob": {Value: "OCILob::save"},
"ocisavelobfile": {Value: "OCILob::import"},
"ociserverversion": {Value: "oci_server_version"},
"ocisetprefetch": {Value: "oci_set_prefetch"},
"ocistatementtype": {Value: "oci_statement_type"},
"ociwritelobtofile": {Value: "OCILob::export"},
"ociwritetemporarylob": {Value: "OCILob::writeTemporary"},
"odbc_do": {Value: "odbc_exec"},
"odbc_field_precision": {Value: "odbc_field_len"},
"pg_clientencoding": {Value: "pg_client_encoding"},
"pg_setclientencoding": {Value: "pg_set_client_encoding"},
"pos": {Value: "current"},
"recode": {Value: "recode_string"},
"show_source": {Value: "highlight_file"},
"sizeof": {Value: "count"},
"snmpwalkoid": {Value: "snmprealwalk"},
"strchr": {Value: "strstr"},
"xptr_new_context": {Value: "xpath_new_context"},
`checkdnsrr`: {Value: `dns_check_record`},
`getmxrr`: {Value: `dns_get_mx`},
`magic_quotes_runtime`: {Value: `set_magic_quotes_runtime`},
`stream_register_wrapper`: {Value: `stream_wrapper_register`},
`set_file_buffer`: {Value: `stream_set_write_buffer`},
`socket_set_blocking`: {Value: `stream_set_blocking`},
Expand Down
10 changes: 8 additions & 2 deletions src/linter/block_linter.go
Original file line number Diff line number Diff line change
Expand Up @@ -1040,9 +1040,15 @@ func (b *blockLinter) checkFunctionCall(e *ir.FunctionCallExpr) {
call := resolveFunctionCall(b.walker.ctx.sc, b.classParseState(), b.walker.ctx.customTypes, e)
fqName := call.funcName

var trimName = strings.TrimPrefix(fqName, `\`)
var phpMasterFunc = phpcore.FuncAliases[trimName]
if phpMasterFunc != nil {
b.report(e, LevelWarning, "phpAliases", "Use %s instead of '%s'", phpMasterFunc.Value, trimName)
b.walker.r.addQuickFix("phpAliases", b.quickfix.PhpAliasesReplace(e.Function.(*ir.Name), phpMasterFunc.Value))
}

if call.isClosure {
varName := strings.TrimPrefix(fqName, `\`)
b.walker.untrackVarName(varName)
b.walker.untrackVarName(trimName)
} else {
b.checkFunctionAvailability(e, &call)
b.walker.r.checker.CheckNameCase(e.Function, call.funcName, call.info.Name)
Expand Down
8 changes: 8 additions & 0 deletions src/linter/quickfix.go
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,14 @@ func (g *QuickFixGenerator) NullForNotNullableProperty(prop *ir.PropertyStmt) qu
}
}

func (g *QuickFixGenerator) PhpAliasesReplace(prop *ir.Name, masterFunction string) quickfix.TextEdit {
return quickfix.TextEdit{
StartPos: prop.Position.StartPos,
EndPos: prop.Position.EndPos,
Replacement: masterFunction,
}
}

func (g *QuickFixGenerator) notExplicitNullableParam(param ir.Node) quickfix.TextEdit {
var pos *position.Position
var value string
Expand Down
9 changes: 9 additions & 0 deletions src/linter/report.go
Original file line number Diff line number Diff line change
Expand Up @@ -53,6 +53,15 @@ return -9223372036854775808;`,
After: `return PHP_INT_MIN;`,
},

{
Name: "phpAliases",
Default: true,
Quickfix: true,
Comment: `Report php aliases functions.`,
Before: `join("", []);`,
After: `implode("", []);`,
},

{
Name: "discardExpr",
Default: true,
Expand Down
1 change: 1 addition & 0 deletions src/tests/checkers/basic_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -2146,6 +2146,7 @@ function f() {
test.Expect = []string{
`Use float cast instead of real`,
`Use is_float function instead of is_real`,
`Use is_float instead of 'is_real`,
}
test.RunAndMatch()
}
Expand Down
43 changes: 43 additions & 0 deletions src/tests/checkers/php_aliases_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
package checkers

import (
"testing"

"github.com/VKCOM/noverify/src/linttest"
)

func TestPhpAliases(t *testing.T) {
test := linttest.NewSuite(t)
test.AddFile(`<?php
declare(strict_types = "1")
$_ = join("", []);
`)
test.Expect = []string{
`Call to undefined function join`,
`Use implode instead of 'join'`,
}
test.RunAndMatch()
}

func TestPhpAliasesFunctionCall(t *testing.T) {
test := linttest.NewSuite(t)
test.AddFile(`<?php
declare(strict_types = "1")
$_ = join("", []);
function test($d){
ocicollmax();
}
test(join("", []));
`)
test.Expect = []string{
`Use OCICollection::max instead of 'ocicollmax'`,
`Call to undefined function ocicollmax`,
`Use implode instead of 'join'`,
`Call to undefined function join`,
`Use implode instead of 'join'`,
`Call to undefined function join`,
}
test.RunAndMatch()
}
5 changes: 3 additions & 2 deletions src/tests/golden/golden_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,7 @@ func TestGolden(t *testing.T) {
Name: "embeddedrules",
Disable: []string{
"deadCode",
"phpAliases",
},
Deps: []string{
`stubs/phpstorm-stubs/pcre/pcre.php`,
Expand Down Expand Up @@ -95,7 +96,7 @@ func TestGolden(t *testing.T) {

{
Name: "parsedown",
Disable: []string{"missingPhpdoc", "arraySyntax"},
Disable: []string{"missingPhpdoc", "arraySyntax", "phpAliases"},
Deps: []string{
`stubs/phpstorm-stubs/pcre/pcre.php`,
`stubs/phpstorm-stubs/mbstring/mbstring.php`,
Expand All @@ -104,7 +105,7 @@ func TestGolden(t *testing.T) {

{
Name: "underscore",
Disable: []string{"missingPhpdoc"},
Disable: []string{"missingPhpdoc", "phpAliases"},
Deps: []string{
`stubs/phpstorm-stubs/pcre/pcre.php`,
},
Expand Down
9 changes: 9 additions & 0 deletions src/tests/golden/testdata/quickfix/phpAliases.php
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?php
declare(strict_types = "1")
$_ = join("", []);

function test($d){
ocicollmax();
}

test(join("", []));
Original file line number Diff line number Diff line change
@@ -0,0 +1,9 @@
<?php
declare(strict_types = "1")
$_ = implode("", []);

function test($d){
OCICollection::max();
}

test(implode("", []));

0 comments on commit 069d518

Please sign in to comment.