-
-
Notifications
You must be signed in to change notification settings - Fork 1.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
[kotlin] Add support for Kotlin #419
Comments
As a side note. Lint will start operating on UAST (Universal Abstract Syntax Tree) which means, that the Lint rules will also work with Kotlin. It might be possible to achieve the same here since some of the rules are working on an abstraction of the syntax, if not all (not that familiar with the internals of PMD). |
The method for adding new language support to PMD is well established. The particulars are unique to the language however. PMD is basically an AST visitation framework, with Rules being unique visitation patterns. Some languages have custom parsers/AST created (e.g. Java), while others leverage available object graphs to wrap as AST (e.g. XML DOM). If a canonical parser/AST is available this could be used to quickly bootstrap Kotlin language support. Since languages evolve over time, it is useful for the AST to be relatively stable across versions of the language, or Rules may stop working. A general outline of this process is here: https://pmd.github.io/pmd-5.7.0/customizing/new-language.html AFAIK, there is not currently support for meta-language Rules, so if you want a Rule that works for Java to work for Kotlin, you'd need to re-implement it for Kotlin. |
Hi guys, any news here? Are you planing to do it? |
@xanderblinov this is still on the roadmap. As previously stated, this needs either:
We are inclined on 1 as it's a one time cost for multiple languages we want to fully support (swift, kotlin, go...), while 2 is an effort needed per language. However, PRs are always welcomed to help get this shipped faster. If anyone wants to contribute, reach out to me on gitter https://gitter.im/pmd/pmd |
Kotlin developer here. |
@jsotuyod what do you think about directly complier call and using UAST? |
I'm hesitant to do so. It would require PMD to bundle the kotlin compiler, which is dependent on the Kotlin version in use (experimental features may break even in minor versions, and there are no compatibility promises for an eventual Kotlin 2 release, as per https://kotlinlang.org/docs/reference/compatibility.html). As far as I know, the Kotlin compiler, unlike the Java compiler, can't be told "this code is in version X", so supporting multiple Kotlin versions require bundling different versions of the compiler, which in turn means a really heavyweight PMD binary, and not doing so means forcing users to stick to an outdated PMD version or upgrade their codebases to whatever Kotlin version we decide to support. Unless IntelliJ decides to join forces with us (I'd love to!), the same way Salesforce did for Apex / VisualForce support, and work with us to avoid these issues, I've some doubts this approach will be sustainable. Using the Kotlin parser to get the UAST still needs mapping those nodes to our own On the other hand, we do have type and reference resolution in place for JVM-based languages. I know Kotlin can compile to other platforms, but probably most of our users for the time being are focused on Android support, so we are probably just fine were we stand at the moment. |
This is technically a "Fork" of the work done by Poeschl which is, itself, a fork of the People are welcome to expand upon this project and make the tests pass. I had to kill a lot of the old tests because the compiler API's that were used to make the checks were removed/changed. I'd like to publish this project eventually so I can use the CPD feature in my companies builds. I'm also happy to roll this into the PMD project if there's an interest. |
Do I understand correctly that the latest PMD release (6.10.0) supports CPD for Kotlin? 🙏 |
@Zhuinden yes, you understand perfectly. |
For everyone following this, starting from PMD 6.13.0 the CPD support for Kotlin is now complete, which means it now supports suppressions through comments and Full Kotlin support for PMD rules will probably be part of PMD 7.0.0 (although we may have no actual rules at that point yet, but contributions are welcomed) |
We will also add a wiki page so you can enable full CPD support for your second favourite language ❤️ in a few steps |
@jsotuyod are there any rules for Kotlin? What one has to do to add new rules for Kotlin? |
Curious about the status of Kotlin support by PMD. Is it going to be in 7.0? Is there an early version to try out? |
If someone wants to start the Kotlin for PMD approach, I would suggest to use PMD 7 as the basis, as there we have support for Antlr-based grammars. There is now a official antlr-grammar available for Kotlin - https://github.com/Kotlin/kotlin-spec Having support for a language is one piece, the other piece would be to provide interesting and useful rules. |
We are looking into this. What would be a good way to contribute? Open an issue to discuss it? |
Thanks for your offer. I think, this issue here is the issue to discuss adding support for Kotlin. PMD 7 development happens in the branch In PMD 7, there is one language that uses Antlr, where you can have a look, how this is integrated: Swift. The PMD 7 documentation page is here: https://docs.pmd-code.org/pmd-doc-7.0.0-SNAPSHOT/ - e.g. Swift has 4 rules. The Antlr integration documentation is not integrated yet - so the docs Adding PMD support for a new language only describes the javacc way. But there is also a pull request #1881 which contains the new doc for Antlr-based languages. |
Hi Andreas,
Good to know, the PR #1881 documentation is what we need.
I cloned team-raptor/pmd, went to 7.0 branch and found I need to generate
those docs.
To generate the doc, I found need jekyll and bundle. For that I installed
ruby. For that I installed homebrew. And then I get into all kinds of
dependency errors between various gems and ruby.
Do you maybe have an easier way to get to that documentation? Thanks!
|
@jborgers I've updated the documentation and merged the PR. The doc is now available at https://docs.pmd-code.org/pmd-doc-7.0.0-SNAPSHOT/pmd_devdocs_major_adding_new_language_antlr.html Please let us know, if there is anything wrong/unclear/doesn't work. Also make sure to clone pmd/pmd - be careful when cloning forks, they might not be up-to-date... |
@adangel thanks, this is really helpful. How can I provide feedback on typo's/unclearness etc.? Yes, using my fresh fork now (jborgers/pmd), branch 7.0 |
Interesting suggestion. I just did. The parser grammar needs to be the same as the file name, so Kotlin instead of KotlinParser. The result is that indeed the KotlinVisitor etc. classes have the right name, however, the KotlinParser class is now called Kotlin. So maybe this renaming needs to be automated in one of the scripts or a pom? |
My generated KotlinParser now has errors: static classes extending org.antlr.v4.runtime.ParserRuleContext call super.acceptVisitor(visitor, data) while this method does not exist. In SwiftParser, I see the static classes extend Swift specific classes. That didn't get generated for Kotlin(Parser). |
@adangel @oowekyala Help appreciated, I am stuck. |
I'm not sure where you pushed your changes. Could you open a draft PR with the relevant branch, and give us write access (a checkbox when you open the PR)? This way we could take a look easily |
Okay, I just did. Let me know if easily taking a look works now |
Great, it seems to work now! With generated files, all compiles. |
I mostly just fixed the Ant script, I'll update the docs later.
The designer should just work, but you'll have to follow the instructions at the top of this page to use it with pmd 7. |
The syntax tree is correct - that's how the grammar is written. @oowekyala made me aware, that ANTLR doesn't create a abstract syntax tree but instead creates a parse tree. And that is, what you are seeing. Basically everything down to the tokens is a node in the tree. What we would need is some kind of post-processing of the parse tree, to make it look nicer and simpler to use for static analysis. E.g. the name of the function would be a perfect fit for an attribute on the node "FunctionDeclaration" (instead of having the node "T-Identifier"). The deep tree also comes from the expression production. In the Javacc grammar for Java, we already simplify this part of the tree with the help of a javacc feature (conditional tree - it creates only a tree node if there are >1 nodes). While the tree doesn't look as good as it could, it's still usable. Most important is, that you write tests for the rules, you create (see https://pmd.github.io/latest/pmd_userdocs_extending_testing.html) because then you can ensure that the rules work in case we improve the tree later. |
Thanks. Yes, we will create tests for the rules, like we have for our performance Java rules. Starting with the designer to create the first Kotlin XPath rule, we have some questions:
Update: it seems doable to create a custom XPath function for kotlin in a similar way as in
object TestFoo {
fun testBar() = 1
} |
@adangel @oowekyala any ideas? |
I'm afraid all of those are actually very real limitations, for which I have no good solution...
|
@oowekyala thanks for your elaborate answer! As there are not many options for point 1 and 2, we will start by creating a very simple XPath rule and make some unit tests. For point 3: I checked the same kotlin code in the |
That was my doing, putting everything in one .g4 file like for Swift. It seemed to work. Great you solved it. |
Update:
Next, we will try to migrate one of our Java performance rules into a Kotlin rule and we will check if we can call the "FunctionNameTooShort" from existing tools/plugins, e.g. in Sonar. |
We are still working on this, see: [kotlin] Kotlin support #3505 |
Status update for this issue:
I'll create issues for these points and close this one as completed. |
Since Google announced official support for Kotlin, a lot more people are going to be using the language (including me 😀) so it would be nice to have some rules for it.
The text was updated successfully, but these errors were encountered: