• Stars
    star
    247
  • Rank 164,117 (Top 4 %)
  • Language
    Scala
  • License
    MIT License
  • Created almost 11 years ago
  • Updated 8 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

Acyclic is a Scala compiler plugin to let you prohibit circular dependencies between files

Acyclic

Join the chat

Acyclic is a Scala compiler plugin that allows you to mark files within a build as acyclic, turning circular dependencies between files into compilation errors.

Introduction

Acyclic is a Scala compiler plugin that allows you to mark files within a build as acyclic, turning circular dependencies between files into compilation errors.

For example, the following two files have a circular dependency between them:

package fail.simple

class A {
  val b: B = null
}
package fail.simple

class B {
  val a: A = null
}

In this case it is very obvious that there is a circular dependency, but in larger projects the fact that a circular dependency exists can be difficult to spot. With Acyclic, you can annotate either source file with an acyclic import:

package fail.simple
import acyclic.file

class A {
  val b: B = null
}

And attempting to compile these files together will then result in a compilation error:

error: Unwanted cyclic dependency

src/test/resources/fail/simple/B.scala:4:
  val a1: A = new A
                  ^
symbol: class A
More dependencies at lines 5

src/test/resources/fail/simple/A.scala:6:
  val b: B = null
      ^
symbol: class B

This applies to term-dependencies, type-dependencies, as well as cycles that span more than two files. Circular dependencies between files is something that people often don’t want, but are difficult to avoid as introducing cycles is hard to detect while working or during code review. Acyclic is designed to help you guard against unwanted cycles at compile-time, and tells you exactly where the cycles are when they appear so you can deal with them.

A more realistic example of a cycle that Acyclic may find is this one taken from a cycle in uTest:

