Skip to content
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

vim: Add :RustRun and associated commands #14480

Closed
wants to merge 4 commits into from

Conversation

lilyball
Copy link
Contributor

Define new vim commands :RustRun, :RustExpand, :RustEmitIr, and :RustEmitAsm, including full documentation on these commands and all the options.

@lilyball
Copy link
Contributor Author

cc @chris-morgan

@Valloric
Copy link
Contributor

Naming the commands Run and Expand is a very bad idea; there's only one namespace for commands in Vim and taking up a name as generic and common as Run without a prefix is certainly going to conflict with other Vim plugins. It's also just bad Vim plugin hygiene.

It's customary for Vim plugins to prefix all of their commands with a custom prefix. Or just use uncommon command names.

Please don't exacerbate the problem of Vim plugins colliding. We have enough of that as it is in the Vim community.

@lilyball
Copy link
Contributor Author

@Valloric They're buffer-local commands that are only defined when the buffer has the rust filetype.

@richo
Copy link
Contributor

richo commented May 30, 2014

I have some FUD about :Run executing arbitrary code. Not sure that it shouldn't be a thing, but I'm wonder if it should have to be explicitly bound up in the vimrc and just exposed as a function by default.

With that said, I would love if the compile step could be decoupled, and hooked up to the :make infrastructure to populate the quickfix list with any errors or warnings (I use :make a ton for that reason)

@Valloric
Copy link
Contributor

@kballard I've noticed that, and while it makes the problem less of an issue than otherwise, it's still a bad idea to keep the names. Most plugins aren't filetype-specific so will collide with these names if they use Run (which they shouldn't, but one problem at a time).

In general, unprefixed commands are left for the user to create/map in their vimrc. Again, this is basic Vim plugin hygiene.

@lilyball
Copy link
Contributor Author

@richo Not sure what you mean about FUD. The whole point is to provide something useful. :Run can't really be run by accident. Forcing people to bind their own mapping/command to the function will just make it harder for non-vim-scripters to actually use the functionality.

As for :make, that won't be compatible with being able to run unnamed buffers (because it isn't designed for that). As for quickfix, I highly recommend using Syntastic. One interesting side-effect of how :Run works is it will trigger Syntastic warnings even for unnamed buffers (because the buffer temporarily gains a name), assuming you have Syntastic configured to check on save.

@lilyball
Copy link
Contributor Author

@Valloric Any non-buffer-local, non-filetype-specific plugin that uses the name :Run is a bad plugin. A non-prefixed name like that seems largely appropriate only for buffer-local, filetype-specific plugins, precisely because that way it won't conflict.

In general, unprefixed commands are left for the user to create/map in their vimrc.

That's a reasonable argument. But what about mappings? You can't prefix those. It seems mildly odd to require prefixes on commands when mappings tend to be more common and more likely to conflict.

Again, this is basic Vim plugin hygiene.

Is there any authoritative source for this, or just your personal opinion/experience? I certainly agree with you that commands should be prefixed if they are global, but I'm not convinced that's necessary for buffer-local filteype-specific commands.

As an example of precedent, Vim's built-in ftplugin/gitrebase.vim (used for interactive Git rebases) defines a buffer-local set of commands :Pick/:Squash/:Edit/:Reword/:Fixup/:Cycle. This seems like official blessing for using unprefixed commands in this scenario.

@lilyball
Copy link
Contributor Author

Here's an online source for ftplugin/gitrebase.vim: https://github.com/tpope/vim-git/blob/master/ftplugin/gitrebase.vim

Note that this is by Tim Pope, which seems pretty authoritative to me.

@richo
Copy link
Contributor

richo commented May 30, 2014

Sorry, that was very much two different points.

I'm not sure I agree with you about barrier to entry. It just means that instead of saying:

You an use :Run to ...

instead it should read:

if you put command Run call RustRun in your vimrc you can use :Run to ....

I'm not convinced that's too much extra hassle.

the :make thing was entirely a wishlist thing. Not a blocker, although I'm guessing that the machinery to do it does exist somewhere (given the output of the shell command, shove it into the thing that populates the quickfix list). I'm happy to tackle this if noone beats me to it.

