2020-11-14 09:20:07 +00:00
<!DOCTYPE html>
< html >
< head >
< meta charset = "utf-8" >
< meta http-equiv = "X-UA-Compatible" content = "IE=edge" >
< meta name = "viewport" content = "width=device-width, initial-scale=1" >
< meta name = "description" content = "Learn how to define your own properties both for Java and XPath rules." >
< meta name = "keywords" content = "extendinguserdocs, " >
< title > Defining rule properties | PMD Source Code Analyzer< / title >
2023-04-28 11:49:15 +00:00
< link rel = "stylesheet" type = "text/css" href = "assets/fontawesome-free-5.15.4-web/css/all.min.css" >
2023-03-03 20:15:41 +00:00
< link rel = "stylesheet" type = "text/css" href = "assets/bootstrap-4.5.2-dist/css/bootstrap.min.css" >
2020-11-14 09:20:07 +00:00
2023-03-03 20:15:41 +00:00
< link rel = "stylesheet" type = "text/css" href = "css/syntax.css" >
< link rel = "stylesheet" type = "text/css" href = "css/modern-business.css" >
< link rel = "stylesheet" type = "text/css" href = "css/customstyles.css" >
< link rel = "stylesheet" type = "text/css" href = "css/theme-green.css" >
< link rel = "stylesheet" type = "text/css" href = "css/pmd-customstyles.css" >
2020-11-14 09:20:07 +00:00
2023-04-28 11:49:15 +00:00
< link rel = "shortcut icon" href = "images/logo/favicon.ico" type = "image/x-icon" >
< link rel = "icon" href = "images/logo/favicon.ico" type = "image/x-icon" >
2020-11-14 09:20:07 +00:00
2023-03-03 20:15:41 +00:00
< link rel = "alternate" type = "application/rss+xml" title = "" href = "feed.xml" >
2020-11-14 09:20:07 +00:00
< / head >
< body >
<!-- Content is offset by the height of the topnav bar. -->
<!-- There's already a padding - top rule in modern - business.css, but it apparently doesn't work on Firefox 60 and Chrome 67 -->
< div id = "topbar-content-offset" >
<!-- Navigation -->
2023-03-03 20:15:41 +00:00
< nav class = "navbar navbar-expand-lg fixed-top navbar-dark" >
2020-11-14 09:20:07 +00:00
< div class = "container topnavlinks" >
2023-03-03 20:15:41 +00:00
< a class = "navbar-brand fas fa-home fa-lg" href = "index.html" > < span class = "projectTitle" > PMD Source Code Analyzer Project< / span > < / a >
< button class = "navbar-toggler" type = "button" data-toggle = "collapse" data-target = "#navbarSupportedContent" aria-controls = "navbarSupportedContent" aria-expanded = "false" aria-label = "Toggle navigation" >
< span class = "navbar-toggler-icon" > < / span >
< / button >
< div class = "collapse navbar-collapse" id = "navbarSupportedContent" >
< ul class = "navbar-nav mr-auto mt-2 mt-lg-0" > < / ul >
< ul class = "navbar-nav" >
2020-11-14 09:20:07 +00:00
<!-- toggle sidebar button -->
2023-03-03 20:15:41 +00:00
< li class = "nav-item" > < a id = "tg-sb-link" class = "nav-link" href = "#" > < i id = "tg-sb-icon" class = "fas fa-toggle-on" > < / i > Nav< / a > < / li >
2020-11-14 09:20:07 +00:00
<!-- entries without drop - downs appear here -->
2023-03-03 20:15:41 +00:00
< li class = "nav-item" > < a class = "nav-link" href = "https://github.com/pmd/pmd/releases/latest" target = "_blank" > Download< / a > < / li >
< li class = "nav-item" > < a class = "nav-link" href = "https://github.com/pmd/pmd" target = "_blank" > Fork us on github< / a > < / li >
2020-11-14 09:20:07 +00:00
<!-- entries with drop - downs appear here -->
<!-- conditional logic to control which topnav appears for the audience defined in the configuration file. -->
< / ul >
2023-03-03 20:15:41 +00:00
< form class = "form-inline my-2 my-lg-0" >
< input class = "form-control mr-sm-2" type = "search" placeholder = "search..." id = "search-input" >
< ul id = "results-container" > < / ul >
< / form >
2020-11-14 09:20:07 +00:00
< / div >
2023-03-03 20:15:41 +00:00
< / div >
2020-11-14 09:20:07 +00:00
< / nav >
<!-- Page Content -->
2022-02-25 09:51:51 +00:00
< div class = "container-toc-wrapper" >
< div class = "container" >
< div class = "col-lg-12" > < / div >
<!-- Content Row -->
< div class = "row" >
2020-11-14 09:20:07 +00:00
2022-02-25 09:51:51 +00:00
<!-- Sidebar Column -->
< div class = "col-md-3" id = "tg-sb-sidebar" >
2020-11-14 09:20:07 +00:00
< ul id = "mysidebar" class = "nav" >
2024-10-25 08:59:30 +00:00
< li class = "sidebarTitle" > PMD 7.8.0-SNAPSHOT< / li >
< div class = "sidebarTitleDate" > Release date: 29-November-2024< / div >
2020-11-14 09:20:07 +00:00
< li >
< a href = "#" > About< / a >
< ul >
< li > < a href = "index.html" > Home< / a > < / li >
2023-03-23 09:46:34 +00:00
< li > < a href = "pmd_release_notes.html" > Release notes< / a > < / li >
2020-11-14 09:20:07 +00:00
2023-03-23 09:46:34 +00:00
< li > < a href = "pmd_release_notes_pmd7.html" > Release notes (PMD 7)< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_about_help.html" > Getting help< / a > < / li >
2024-06-20 14:27:53 +00:00
< li > < a href = "pmd_about_release_policies.html" > Release policies< / a > < / li >
< li > < a href = "pmd_about_support_lifecycle.html" > Support lifecycle< / a > < / li >
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
< li >
< a href = "#" > User Documentation< / a >
< ul >
2023-08-31 14:51:16 +00:00
< li > < a href = "pmd_userdocs_migrating_to_pmd7.html" > Migration Guide for PMD 7< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_userdocs_installation.html" > Installation and basic CLI usage< / a > < / li >
< li > < a href = "pmd_userdocs_making_rulesets.html" > Making rulesets< / a > < / li >
< li > < a href = "pmd_userdocs_configuring_rules.html" > Configuring rules< / a > < / li >
< li > < a href = "pmd_userdocs_best_practices.html" > Best practices< / a > < / li >
< li > < a href = "pmd_userdocs_suppressing_warnings.html" > Suppressing warnings< / a > < / li >
< li > < a href = "pmd_userdocs_incremental_analysis.html" > Incremental analysis< / a > < / li >
< li > < a href = "pmd_userdocs_cli_reference.html" > PMD CLI reference< / a > < / li >
< li > < a href = "pmd_userdocs_report_formats.html" > PMD Report formats< / a > < / li >
2022-09-29 15:04:50 +00:00
< li > < a href = "pmd_userdocs_3rdpartyrulesets.html" > 3rd party rulesets< / a > < / li >
2020-11-14 09:20:07 +00:00
< li class = "subfolders" >
< a href = "#" > CPD reference< / a >
< ul >
< li > < a href = "pmd_userdocs_cpd.html" > Copy-paste detection< / a > < / li >
< li > < a href = "pmd_userdocs_cpd_report_formats.html" > CPD Report formats< / a > < / li >
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > Extending PMD< / a >
< ul >
< li > < a href = "pmd_userdocs_extending_writing_rules_intro.html" > Introduction to writing rules< / a > < / li >
< li > < a href = "pmd_userdocs_extending_your_first_rule.html" > Your first rule< / a > < / li >
< li > < a href = "pmd_userdocs_extending_writing_xpath_rules.html" > XPath rules< / a > < / li >
< li > < a href = "pmd_userdocs_extending_writing_java_rules.html" > Java rules< / a > < / li >
< li > < a href = "pmd_userdocs_extending_designer_reference.html" > Rule designer reference< / a > < / li >
< li class = "active" > < a href = "pmd_userdocs_extending_defining_properties.html" > Defining rule properties< / a > < / li >
< li > < a href = "pmd_userdocs_extending_rule_guidelines.html" > Rule guidelines< / a > < / li >
< li > < a href = "pmd_userdocs_extending_testing.html" > Testing your rules< / a > < / li >
2024-02-09 17:22:16 +00:00
< li > < a href = "pmd_userdocs_extending_ast_dump.html" > Creating (XML) dump of the AST< / a > < / li >
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > Tools / Integrations< / a >
< ul >
< li > < a href = "pmd_userdocs_tools_maven.html" > Maven PMD Plugin< / a > < / li >
< li > < a href = "pmd_userdocs_tools_gradle.html" > Gradle< / a > < / li >
< li > < a href = "pmd_userdocs_tools_ant.html" > Ant< / a > < / li >
< li > < a href = "pmd_userdocs_tools_java_api.html" > PMD Java API< / a > < / li >
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_userdocs_tools_bld.html" > bld PMD Extension< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_userdocs_tools_ci.html" > CI integrations< / a > < / li >
< li > < a href = "pmd_userdocs_tools.html" > Other Tools / Integrations< / a > < / li >
< / ul >
< / li >
< / ul >
< / li >
< li >
< a href = "#" > Rule Reference< / a >
< ul >
< li class = "subfolders" >
< a href = "#" > Apex Rules< / a >
< ul >
< li > < a href = "pmd_rules_apex.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_apex_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_apex_codestyle.html" > Code Style< / a > < / li >
< li > < a href = "pmd_rules_apex_design.html" > Design< / a > < / li >
< li > < a href = "pmd_rules_apex_documentation.html" > Documentation< / a > < / li >
< li > < a href = "pmd_rules_apex_errorprone.html" > Error Prone< / a > < / li >
< li > < a href = "pmd_rules_apex_performance.html" > Performance< / a > < / li >
< li > < a href = "pmd_rules_apex_security.html" > Security< / a > < / li >
< / ul >
< / li >
2022-04-28 13:34:56 +00:00
< li class = "subfolders" >
< a href = "#" > HTML Rules< / a >
< ul >
< li > < a href = "pmd_rules_html.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_html_bestpractices.html" > Best Practices< / a > < / li >
< / ul >
< / li >
2020-11-14 09:20:07 +00:00
< li class = "subfolders" >
< a href = "#" > Java Rules< / a >
< ul >
< li > < a href = "pmd_rules_java.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_java_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_java_codestyle.html" > Code Style< / a > < / li >
< li > < a href = "pmd_rules_java_design.html" > Design< / a > < / li >
< li > < a href = "pmd_rules_java_documentation.html" > Documentation< / a > < / li >
< li > < a href = "pmd_rules_java_errorprone.html" > Error Prone< / a > < / li >
< li > < a href = "pmd_rules_java_multithreading.html" > Multithreading< / a > < / li >
< li > < a href = "pmd_rules_java_performance.html" > Performance< / a > < / li >
< li > < a href = "pmd_rules_java_security.html" > Security< / a > < / li >
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > Java Server Pages Rules< / a >
< ul >
< li > < a href = "pmd_rules_jsp.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_jsp_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_jsp_codestyle.html" > Code Style< / a > < / li >
< li > < a href = "pmd_rules_jsp_design.html" > Design< / a > < / li >
< li > < a href = "pmd_rules_jsp_errorprone.html" > Error Prone< / a > < / li >
< li > < a href = "pmd_rules_jsp_security.html" > Security< / a > < / li >
< / ul >
< / li >
2023-03-03 20:15:41 +00:00
< li class = "subfolders" >
< a href = "#" > JavaScript Rules< / a >
< ul >
< li > < a href = "pmd_rules_ecmascript.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_ecmascript_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_ecmascript_codestyle.html" > Code Style< / a > < / li >
< li > < a href = "pmd_rules_ecmascript_errorprone.html" > Error Prone< / a > < / li >
2024-08-29 16:28:37 +00:00
< li > < a href = "pmd_rules_ecmascript_performance.html" > Performance< / a > < / li >
2023-03-03 20:15:41 +00:00
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > Kotlin Rules< / a >
< ul >
< li > < a href = "pmd_rules_kotlin.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_kotlin_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_kotlin_errorprone.html" > Error Prone< / a > < / li >
< / ul >
< / li >
2020-11-14 09:20:07 +00:00
< li class = "subfolders" >
< a href = "#" > Maven POM Rules< / a >
< ul >
< li > < a href = "pmd_rules_pom.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_pom_errorprone.html" > Error Prone< / a > < / li >
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > Modelica Rules< / a >
< ul >
< li > < a href = "pmd_rules_modelica.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_modelica_bestpractices.html" > Best Practices< / a > < / li >
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > PLSQL Rules< / a >
< ul >
< li > < a href = "pmd_rules_plsql.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_plsql_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_plsql_codestyle.html" > Code Style< / a > < / li >
< li > < a href = "pmd_rules_plsql_design.html" > Design< / a > < / li >
< li > < a href = "pmd_rules_plsql_errorprone.html" > Error Prone< / a > < / li >
< / ul >
< / li >
< li class = "subfolders" >
2023-10-20 00:49:52 +00:00
< a href = "#" > Salesforce Visualforce Rules< / a >
2020-11-14 09:20:07 +00:00
< ul >
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_rules_visualforce.html" > Index< / a > < / li >
2020-11-14 09:20:07 +00:00
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_rules_visualforce_security.html" > Security< / a > < / li >
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
2022-11-18 15:25:16 +00:00
< li class = "subfolders" >
< a href = "#" > Scala Rules< / a >
< ul >
< li > < a href = "pmd_rules_scala.html" > Index< / a > < / li >
< / ul >
< / li >
2023-03-03 20:15:41 +00:00
< li class = "subfolders" >
< a href = "#" > Swift Rules< / a >
< ul >
< li > < a href = "pmd_rules_swift.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_swift_bestpractices.html" > Best Practices< / a > < / li >
< li > < a href = "pmd_rules_swift_errorprone.html" > Error Prone< / a > < / li >
< / ul >
< / li >
2020-11-14 09:20:07 +00:00
< li class = "subfolders" >
2023-10-20 00:49:52 +00:00
< a href = "#" > Velocity Template Language (VTL) Rules< / a >
2020-11-14 09:20:07 +00:00
< ul >
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_rules_velocity.html" > Index< / a > < / li >
2020-11-14 09:20:07 +00:00
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_rules_velocity_bestpractices.html" > Best Practices< / a > < / li >
2020-11-14 09:20:07 +00:00
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_rules_velocity_design.html" > Design< / a > < / li >
2020-11-14 09:20:07 +00:00
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_rules_velocity_errorprone.html" > Error Prone< / a > < / li >
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
2022-11-18 15:25:16 +00:00
< li class = "subfolders" >
< a href = "#" > WSDL Rules< / a >
< ul >
< li > < a href = "pmd_rules_wsdl.html" > Index< / a > < / li >
< / ul >
< / li >
2020-11-14 09:20:07 +00:00
< li class = "subfolders" >
< a href = "#" > XML Rules< / a >
< ul >
< li > < a href = "pmd_rules_xml.html" > Index< / a > < / li >
2023-10-19 09:25:45 +00:00
< li > < a href = "pmd_rules_xml_bestpractices.html" > Best Practices< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_rules_xml_errorprone.html" > Error Prone< / a > < / li >
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > XSL Rules< / a >
< ul >
< li > < a href = "pmd_rules_xsl.html" > Index< / a > < / li >
< li > < a href = "pmd_rules_xsl_codestyle.html" > Code Style< / a > < / li >
< li > < a href = "pmd_rules_xsl_performance.html" > Performance< / a > < / li >
< / ul >
< / li >
< / ul >
< / li >
< li >
2023-03-03 20:15:41 +00:00
< a href = "#" > Language-Specific Documentation< / a >
2020-11-14 09:20:07 +00:00
< ul >
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_index.html" > Overview< / a > < / li >
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_languages_configuration.html" > Language configuration< / a > < / li >
2020-11-14 09:20:07 +00:00
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_languages_apex.html" > Apex< / a > < / li >
2022-03-27 15:04:13 +00:00
2023-03-03 20:15:41 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_cpp.html" > C/C++< / a > < / li >
< li > < a href = "pmd_languages_cs.html" > C#< / a > < / li >
< li > < a href = "pmd_languages_coco.html" > Coco< / a > < / li >
< li > < a href = "pmd_languages_dart.html" > Dart< / a > < / li >
< li > < a href = "pmd_languages_fortran.html" > Fortran< / a > < / li >
< li > < a href = "pmd_languages_gherkin.html" > Gherkin< / a > < / li >
< li > < a href = "pmd_languages_go.html" > Go< / a > < / li >
< li > < a href = "pmd_languages_html.html" > HTML< / a > < / li >
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_languages_java.html" > Java< / a > < / li >
2022-03-27 15:04:13 +00:00
2023-04-29 14:37:59 +00:00
< li > < a href = "pmd_languages_js_ts.html" > JavaScript / TypeScript< / a > < / li >
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_languages_jsp.html" > JSP< / a > < / li >
2020-11-14 09:20:07 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_julia.html" > Julia< / a > < / li >
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_languages_kotlin.html" > Kotlin< / a > < / li >
2020-11-14 09:20:07 +00:00
2021-03-26 08:22:23 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_lua.html" > Lua< / a > < / li >
< li > < a href = "pmd_languages_matlab.html" > Matlab< / a > < / li >
< li > < a href = "pmd_languages_modelica.html" > Modelica< / a > < / li >
< li > < a href = "pmd_languages_objectivec.html" > Objective-C< / a > < / li >
< li > < a href = "pmd_languages_perl.html" > Perl< / a > < / li >
< li > < a href = "pmd_languages_php.html" > PHP< / a > < / li >
2021-03-26 08:22:23 +00:00
< li > < a href = "pmd_languages_plsql.html" > PLSQL< / a > < / li >
2021-10-29 17:50:40 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_python.html" > Python< / a > < / li >
2022-04-28 13:34:56 +00:00
2022-07-01 07:55:16 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_ruby.html" > Ruby< / a > < / li >
2022-07-01 07:55:16 +00:00
2023-04-29 16:54:57 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_scala.html" > Scala< / a > < / li >
2023-04-29 16:54:57 +00:00
2023-05-19 09:58:17 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_swift.html" > Swift< / a > < / li >
2023-10-20 00:42:49 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_tsql.html" > T-SQL< / a > < / li >
2023-10-20 00:42:49 +00:00
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_visualforce.html" > Visualforce< / a > < / li >
2024-03-04 19:16:42 +00:00
< li > < a href = "pmd_languages_velocity.html" > Velocity Template Language (VTL)< / a > < / li >
2023-10-20 00:49:52 +00:00
< li > < a href = "pmd_languages_xml.html" > XML and XML dialects< / a > < / li >
2023-05-19 09:58:17 +00:00
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
< li >
< a href = "#" > Developer Documentation< / a >
< ul >
< li > < a href = "pmd_devdocs_development.html" > Developer resources< / a > < / li >
< li > < a href = "pmd_devdocs_building.html" > Building PMD from source< / a > < / li >
2024-09-26 14:58:11 +00:00
< li > < a href = "https://github.com/pmd/pmd/blob/main/CONTRIBUTING.md" target = "_blank" > Contributing< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_devdocs_writing_documentation.html" > Writing documentation< / a > < / li >
< li > < a href = "pmd_devdocs_roadmap.html" > Roadmap< / a > < / li >
< li > < a href = "pmd_devdocs_how_pmd_works.html" > How PMD works< / a > < / li >
< li > < a href = "pmd_devdocs_pmdtester.html" > Pmdtester< / a > < / li >
< li > < a href = "pmd_devdocs_rule_deprecation_policy.html" > Rule Deprecation Policy< / a > < / li >
< li class = "subfolders" >
< a href = "#" > Major contributions< / a >
< ul >
2021-08-27 14:58:36 +00:00
< li > < a href = "pmd_devdocs_major_rule_guidelines.html" > Rule Guidelines< / a > < / li >
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_devdocs_major_adding_new_language_javacc.html" > Adding a new language (JavaCC)< / a > < / li >
2020-11-14 09:20:07 +00:00
2023-04-20 15:12:02 +00:00
< li > < a href = "pmd_devdocs_major_adding_new_language_antlr.html" > Adding a new language (ANTLR)< / a > < / li >
2020-11-14 09:20:07 +00:00
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_devdocs_major_adding_new_cpd_language.html" > Adding a new CPD language< / a > < / li >
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
< li class = "subfolders" >
< a href = "#" > Experimental features< / a >
< ul >
2023-10-20 00:49:52 +00:00
< li > < a href = "tag_experimental.html" > List of experimental Features< / a > < / li >
2020-11-14 09:20:07 +00:00
< / ul >
< / li >
< / ul >
< / li >
< li >
< a href = "#" > Project documentation< / a >
< ul >
< li class = "subfolders" >
< a href = "#" > Trivia about PMD< / a >
< ul >
< li > < a href = "pmd_projectdocs_trivia_news.html" > PMD in the press< / a > < / li >
< li > < a href = "pmd_projectdocs_trivia_products.html" > Products & books related to PMD< / a > < / li >
< li > < a href = "pmd_projectdocs_trivia_similarprojects.html" > Similar projects< / a > < / li >
< li > < a href = "pmd_projectdocs_trivia_meaning.html" > What does 'PMD' mean?< / a > < / li >
< / ul >
< / li >
2023-03-03 20:15:41 +00:00
< li > < a href = "pmd_projectdocs_logo.html" > Logo< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_projectdocs_faq.html" > FAQ< / a > < / li >
< li > < a href = "license.html" > License< / a > < / li >
< li > < a href = "pmd_projectdocs_credits.html" > Credits< / a > < / li >
< li > < a href = "pmd_release_notes_old.html" > Old release notes< / a > < / li >
2022-09-30 10:03:06 +00:00
< li > < a href = "pmd_projectdocs_decisions.html" > Decisions< / a > < / li >
2020-11-14 09:20:07 +00:00
< li class = "subfolders" >
< a href = "#" > Project management< / a >
< ul >
2021-04-23 20:35:51 +00:00
< li > < a href = "pmd_projectdocs_committers_infrastructure.html" > Infrastructure< / a > < / li >
2020-11-14 09:20:07 +00:00
< li > < a href = "pmd_projectdocs_committers_releasing.html" > Release process< / a > < / li >
< li > < a href = "pmd_projectdocs_committers_merging_pull_requests.html" > Merging pull requests< / a > < / li >
< li > < a href = "pmd_projectdocs_committers_main_landing_page.html" > Main Landing page< / a > < / li >
< / ul >
< / li >
< / ul >
< / li >
< / ul >
2022-02-25 09:51:51 +00:00
< / div >
2020-11-14 09:20:07 +00:00
2022-02-25 09:51:51 +00:00
<!-- Content Column -->
< div class = "col-md-9" id = "tg-sb-content" >
2022-03-24 11:42:13 +00:00
< header >
< div class = "row" >
< div class = "col-lg-12" >
< a href = "./" role = "button"
>< i class = "fa fa-home fa-lg" > < /i
>< / a >
» Defining rule properties
< a
target="_blank"
2024-09-26 14:58:11 +00:00
href="https://github.com/pmd/pmd/blob/main/docs/pages/pmd/userdocs/extending/defining_properties.md"
2023-03-03 20:15:41 +00:00
class="float-right"
2022-03-24 11:42:13 +00:00
role="button"
2023-03-16 10:46:54 +00:00
>< i class = "fab fa-github fa-lg" > < / i > Edit on GitHub< /a
2022-03-24 11:42:13 +00:00
>
< / div >
< / div >
< hr / >
< / header >
< div class = "post-header" >
2020-11-14 09:20:07 +00:00
< h1 class = "post-title-main" > Defining rule properties< / h1 >
< / div >
2024-09-26 14:58:11 +00:00
< div class = "post-content" data-github-edit-url = "https://github.com/pmd/pmd/blob/main/docs/pages/pmd/userdocs/extending/defining_properties.md" >
2020-11-14 09:20:07 +00:00
< div class = "summary" > Learn how to define your own properties both for Java and XPath rules.< / div >
2023-08-22 15:14:23 +00:00
< details id = "inline-toc-details" >
< summary > Table of Contents< / summary >
< div id = "inline-toc" > <!-- empty, move TOC here when screen size too small --> < / div >
< / details >
2023-03-03 20:15:41 +00:00
2020-11-14 09:20:07 +00:00
< p > Rule properties are a way to make your rules configurable directly from the
2024-01-12 09:15:55 +00:00
ruleset XML. Their usage is described on the
< a href = "pmd_userdocs_configuring_rules.html#rule-properties" > Configuring Rules< / a > page.< / p >
2020-11-14 09:20:07 +00:00
< p > If you’ re a rule developer, you may want to think about what would be useful for
2024-01-12 09:15:55 +00:00
a user of your rule to parameterize. It could be a numeric report level, a boolean
2020-11-14 09:20:07 +00:00
flag changing the behaviour of your rule… Chances are there < em > is< / em > some detail
that can be abstracted away from your implementation, and in that case, this
page can help you squeeze that sweet flexibility out of your rule.< / p >
< h2 id = "overview-of-properties" > Overview of properties< / h2 >
2024-01-12 09:15:55 +00:00
< p > The basic thing you need to do as a developer is to define a < strong > property descriptor< / strong > and declare
that your rule uses it. A property descriptor defines a number of attributes for your property:< / p >
2020-11-14 09:20:07 +00:00
< ul >
< li > Its < em > name< / em > , with which the user will refer to your property;< / li >
< li > Its < em > description< / em > , for documentation purposes;< / li >
< li > Its < em > default value< / em > < / li >
< / ul >
2023-08-22 18:48:34 +00:00
< p > All of these attributes can be specified in a single Java statement (or XML element for XPath rules).< / p >
2020-11-14 09:20:07 +00:00
< h2 id = "for-java-rules" > For Java rules< / h2 >
< p > The procedure to define a property is quite straightforward:< / p >
< ul >
< li > Create a property descriptor of the type you want, by using a
2024-10-25 08:59:30 +00:00
builder from < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertyFactory.html#" > < code > PropertyFactory< / code > < / a > < / li >
< li > Call < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertySource.html#definePropertyDescriptor(net.sourceforge.pmd.properties.PropertyDescriptor)" > < code > definePropertyDescriptor(PropertyDescriptor)< / code > < / a >
2024-01-12 09:15:55 +00:00
in the rule’ s noarg constructor.< / li >
2020-11-14 09:20:07 +00:00
< / ul >
2024-01-12 09:15:55 +00:00
< p > You can then retrieve the value of the property at any time using
2024-10-25 08:59:30 +00:00
< a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertySource.html#getProperty(net.sourceforge.pmd.properties.PropertyDescriptor)" > < code > getProperty(PropertyDescriptor)< / code > < / a > .< / p >
2024-01-12 09:15:55 +00:00
2024-10-25 08:59:30 +00:00
< p > Note: The base class for all rule implementations is < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/lang/rule/AbstractRule.html#" > < code > AbstractRule< / code > < / a > , which
is a < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertySource.html#" > < code > PropertySource< / code > < / a > . So you can directly call < code class = "language-plaintext highlighter-rouge" > definePropertyDescriptor(...)< / code >
2024-01-12 09:15:55 +00:00
or < code class = "language-plaintext highlighter-rouge" > getProperty(...)< / code > within your rule.< / p >
2020-11-14 09:20:07 +00:00
< h3 id = "creating-a-descriptor" > Creating a descriptor< / h3 >
< p > Properties can be built using type-specific < strong > builders< / strong > , which can be obtained
2024-10-25 08:59:30 +00:00
from the factory methods of < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertyFactory.html#" > < code > PropertyFactory< / code > < / a > . For example, to build a
2023-08-22 18:48:34 +00:00
string property, you would call< / p >
2020-11-14 09:20:07 +00:00
< div class = "language-java highlighter-rouge" > < div class = "highlight" > < pre class = "highlight" > < code > < span class = "nc" > PropertyFactory< / span > < span class = "o" > .< / span > < span class = "na" > stringProperty< / span > < span class = "o" > (< / span > < span class = "s" > "myProperty"< / span > < span class = "o" > )< / span >
< span class = "o" > .< / span > < span class = "na" > desc< / span > < span class = "o" > (< / span > < span class = "s" > "This is my property"< / span > < span class = "o" > )< / span >
< span class = "o" > .< / span > < span class = "na" > defaultValue< / span > < span class = "o" > (< / span > < span class = "s" > "foo"< / span > < span class = "o" > )< / span >
< span class = "o" > .< / span > < span class = "na" > build< / span > < span class = "o" > ();< / span >
< / code > < / pre > < / div > < / div >
2024-01-12 09:15:55 +00:00
< p > This is fairly more readable than a constructor call, but keep in mind the description
and the default value are not optional.< / p >
2020-11-14 09:20:07 +00:00
< p > For < strong > numeric properties< / strong > , you can add constraints on the range of acceptable values, e.g.< / p >
< div class = "language-java highlighter-rouge" > < div class = "highlight" > < pre class = "highlight" > < code > < span class = "nc" > PropertyFactory< / span > < span class = "o" > .< / span > < span class = "na" > intProperty< / span > < span class = "o" > (< / span > < span class = "s" > "myIntProperty"< / span > < span class = "o" > )< / span >
< span class = "o" > .< / span > < span class = "na" > desc< / span > < span class = "o" > (< / span > < span class = "s" > "This is my property"< / span > < span class = "o" > )< / span >
< span class = "o" > .< / span > < span class = "na" > defaultValue< / span > < span class = "o" > (< / span > < span class = "mi" > 3< / span > < span class = "o" > )< / span >
2023-08-22 18:48:34 +00:00
< span class = "o" > .< / span > < span class = "na" > require< / span > < span class = "o" > (< / span > < span class = "n" > positive< / span > < span class = "o" > ())< / span > < span class = "c1" > // must be > 0< / span >
< span class = "o" > .< / span > < span class = "na" > require< / span > < span class = "o" > (< / span > < span class = "n" > below< / span > < span class = "o" > (< / span > < span class = "mi" > 100< / span > < span class = "o" > ))< / span > < span class = "c1" > // must be < = 100< / span >
2020-11-14 09:20:07 +00:00
< span class = "o" > .< / span > < span class = "na" > build< / span > < span class = "o" > ();< / span >
< / code > < / pre > < / div > < / div >
2024-10-25 08:59:30 +00:00
< p > Predefined constraints such as < code class = "language-plaintext highlighter-rouge" > positive< / code > and < code class = "language-plaintext highlighter-rouge" > below< / code > are available in the class < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/NumericConstraints.html#" > < code > NumericConstraints< / code > < / a > .
A custom constraint can be implemented by implementing the interface < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertyConstraint.html#" > < code > PropertyConstraint< / code > < / a > .< / p >
2020-11-14 09:20:07 +00:00
2023-08-22 18:48:34 +00:00
< p > < strong > Enumerated properties< / strong > do not have a specific value type, instead,
2020-11-14 09:20:07 +00:00
you can choose any type of value, provided the values are from a closed set. To make
that actionable, you give string labels to each of the acceptable values, and the user
will provide one of those labels as a value in the XML. The property will give you back
the associated value, not the label. Here’ s an example:< / p >
2023-08-22 18:48:34 +00:00
< div class = "language-java highlighter-rouge" > < div class = "highlight" > < pre class = "highlight" > < code > < span class = "kd" > enum< / span > < span class = "nc" > Mode< / span > < span class = "o" > {< / span >
< span class = "nc" > Easy< / span > < span class = "o" > ,< / span > < span class = "nc" > Hard< / span >
2020-11-14 09:20:07 +00:00
< span class = "o" > }< / span >
2023-08-22 18:48:34 +00:00
< span class = "c1" > // Using this method, the labels are the `toString` of each enum constant.< / span >
< span class = "c1" > // To customize this look at the overloads of `enumProperty`.< / span >
< span class = "kd" > static< / span > < span class = "nc" > PropertyDescriptor< / span > < span class = "o" > < < / span > < span class = "nc" > Mode< / span > < span class = "o" > > < / span > < span class = "n" > modeProperty< / span >
< span class = "o" > =< / span > < span class = "nc" > PropertyFactory< / span > < span class = "o" > .< / span > < span class = "na" > enumProperty< / span > < span class = "o" > (< / span > < span class = "s" > "modeProperty"< / span > < span class = "o" > ,< / span > < span class = "nc" > Mode< / span > < span class = "o" > .< / span > < span class = "na" > class< / span > < span class = "o" > )< / span >
2020-11-14 09:20:07 +00:00
< span class = "o" > .< / span > < span class = "na" > desc< / span > < span class = "o" > (< / span > < span class = "s" > "This is my property"< / span > < span class = "o" > )< / span >
2023-08-22 18:48:34 +00:00
< span class = "o" > .< / span > < span class = "na" > defaultValue< / span > < span class = "o" > (< / span > < span class = "nc" > Mode< / span > < span class = "o" > .< / span > < span class = "na" > Easy< / span > < span class = "o" > )< / span >
2020-11-14 09:20:07 +00:00
< span class = "o" > .< / span > < span class = "na" > build< / span > < span class = "o" > ();< / span >
< / code > < / pre > < / div > < / div >
< h3 id = "example" > Example< / h3 >
2024-09-26 14:58:11 +00:00
< p > You can see an example of properties used in a PMD rule such as < a href = "https://github.com/pmd/pmd/blob/main/pmd-java/src/main/java/net/sourceforge/pmd/lang/java/rule/bestpractices/AvoidReassigningLoopVariablesRule.java#L40" > AvoidReassigningLoopVariables< / a > .
2020-11-14 09:20:07 +00:00
There are several things to notice here:< / p >
< ul >
< li > The property descriptors are declared < code class = "language-plaintext highlighter-rouge" > static final< / code > , which should generally be
the case, as descriptors are immutable and can be shared between instances of the same rule;< / li >
2024-10-25 08:59:30 +00:00
< li > The property is declared using < a href = "https://docs.pmd-code.org/apidocs/pmd-core/7.8.0-SNAPSHOT/net/sourceforge/pmd/properties/PropertySource.html#definePropertyDescriptor(net.sourceforge.pmd.properties.PropertyDescriptor)" > < code > definePropertyDescriptor< / code > < / a > ` < em > in the constructor< / em > ,
2020-11-14 09:20:07 +00:00
which ensures the property gets recognised by PMD at the time the properties
are overridden (which happens before rule execution);< / li >
< li > The value of the property is < em > not retrieved in the constructor< / em > , but in one of
the < code class = "language-plaintext highlighter-rouge" > visit< / code > methods (typically on the highest node in the tree, since the property
doesn’ t change).< / li >
< / ul >
< h2 id = "for-xpath-rules" > For XPath rules< / h2 >
2024-01-12 09:15:55 +00:00
< p > XPath rules can also define their own properties. To do so, you must add a < code class = "language-plaintext highlighter-rouge" > property< / code > element in
the < code class = "language-plaintext highlighter-rouge" > properties< / code > element of your rule, which < strong > declares the < code class = "language-plaintext highlighter-rouge" > type< / code > attribute< / strong > . This attribute conditions
what type the underlying property has, and can have the following values:< / p >
2020-11-14 09:20:07 +00:00
< table >
< thead >
< tr >
< th > < code class = "language-plaintext highlighter-rouge" > type< / code > attribute< / th >
< th > XSD type< / th >
< / tr >
< / thead >
< tbody >
< tr >
< td > Integer< / td >
< td > xs:integer< / td >
< / tr >
< tr >
< td > Long< / td >
< td > xs:integer< / td >
< / tr >
< tr >
< td > Double< / td >
< td > xs:decimal< / td >
< / tr >
< tr >
< td > Boolean< / td >
< td > xs:boolean< / td >
< / tr >
< tr >
< td > String< / td >
< td > xs:string< / td >
< / tr >
< tr >
< td > Character< / td >
< td > xs:string< / td >
< / tr >
< tr >
< td > Regex< / td >
< td > xs:string< / td >
< / tr >
< / tbody >
< / table >
2024-01-12 09:15:55 +00:00
< p > Note that enumerated properties are not available in XPath rules.< / p >
2020-11-14 09:20:07 +00:00
< p > Properties defined in XPath also < em > must< / em > declare the < code class = "language-plaintext highlighter-rouge" > description< / code > attribute.
Numeric properties also expect the < code class = "language-plaintext highlighter-rouge" > min< / code > and < code class = "language-plaintext highlighter-rouge" > max< / code > attributes for now. Here are
a few examples to sum it up:< / p >
< div class = "language-xml highlighter-rouge" > < div class = "highlight" > < pre class = "highlight" > < code > < span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "stringProp"< / span > < span class = "na" > type=< / span > < span class = "s" > "Boolean"< / span > < span class = "na" > value=< / span > < span class = "s" > "true"< / span > < span class = "na" > description=< / span > < span class = "s" > "A BooleanProperty."< / span > < span class = "nt" > /> < / span >
< span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "intProp"< / span > < span class = "na" > type=< / span > < span class = "s" > "Integer"< / span > < span class = "na" > value=< / span > < span class = "s" > "3"< / span > < span class = "na" > min=< / span > < span class = "s" > "1"< / span > < span class = "na" > max=< / span > < span class = "s" > "20"< / span > < span class = "na" > description=< / span > < span class = "s" > "An IntegerProperty."< / span > < span class = "nt" > /> < / span >
< / code > < / pre > < / div > < / div >
< p > You can then use the property in XPath with the syntax < code class = "language-plaintext highlighter-rouge" > $propertyName< / code > , for example:< / p >
< div class = "language-xml highlighter-rouge" > < div class = "highlight" > < pre class = "highlight" > < code > < span class = "nt" > < rule< / span > < span class = "na" > name=< / span > < span class = "s" > "MyXpathRule"< / span > < span class = "err" > ...< / span > < span class = "nt" > > < / span >
< span class = "nt" > < properties> < / span >
< span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "maxStatements"< / span > < span class = "na" > type=< / span > < span class = "s" > "Integer"< / span > < span class = "na" > value=< / span > < span class = "s" > "10"< / span > < span class = "na" > min=< / span > < span class = "s" > "1"< / span > < span class = "na" > max=< / span > < span class = "s" > "40"< / span >
< span class = "na" > description=< / span > < span class = "s" > "Max number of statements per method"< / span > < span class = "nt" > /> < / span >
< span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "xpath"< / span > < span class = "nt" > > < / span >
2024-01-12 09:15:55 +00:00
< span class = "nt" > < value> < / span >
< span class = "cp" > < ![CDATA[
//MethodDeclaration/Block[count(./*) > < / span > $maxStatements]
]]> < span class = "nt" > < /value> < / span >
< span class = "nt" > < /property> < / span >
2020-11-14 09:20:07 +00:00
< span class = "nt" > < /properties> < / span >
< span class = "nt" > < /rule> < / span >
< / code > < / pre > < / div > < / div >
< h3 id = "multivalued-properties" > Multivalued properties< / h3 >
< p > Multivalued properties are also allowed and their < code class = "language-plaintext highlighter-rouge" > type< / code > attribute has the form
2023-08-22 18:48:34 +00:00
< code class = "language-plaintext highlighter-rouge" > List[Boolean]< / code > or < code class = "language-plaintext highlighter-rouge" > List[Character]< / code > , with every above type allowed. These properties
make use of the < strong > sequence datatype< / strong > provided by XPath 2.0 and above.
When providing multiple values, the delimiter is a simple comma (“< code class = "language-plaintext highlighter-rouge" > ,< / code > ”). The comma can be escaped
with a backslash when needed.< / p >
2020-11-14 09:20:07 +00:00
< div class = "language-xml highlighter-rouge" > < div class = "highlight" > < pre class = "highlight" > < code > < span class = "nt" > < rule< / span > < span class = "na" > name=< / span > < span class = "s" > "MyXpathRule"< / span > < span class = "err" > ...< / span > < span class = "nt" > > < / span >
< span class = "nt" > < properties> < / span >
< span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "intProp"< / span > < span class = "na" > type=< / span > < span class = "s" > "List[Integer]"< / span > < span class = "na" > value=< / span > < span class = "s" > "1,2,5"< / span > < span class = "na" > description=< / span > < span class = "s" > "An IntegerMultiProperty."< / span > < span class = "nt" > /> < / span >
2023-08-22 18:48:34 +00:00
< span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "reportedIdentifiers"< / span > < span class = "na" > type=< / span > < span class = "s" > "List[String]"< / span > < span class = "na" > value=< / span > < span class = "s" > "foo,bar"< / span >
2020-11-14 09:20:07 +00:00
< span class = "na" > description=< / span > < span class = "s" > "A StringMultiProperty."< / span > < span class = "nt" > /> < / span >
< span class = "nt" > < property< / span > < span class = "na" > name=< / span > < span class = "s" > "xpath"< / span > < span class = "nt" > > < / span >
2024-01-12 09:15:55 +00:00
< span class = "nt" > < value> < / span > < span class = "cp" > < ![CDATA[
2024-01-26 15:09:56 +00:00
//VariableId[@Image = $reportedIdentifiers]
2024-01-12 09:15:55 +00:00
]]> < / span > < span class = "nt" > < /value> < / span >
< span class = "nt" > < /property> < / span >
2020-11-14 09:20:07 +00:00
< span class = "nt" > < /properties> < / span >
< span class = "nt" > < /rule> < / span >
< / code > < / pre > < / div > < / div >
2024-01-12 09:15:55 +00:00
< p > Notice that in the example above, < code class = "language-plaintext highlighter-rouge" > @Name = $reportedIdentifiers< / code > doesn’ t test
< code class = "language-plaintext highlighter-rouge" > @Name< / code > for equality with the whole sequence < code class = "language-plaintext highlighter-rouge" > ('foo', 'bar')< / code > , it tests whether
the sequence < em > contains< / em > < code class = "language-plaintext highlighter-rouge" > @Name< / code > . That is, the above rule will report all variables
2020-11-14 09:20:07 +00:00
named < code class = "language-plaintext highlighter-rouge" > foo< / code > or < code class = "language-plaintext highlighter-rouge" > bar< / code > . All other XPath 2.0 < a href = "https://www.w3.org/TR/xpath-functions/#sequence-functions" > functions operating on sequences< / a >
are supported.< / p >
2023-03-03 20:15:41 +00:00
< div class = "alert alert-success" role = "alert" > < i class = "fas fa-check-square-o" > < / i > < b > Tip:< / b > You can also < a href = "pmd_userdocs_extending_designer_reference.html#rule-properties" > define properties directly in the designer< / a > < / div >
2020-11-14 09:20:07 +00:00
< div class = "tags" >
< b > Tags: < / b >
2023-03-03 20:15:41 +00:00
< a href = "tag_extending.html" class = "btn btn-outline-secondary navbar-btn cursorNorm" role = "button" > extending< / a >
2020-11-14 09:20:07 +00:00
2023-03-03 20:15:41 +00:00
< a href = "tag_userdocs.html" class = "btn btn-outline-secondary navbar-btn cursorNorm" role = "button" > userdocs< / a >
2020-11-14 09:20:07 +00:00
< / div >
< / div >
< footer >
2022-03-24 11:42:13 +00:00
< hr / >
< div >
This documentation is written in markdown. < br / >
If there is something missing or can be improved, edit this page on
github and create a PR:
< a
target="_blank"
2024-09-26 14:58:11 +00:00
href="https://github.com/pmd/pmd/blob/main/docs/pages/pmd/userdocs/extending/defining_properties.md"
2022-03-24 11:42:13 +00:00
role="button"
2023-04-19 03:50:18 +00:00
>< i class = "fab fa-github fa-lg" > < / i > Edit on GitHub< /a
2022-03-24 11:42:13 +00:00
>
< / div >
< hr / >
< div class = "row" >
< div class = "col-lg-12 footer" >
2024-01-05 14:22:28 +00:00
© 2024 PMD Open Source Project. All rights
2022-03-24 11:42:13 +00:00
reserved. < br / >
< span > Page last updated:< / span >
2024-11-21 14:49:47 +00:00
December 2023 (7.0.0)< br / > Site last generated: Nov 21, 2024 < br / >
2022-03-24 11:42:13 +00:00
< p >
2023-03-03 20:15:41 +00:00
< img src = "images/logo/pmd-logo-70px.png" alt = "PMD
2022-03-24 11:42:13 +00:00
logo"/>
< / p >
< / div >
< / div >
2020-11-14 09:20:07 +00:00
< / footer >
2022-02-25 09:51:51 +00:00
< / div >
<!-- /.row -->
2020-11-14 09:20:07 +00:00
< / div >
2022-02-25 09:51:51 +00:00
<!-- /.container -->
< / div >
<!-- Sticky TOC column -->
< div class = "toc-col" >
2022-03-24 11:42:13 +00:00
< div id = "toc" > < / div >
2020-11-14 09:20:07 +00:00
< / div >
2022-02-25 09:51:51 +00:00
<!-- /.toc - container - wrapper -->
2020-11-14 09:20:07 +00:00
< / div >
< / div >
2023-03-03 20:15:41 +00:00
< script type = "application/javascript" src = "assets/jquery-3.5.1/jquery-3.5.1.min.js" > < / script >
< script type = "application/javascript" src = "assets/anchorjs-4.2.2/anchor.min.js" > < / script >
< script type = "application/javascript" src = "assets/navgoco-0.2.1/src/jquery.navgoco.min.js" > < / script >
< script type = "application/javascript" src = "assets/bootstrap-4.5.2-dist/js/bootstrap.bundle.min.js" > < / script >
< script type = "application/javascript" src = "assets/Simple-Jekyll-Search-1.0.8/dest/jekyll-search.js" > < / script >
< script type = "application/javascript" src = "assets/jekyll-table-of-contents/toc.js" > < / script >
< script type = "application/javascript" src = "js/tabstate.js" > < / script >
< script type = "application/javascript" src = "js/customscripts.js" > < / script >
< / body >
2020-11-14 09:20:07 +00:00
< / html >