[error] Circular dependency between acyclic files:
[info] /Users/haoyi/Dropbox (Personal)/Workspace/utest/shared/main/scala/utest/Formatter.scala:58: acyclic
[info]       val traceStr = r.value match{
[info]                        ^
[info] Other dependencies at lines: 20, 54, 44, 66, 40, 15, 67, 2, 45, 42
[info] /Users/haoyi/Dropbox (Personal)/Workspace/utest/shared/main/scala/utest/framework/Model.scala:76:
[info]           v.runAsync(onComplete, path :+ i, strPath :+ v.value.name, thisError)
[info]           ^
[info] Other dependencies at lines: 120
[info] /Users/haoyi/Dropbox (Personal)/Workspace/utest/shared/main/scala/utest/package.scala:72:
[info]   val TestSuite = framework.TestSuite
[info]       ^
[info] Other dependencies at lines: 73
[info] /Users/haoyi/Dropbox (Personal)/Workspace/utest/shared/main/scala/utest/framework/TestSuite.scala:37:
[info]         log(formatter.formatSingle(path, s))
[info]                       ^
[info] Other dependencies at lines: 41, 33

As you can see, there is a dependency cycle between Formatter.scala, Model.scala, package.scala and TestSuite.scala. package.scala has been explicitly marked acyclic, and so compilation fails with an error. Apart from the line shown, Acyclic also gives other lines in the same file which contain dependencies contributing to this cycle.

Spotting this dependency cycle spanning 4 different files, and knowing exactly which pieces of code are causing it, is something that is virtually impossible to do manually via inspection or code-review. Using Acyclic, there is no chance of accidentally introducing a dependency cycle you don’t want, and even when you do, it shows you exactly what’s causing the cycle that you need to fix to make it go away.

Package Cycles

Acyclic also allows you to annotate entire packages as acyclic by placing a import acyclic.pkg inside the package object. Consider the following set of files:

// c/C1.scala
package fail.halfpackagecycle.c

object C1
// c/C2.scala
package fail.halfpackagecycle
package c

class C2 {
  lazy val b = new B
}
// c/package.scala
package fail.halfpackagecycle

package object c {
  import acyclic.pkg
}
// A.scala
package fail.halfpackagecycle

class A {
  val thing = c.C1
}
// B.scala
package fail.halfpackagecycle

class B extends A

These 5 files do not have any file-level cycles, and form a nice linear dependency chain:

c/C2.scala -> B.scala -> A.scala -> c/C1.scala

However, we may want to preserve the invariant that the package c does not have any cyclic dependencies with other packages or files.. By annotating the package with import acyclic.pkg in its package objects as shown above, we can make this circular package dependency error out:

error: Unwanted cyclic dependency

src/test/resources/fail/halfpackagecycle/B.scala:3:
class B extends A
        ^
symbol: constructor A

src/test/resources/fail/halfpackagecycle/A.scala:4:
  val thing = c.C1
      ^
symbol: object C1

package fail.halfpackagecycle.c
src/test/resources/fail/halfpackagecycle/c/C2.scala:5:
  lazy val b = new B
           ^
symbol: class B

Since, c as a whole must be acyclic, the dependency cycle between c, B.scala and A.scala is prohibited, and Acyclic errors out. As you can see, it tells you exactly where the dependencies are in the source files, giving you an opportunity to find and remove them. Here’s a realistic example from Scala.Rx:

[error] Unwanted cyclic dependency
[info]
[info] /Users/haoyi/Dropbox (Personal)/Workspace/scala.rx/shared/main/scala/rx/core/Dynamic.scala:10:
[info] import rx.ops.Spinlock
[info]        ^
[info] symbol: value <import>
[info] More dependencies at lines 29 60 33 41 27 23
[info]
[info] package rx.ops
[info] /Users/haoyi/Dropbox (Personal)/Workspace/scala.rx/shared/main/scala/rx/ops/Async.scala:78:
[info]       super.ping(incoming)
[info]             ^
[info] symbol: method ping
[info] More dependencies at lines 69 101 97 95 67

As you can see, Dynamic.scala in rx.core was accidentally depending on Spinlock in rx.ops. That cross-module dependency from rx.core to rx.ops should not exist, and the proper solution was to move Spinlock over to rx.core. Without Acyclic, this circular dependency would likely have gone un-noticed.

How to Use

Mill

For Mill, use the following:

def compileIvyDeps = Agg(ivy"com.lihaoyi:::acyclic:0.3.8")
def scalacPluginIvyDeps = Agg(ivy"com.lihaoyi:::acyclic:0.3.8")

sbt

To use, add the following to your build.sbt:

libraryDependencies += ("com.lihaoyi" %% "acyclic" % "0.3.8" cross (CrossVersion.full)) % "provided"

autoCompilerPlugins := true

addCompilerPlugin("com.lihaoyi" %% "acyclic" % "0.3.8" cross (CrossVersion.full))

Force

If you want to enforce acyclicity across all your files, you can pass in the command-line compiler flag:

-P:acyclic:force

Or via SBT:

scalacOptions += "-P:acyclic:force"

This will make the acyclic plugin complain if any file in your project is involved in an import cycle, without needing to annotate everything with import acyclic.file. If you want to white-list a small number of files whose cycles you’ve decided are OK, you can use

import acyclic.skipped

to tell the acyclic plugin to ignore them.

Warnings instead of errors

If you want the plugin to only emit warnings instead of errors, add warn to the plugin’s flags.

scalacOptions += "-P:acyclic:warn"

Who uses it?

Acyclic is currently being used in uTest, Scalatags and Scala.Rx, and helped remove many cycle between files which had no good reason for being cyclic. It is also being used to verify the acyclicity of its own code. It works with Scala 2.11, 2.12 and 2.13.

If you’re using incremental compilation, you may need to do a clean compile for Acyclic to find all unwanted cycles in the compilation run.

Limitations

Acyclic has problems in a number of cases:

  • If you use curly-braced package XXX {} acyclic inside your source files, it does the wrong thing. Acyclic assumes all packages are listed in a sequence of statements at the top of each file

  • Under incremental compilation, Acyclic does not always find all possible cycles, since one cycles within the files currently getting compiled will get caught. A solution is to do a clean build every once in a while.

License

Acyclic is published under the MIT License:

The MIT License (MIT)

Copyright (c) 2014 Li Haoyi

Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions:

The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software.

THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.

ChangeLog

0.3.8

  • Added support for Scala 2.13.11

0.3.7

  • Added support for Scala 2.12.18

0.3.6

  • Added support for Scala 2.13.10

0.3.5

  • Added support for Scala 2.13.9

0.3.4

  • Added support for Scala 2.12.17

0.3.3

  • Added support for Scala 2.12.16

0.3.2

  • Added plugin option warn to emit compiler warnings instead of errors

0.3.1

  • Support for Scala 2.13.8

0.3.0

  • Cross-build across all scala point versions

0.2.0

  • Support for Scala 2.13.0 final

0.1.7

  • Fix import acyclic.skipped, which was broken in 0.1.6

0.1.6

  • You can now use the scalac option -P:acyclic:force (scalaOptions += "-P:acyclic:force" in SBT) to enforce acyclicity across your entire codebase.

0.1.5

  • Scala 2.12.x support

0.1.4

  • Loosen restrictions on compiler plugin placement, to allow better interactions with other plugins. Also, acyclic.file is now @compileTimeOnly to provide better errors

0.1.3

  • Ignore, but don’t crash, on Java sources

More Repositories

1

Ammonite

Scala Scripting
Scala
2,601
star
2

mill

Mill is a fast JVM build tool that supports Java and Scala. Mill aims to make your project’s build process performant, maintainable, and flexible
Scala
2,027
star
3

fastparse

Writing Fast Parsers Fast in Scala
Scala
1,091
star
4

scalatags

ScalaTags is a small XML/HTML construction library for Scala.
Scala
731
star
5

upickle

uPickle: a simple, fast, dependency-free JSON & Binary (MessagePack) serialization library for Scala
Scala
711
star
6

requests-scala

A Scala port of the popular Python Requests HTTP client: flexible, intuitive, and straightforward to use.
Scala
694
star
7

os-lib

OS-Lib is a simple, flexible, high-performance Scala interface to common OS filesystem and subprocess APIs
Scala
690
star
8

cask

Cask: a Scala HTTP micro-framework
Scala
525
star
9

sourcecode

Scala library providing "source" metadata to your program, similar to Python's __name__, C++'s __LINE__ or Ruby's __FILE__.
Scala
521
star
10

utest

A simple testing framework for Scala
Scala
487
star
11

fansi

Scala/Scala.js library for manipulating Fancy Ansi colored strings
Scala
224
star
12

PPrint

Pretty-printing value, types and type-signatures in Scala
Scala
217
star
13

scalasql

Scala ORM to query SQL databases from Scala via concise, type-safe, and familiar case classes and collection operations. Connects to Postgres, MySql, H2, and Sqlite out of the box
Scala
180
star
14

mainargs

A small, convenient, dependency-free library for command-line argument parsing in Scala
Scala
173
star
15

castor

Castor is a lightweight, typed Actor library for Scala and Scala.js
Scala
127
star
16

geny

Provides the geny.Generator data type, the dual to a scala.Iterator that can ensure resource cleanup
Scala
88
star
17

unroll

Scala
18
star
18

mill-scala-hello.g8

Giter8 template for a Mill project with a Scala Hello App
Shell
10
star
19

mill-moduledefs

Scalac compiler plugin to support Mill modules
Scala
6
star