Convert Figma logo to code with AI

segmentio logogolines

A golang formatter that fixes long lines

1,008
65
1,008
67

Top Related Projects

3,495

A stricter gofmt

Fast linters runner for Go

Staticcheck - The advanced Go linter

Quick Overview

The golines project is a Go code formatter that automatically wraps long lines of code to a specified maximum line length. It is designed to improve code readability and maintainability by ensuring consistent formatting across a codebase.

Pros

  • Automatic Formatting: golines automatically formats code to a specified maximum line length, reducing the need for manual formatting.
  • Configurable: The tool allows users to customize the maximum line length and other formatting options to suit their preferences.
  • Integrates with Go Tools: golines can be integrated with other Go tools, such as goimports and gofmt, to provide a seamless code formatting experience.
  • Actively Maintained: The project is actively maintained, with regular updates and bug fixes.

Cons

  • Limited Scope: golines is focused solely on line wrapping and does not provide other code formatting features, such as indentation or variable naming conventions.
  • Potential Conflicts with Other Formatters: If used in conjunction with other code formatters, golines may produce unexpected results or conflicts.
  • Potential Performance Impact: Depending on the size of the codebase, the formatting process performed by golines may have a noticeable performance impact.
  • Lack of IDE Integration: golines does not currently provide native integration with popular IDEs, which may limit its adoption for some developers.

Code Examples

Here are a few examples of how to use golines in your Go code:

  1. Formatting a Single File:
package main

import (
    "fmt"
    "strings"
)

func main() {
    longString := "This is a very long string that needs to be wrapped to improve readability and maintainability."
    wrappedString := wrapString(longString, 40)
    fmt.Println(wrappedString)
}

func wrapString(str string, maxLineLength int) string {
    words := strings.Split(str, " ")
    var lines []string
    currentLine := ""

    for _, word := range words {
        if len(currentLine)+len(word) > maxLineLength {
            lines = append(lines, currentLine)
            currentLine = ""
        }
        currentLine += word + " "
    }

    if currentLine != "" {
        lines = append(lines, currentLine)
    }

    return strings.Join(lines, "\n")
}
  1. Formatting an Entire Directory:
package main

import (
    "fmt"
    "os"
    "path/filepath"

    "github.com/segmentio/golines"
)

func main() {
    err := filepath.Walk(".", func(path string, info os.FileInfo, err error) error {
        if err != nil {
            return err
        }

        if !info.IsDir() && filepath.Ext(path) == ".go" {
            err = golines.FormatFile(path, &golines.Options{
                MaxLineLength: 80,
            })
            if err != nil {
                return err
            }
            fmt.Printf("Formatted file: %s\n", path)
        }
        return nil
    })
    if err != nil {
        fmt.Println("Error:", err)
    }
}
  1. Integrating golines with goimports:
package main

import (
    "fmt"
    "os"

    "github.com/segmentio/golines"
    "golang.org/x/tools/imports"
)