@lilyball
Copy link
Contributor Author

@richo Telling people that they have to put some arcane command line in their vimrc in order to use core functionality seems like a rather serious issue. I can't think of a single plugin that has ever asked me to define my own command. And rarely am I even required to define my own mapping (plugins usually provide default mappings, with a variable to suppress that so you can provide your own in case of conflicts).

@Valloric
Copy link
Contributor

Any non-buffer-local, non-filetype-specific plugin that uses the name :Run is a bad plugin.

Agreed 100%, but they certainly do exist. I'm actually far more worried about this conflicting with a user's custom command.

That's a reasonable argument. But what about mappings? You can't prefix those. It seems mildly odd to require prefixes on commands when mappings tend to be more common and more likely to conflict.

And that's why plugins shouldn't be creating default mappings unless they are critical to the functioning of the plugin and even then, there must be a way for the user to customize the mapping the plugin creates.

You haven't allowed that in your plugin; it's Run or errors about conflicts.

Is there any authoritative source for this, or just your personal opinion/experience? I certainly agree with you that commands should be prefixed if they are global, but I'm not convinced that's necessary for buffer-local filteype-specific commands.

It's less necessary, but even such buffer-local commands conflict with global commands. I can guarantee you without a doubt that there's a non-trivial number of users out there with Run as a custom command in their vimrc. And theirs isn't buffer-local.

As an example of precedent, Vim's built-in ftplugin/gitrebase.vim (used for interactive Git rebases) defines a buffer-local set of commands :Pick/:Squash/:Edit/:Reword/:Fixup/:Cycle. This seems like official blessing for using unprefixed commands in this scenario.

Tim Pope is great, but IMO that was a bad call.

Is renaming Run to something like RustRun such a big deal? I have more experience maintaining Vim plugins than I'd like to have and trust me when I tell you that anything that can conflict will conflict. You wouldn't believe the vimrcs out there.

let &foldenable = foldenable
endfunction

function! <SID>Jump_Back()
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why <SID>? Just prefix with s:. Same below.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Good point. This was a copy & paste, but it's certainly worth fixing.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Fixed

@lilyball
Copy link
Contributor Author

Is renaming Run to something like RustRun such a big deal?

It reduces the usability of the command. It's not a huge issue, but I'd rather produce something that's usable and matches existing precedent, then reduce usability for the sake of an overabundance of caution.

If a user has a global command named :Run in their vimrc, well, it will get overwritten when they set their filetype to rust. The plugin uses command! so it won't error out (I did this for ease of debugging when reloading the script repeatedly, but it has the effect here of overwriting a badly-named global command like that).

Do you have any strong precedent you can cite for using prefixed command names in filetype plugins?

@Valloric
Copy link
Contributor

It reduces the usability of the command.

Vim commands are far better off being verbose and descriptive; if they're something the user wants to use often, they can trivially create a mapping: noremap <F5> :RustRun<cr>. Done. If they're used infrequently, then command completion works great.

This is the approach YCM uses (it even recommends custom mappings for commands) and I've never had a single complaint about YCM command name lengths being so long that they hurt usability. And the names are really long; one of the most frequently used YCM commands is YcmForceCompileAndDiagnostics, but it's not an issue since everyone is encouraged to map it to something more convenient.

The plugin uses command! so it won't error out

So it will silently override the user's command. That's even worse; the problem with mappings silently overriding other mappings by default is the reason why we got <unique> as an argument to map (which causes a noisy conflict warning). If the map command were being implemented today, <unique> would be the default.

In other words, you want to tell the user about a conflict so that they can resolve it instead of having something silently override.

Do you have any strong precedent you can cite for using prefixed command names in filetype plugins?

I don't, but your argument boils down to "Others have made this mistake before so why can't I make it too?" I've lost track of the number of hacks I've had to add to YCM because other plugin authors haven't been as diligent as they should have.

@Valloric
Copy link
Contributor

Do you have any strong precedent you can cite for using prefixed command names in filetype plugins?

In fact, I do have it. YCM creates several commands that could easily be filetype-specific (like YcmForceCompileAndDiagnostics, YcmDiags, YcmShowDetailedDiagnostic etc). They're not registered only for specific filetypes because IMO I'd rather give the user a nicer YCM message about the command not being available for the current filetype than the ugly Vim error of "command doesn't exist" which users have reported as being confusing.

All of these commands have a Ycm prefix, and if I made them buffer-local tomorrow (as some of them were a long time ago), I'd still keep the prefix.

@Valloric
Copy link
Contributor

Oh, and here's another benefit of using a consistent prefix for all of your commands: typing :Ycm<TAB> shows you all of the YCM commands available. This means you don't have to go search through the docs.

The prefix increases usability.

@lilyball
Copy link
Contributor Author

@Valloric You're citing as precedent a plugin (which I've never used, BTW) that appears to provide global behavior. I don't see how that's an appropriate precedent for filetype-specific buffer-local behavior.

I appreciate that you have experience writing Vim plugins, but I just don't think your objections are particularly relevant to this case. It would make a lot of sense if this was a global command, or at least affected more than just one specific brand new filetype (e.g. if it were modifying an existing filetype then it would need to take more care not to step on the toes of anything else related to that filetype).

I would really like to see some sort of valid precedent here for what you're arguing for. I've already shown precedent from Tim Pope, in source distributed with Vim itself, for doing exactly what I'm doing here.

@Valloric
Copy link
Contributor

that appears to provide global behavior.

Some commands only work in some filetypes (and are documented as such). Other functionality is global.

Either way, I've exhausted all of my arguments along with my energy for further discussion. I don't see us convincing each other so we'll agree to disagree and call it a day. :)

@lilyball
Copy link
Contributor Author

@richo I just experimented with adding the output of rustc to the quickfix list, and it doesn't work at all with unnamed buffers.

@lilyball
Copy link
Contributor Author

@Valloric My basic point is that the Rust plugin "owns" the rust filetype and thus gets free reign over the buffer-local rust-filetype-specific stuff it defines. And there's precedent for this as I cited. Your plugin may have filetype-specific stuff, but it doesn't "own" those filetypes, so it needs to be careful about conflicts.

That being said, the one argument you made about typing :Ycm<TAB> is rather compelling. But it still raises a big question: when the user discovers commands that way, how do they know what the expected arguments are? They'll still have to read the documentation for that.