func main() {
    file, err := os.ReadFile("example.go")
    if err != nil {
        fmt.Println("Error:", err)
        return
    }

    formattedFile, err := golines.Format(file, &golines.Options{
        MaxLineLength: 80,
    })
    if err != nil {
        fmt.Println("Error:", err)
        return
    }

    importedFile, err := imports.Process("example.go", formattedFile, nil)
    if err != nil {
        fmt.Println("Error:", err)

Competitor Comparisons

3,495

A stricter gofmt

Pros of gofumpt

  • More comprehensive formatting, addressing additional style issues beyond line length
  • Faster execution due to its focus on specific formatting rules
  • Integrated with popular Go tools like goimports and gopls

Cons of gofumpt

  • Less flexible in terms of customization options
  • Doesn't specifically target long lines, which may still require manual intervention

Code Comparison

gofumpt:

func example() {
    if x {
        // gofumpt enforces consistent spacing
        doSomething()
    }
}

golines:

func example() {
    if x {
        // golines focuses on breaking long lines
        doSomething(withLongArgument1,
            withLongArgument2)
    }
}

Summary

gofumpt is a stricter, more opinionated formatter that aims to enforce consistent style across Go projects. It excels in maintaining a uniform codebase but may be less flexible for projects with specific formatting needs. golines, on the other hand, specializes in handling long lines, offering more customization options but with a narrower focus. The choice between the two depends on the project's specific requirements and the development team's preferences for code style and formatting.

Fast linters runner for Go

Pros of golangci-lint

  • Comprehensive linting with support for multiple linters
  • Highly configurable with extensive options
  • Integrates well with CI/CD pipelines

Cons of golangci-lint

  • Can be complex to set up and configure
  • May produce a large number of warnings, requiring careful filtering

Code comparison

golangci-lint configuration example:

linters:
  enable:
    - gofmt
    - golint
    - govet
  disable:
    - errcheck

golines usage example:

golines -w -m 80 ./...

Key differences

  • golines focuses specifically on line length formatting
  • golangci-lint provides a broader range of linting and static analysis tools
  • golines is simpler to use for its specific purpose
  • golangci-lint offers more comprehensive code quality checks

Use cases

  • Use golines when you need to enforce consistent line lengths in your Go code
  • Choose golangci-lint for a more comprehensive code quality and style enforcement tool

Community and maintenance

  • Both projects are actively maintained and have strong community support
  • golangci-lint has a larger user base and more frequent updates due to its broader scope

Staticcheck - The advanced Go linter

Pros of go-tools

  • Offers a comprehensive suite of analysis tools for Go code, including staticcheck, gosimple, and unused
  • Provides deeper static analysis capabilities, detecting a wider range of issues and potential bugs
  • Integrates well with various IDEs and code editors, enhancing the development workflow

Cons of go-tools

  • Primarily focused on code analysis rather than formatting, unlike golines which specializes in line-breaking
  • May have a steeper learning curve due to its broader feature set and multiple tools
  • Can be more resource-intensive when running full analysis on large codebases

Code Comparison

go-tools (staticcheck example):

func example() {
    var x int
    x = 5
    fmt.Println(x)
}

golines:

func example() {
    var x int
    x = 5
    fmt.Println(
        x,
    )
}

In this comparison, go-tools would focus on detecting unused variables or other potential issues, while golines would primarily handle formatting and line-breaking for improved readability.

Convert Figma logo designs to code with AI

Visual Copilot

Introducing Visual Copilot: A new AI model to turn Figma designs to high quality code using your components.

Try Visual Copilot

README

golines test Go Report Card GoDoc

golines

Golines is a Go code formatter that shortens long lines, in addition to all of the formatting fixes done by gofmt.

Motivation

The standard Go formatting tools (gofmt, goimports, etc.) are great, but deliberately don't shorten long lines; instead, this is an activity left to developers.

While there are different tastes when it comes to line lengths in go, we've generally found that very long lines are more difficult to read than their shortened alternatives. As an example:

myMap := map[string]string{"first key": "first value", "second key": "second value", "third key": "third value", "fourth key": "fourth value", "fifth key": "fifth value"}

vs.

myMap := map[string]string{
	"first key": "first value",
	"second key": "second value",
	"third key": "third value",
	"fourth key": "fourth value",
	"fifth key": "fifth value",
}

We built golines to give Go developers the option to automatically shorten long lines, like the one above, according to their preferences.

More background and technical details are available in this blog post.

Examples

See this before and after view of a file with very long lines. More example pairs can be found in the _fixtures directory.

Version support

Newer releases of golines require at least Go 1.18 due to generics-related dependencies. However, the minimum version in go.mod should be considered the minimum required version of Go for any given version of golines. If you need to use golines with an older version of go, install the tool from the v0.9.0 release.

Usage

First, install the tool. If you're using Go 1.21 or newer, run:

go install github.com/segmentio/golines@latest

Otherwise, for older Go versions, run:

go install github.com/segmentio/golines@v0.9.0

Then, run:

golines [paths to format]

The paths can be either directories or individual files. If no paths are provided, then input is taken from stdin (as with gofmt).

By default, the results are printed to stdout. To overwrite the existing files in place, use the -w flag.

Options

Some other options are described in the sections below. Run golines --help to see all available flags and settings.

Line length settings

By default, the tool tries to shorten lines that are longer than 100 columns and assumes that 1 tab = 4 columns. The latter can be changed via the -m and -t flags respectively.

Dry-run mode

Running the tool with the --dry-run flag will show pretty, git-style diffs.

Comment shortening

Shortening long comment lines is harder than shortening code because comments can have arbitrary structure and format. golines includes some basic logic for shortening single-line (i.e., //-prefixed) comments, but this is turned off by default since the quality isn't great. To enable this feature anyway, run with the --shorten-comments flag.

Custom formatters

By default, the tool will use goimports as the base formatter (if found), otherwise it will revert to gofmt. An explicit formatter can be set via the --base-formatter flag; the command provided here should accept its input via stdin and write its output to stdout.

Generated files

By default, the tool will not format any files that look like they're generated. If you want to reformat these too, run with the flag --ignore-generated=false.

Chained method splitting

There are several possible ways to split lines that are part of method chains. The original approach taken by golines was to split on the args, e.g.:

myObj.Method(
	arg1,
	arg2,
	arg3,
).AnotherMethod(
	arg1,
	arg2,
).AThirdMethod(
	arg1,
	arg2,
)

Starting in version 0.3.0, the tool now splits on the dots by default, e.g.:

myObj.Method(arg1, arg2, arg3).
	AnotherMethod(arg1, arg2).
	AThirdMethod(arg1, arg2)

The original behavior can be used by running the tool with the --no-chain-split-dots flag.

Struct tag reformatting

In addition to shortening long lines, the tool also aligns struct tag keys; see the associated before and after examples in the _fixtures directory. To turn this behavior off, run with --no-reformat-tags.

Developer Tooling Integration

vim-go

Add the following lines to your vimrc, substituting 128 with your preferred line length:

let g:go_fmt_command = "golines"
let g:go_fmt_options = {
    \ 'golines': '-m 128',
    \ }

Visual Studio Code

  1. Install the Run on Save extension
  2. Go into the VSCode settings menu, scroll down to the section for the "Run on Save" extension, click the "Edit in settings.json" link
  3. Set the emeraldwalk.runonsave key as follows (adding other flags to the golines command as desired):
    "emeraldwalk.runonsave": {
        "commands": [
            {
                "match": "\\.go$",
                "cmd": "golines ${file} -w"
            }
        ]
    }
  1. Save the settings and restart VSCode

Goland

  1. Go into the Goland settings and click "Tools" -> "File Watchers" then click the plus to create a new file watcher
  2. Set the following properties:
    • Name: golines
    • File type: Go files
    • Scope: Project Files
    • Program: golines
    • Arguments: $FilePath$ -w
    • Output paths to refresh: $FilePath$
  3. In the "Advanced Options" section uncheck the Auto-save edited files to trigger the watcher setting
  4. Confirm by clicking OK
  5. Activate your newly created file watcher in the Goland settings under "Tools" -> "Actions on save"

Others

Coming soon.

How It Works

For each input source file, golines runs through the following process:

  1. Read the file, break it into lines
  2. Add a specially-formatted annotation (comment) to each line that's longer than the configured maximum
  3. Use Dave Brophy's excellent decorated syntax tree library to parse the code plus added annotations
  4. Do a depth-first traversal of the resulting tree, looking for nodes that have an annotation on them
  5. If a node is part of a line that's too long, shorten it by altering the newlines around the node and/or its children
  6. Repeat steps 2-5 until no more shortening can be done
  7. Run the base formatter (e.g., gofmt) over the results, write these to either stdout or the source file

See this blog post for more technical details.

Limitations

The tool has been tested on a variety of inputs, but it's not perfect. Among other examples, the handling of long lines in comments could be improved. If you see anything particularly egregious, please report via an issue.