Is there any good solution for this? For telling the user how to use a command without having them go back to the source (which is where they'd learn about the filetype commands under normal circumstances)? If you have a good answer for this, then that may be worth changing the names for.

@Valloric
Copy link
Contributor

My basic point is that the Rust plugin "owns" the rust filetype and thus gets free reign over the buffer-local rust-filetype-specific stuff it defines.

This still ignores the fact that if a global Run command exists your plugin will conflict with it. That no other Run command is declared for the rust filetype matters little.

That being said, the one argument you made about typing :Ycm<TAB> is rather compelling. But it still raises a big question: when the user discovers commands that way, how do they know what the expected arguments are? They'll still have to read the documentation for that.

Step 1 is to discover that command Foo exists. Step 2 is knowing how to use it and what it does. You need to look up docs for that, but now that you know it exists, that's :h Foo and you're there.

:Prefix<TAB> solves the discoverability problem, which is step 1.

Use cases:

  • "I remember installing that Rust Vim plugin, what are all the available commands again?"
  • "I remember the Rust plugin had a command for showing macro guts or whatever. Can't remember what was it called... :Rust<TAB> oh that's right, it's RustExpand."
  • etc.

@chris-morgan
Copy link
Member

With that said, I would love if the compile step could be decoupled, and hooked up to the :make infrastructure to populate the quickfix list with any errors or warnings (I use :make a ton for that reason)

compiler rustc

@chris-morgan
Copy link
Member

I think I would generally prefer :RustRun et al.

@chris-morgan
Copy link
Member

Oh, by the way: see also #7787.

@lilyball
Copy link
Contributor Author

@Valloric @chris-morgan Ok, I renamed the commands and wrote a full documentation file on it. :help :RustRun now works.

@Valloric
Copy link
Contributor

Ok, I renamed the commands and wrote a full documentation file on it. :help :RustRun now works.

Thank you!

Please also make sure that :help RustRun (so no : before RustRun) works as well. That's more important than :help :RustRun working since the command name is RustRun and : is just the command prompt (and the default key binding to invoke the prompt).

@lilyball
Copy link
Contributor Author

@Valloric :help does substring matches. :help RustRun will find the tag for :RustRun.

@lilyball
Copy link
Contributor Author

Added :RustEmitIr and :RustEmitAsm commands as well.

@lilyball lilyball changed the title vim: Add :Run and :Expand commands vim: Add commands :RustRun, :RustExpand, :RustEmitIr, :RustEmitAsm May 30, 2014
@lilyball lilyball changed the title vim: Add commands :RustRun, :RustExpand, :RustEmitIr, :RustEmitAsm vim: Add :RustRun and associated commands May 30, 2014
@lilyball
Copy link
Contributor Author

At this point I'm satisfied with what I have here. r?

@richo
Copy link
Contributor

richo commented May 31, 2014

FWIW, lgtm.


function! s:Run(path, bang, rustc_args, args)
try
let exepath = tempname()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For Windows, I believe this will need to have .exe appended.

lilyball added 4 commits May 30, 2014 18:32
Define a command :Run to compile and run the current file. This supports
unnamed buffers (by writing to a temporary file). See the comment above
the command definition for notes on usage.

Define <D-r> and <D-R> mappings for :Run to make it easier to invoke in
MacVim.

Define a command :Expand to display the --pretty expanded output for the
current file. This can be configured to use different pretty types. See
the comment above the command definition for notes on usage.

Create an autoload file and put function definitions there to speed up
load time.
@lilyball
Copy link
Contributor Author

@chris-morgan Both issues are now fixed

COMMANDS *rust-commands*

:RustRun [args] *:RustRun*
:RustRun! [rustc-args] [--] [args]
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why not always allow a --? I know that all that is said about the bang at the end of an Ex command is that it “makes the command behave in a different way” (:_! help), but this case doesn’t feel right to me; the bang’s existence or non-existence is merely trivial sugar.

I’d just as soon see it unified:

:RustRun [[rustc-args] --] [args]

Which would allow

:RustRun args
:RustRun -- args
:RustRun rustc-args --
:RustRun rustc-args -- args

These are achievable in the current behaviour thus:

:RustRun args
:RustRun args
:RustRun! rustc-args
:RustRun! rustc-args -- args

I think the precedent in bang behaviour is more along the lines of whether it should :update or not.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I suspect that it's more common to want to pass flags to rustc than to the generated binary. I'd rather type :RustRun! -O than :RustRun -O --. To that end, if I were to get rid of the bang, I'd have :RustRun args be equivalent to the current :RustRun! args and require :RustRun -- args to pass args to the binary. But I also thought that was a bit nonintuitive. For someone who knows the :RustRun command is there but hasn't read the docs in detail will likely assume that argument are passed to the binary.

@japaric
Copy link
Member

japaric commented Jun 14, 2014

As for quickfix, I highly recommend using Syntastic

FWIW, Syntastic has recently removed its Rust checker

@lilyball
Copy link
Contributor Author

@japaric Really? That seems weird. Even if it only detects some errors, it's still better to have it at all than not have it.

flip1995 pushed a commit to flip1995/rust that referenced this pull request Apr 3, 2025
…ust-lang#14480)

This extra condition prevents a problem when removing the '}' in:
```rust
  ( // There was an opening bracket after the parenthesis, which has been removed
    // This is a comment
   })
```
Removing the whitespaces, including the linefeed, before the '}', would
put the closing parenthesis at the end of the `// This is a comment`
line, which would make it part of the comment as well. In this case, it
is best to keep the span on the '}' alone.

changelog: none

Note to reviewer: `manual_inspect` and `collapsible_if` were two lints
exhibiting this problem in cooked up test cases, which have been
included as non-regression tests.
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants