Implement Java protected access modifier using Scala

I have an abstract library package ‘baseplugin’. It includes an abstract BaseClient class that has abstract connect() method that must not be public.

//baseplugin/BaseClient.scala
package baseplugin

class BaseClient {
  protected def connect(): Unit
}

I also have a implementation package, myplugin that is outside the baseplugin package hierarchy. It includes MyClient that extends BaseClient and implements its connect() method. Again, the connect() method must not be public

//myplugin/MyClient.scala
package myplugin
import baseplugin.BaseClient

class MyClient extends BaseClient {
  override protected def connect(): Unit = {
    // implementation logic here.
  }
}

I want to add a strategy class in both packages that connects to NE is different ways.

// baseplugin/ReconnectStrategy.scala
package baseplugin

class BaseReconnectStrategy(client: BaseClient) {
  def reconnect() {
    client.connect()
    // additional base logic here.
  }
}

//myplugin/CustomReconnectStrategy.scala
package myplugin

class CustomReconnectStrategy(client: MyClient) {
  def reconnect() {
    // custom validation here.
    client.connect()
    // custom logic here.
  }
}

However the protected access modifier is not allowing me to access client.connect() in BaseReconnectStrategy class.

If I set it to protected[baseplugin] or private[baseplugin] then the Scala compiler is asking me to add “override protected[baseplugin]” to the overridden connect() method in MyClient.

weaker access privileges in overriding
protected[package baseplugin] def connect(): Unit (defined in class BaseClient)
  override should at least be protected[baseplugin]
  override protected def connect(): Unit = {

If I add it to the overriding method then the compiler throws an exception that “baseplugin” is not in scope:

baseplugin is not an enclosing class
  override protected[baseplugin ] def connect(): Unit = {

What is the scala idomatic way to implement Java protected access modifier when I want both of the following features:

  1. Access the BaseClient.connect() method within same package in BaseReconnectStrategy class
  2. Access the BaseClient.connect() within the subclass MyClient that is in a package outside the baseplugin hierarchy.

I’m using scala-2.13.10 in my package and unfortunately I cannot upgrade it. Can anyone please share your inputs on how I can achieve above requirement?

1 Like

Not sure if there is a better way, but something you can do is make the method public, but make it require an implicit evidence ConnectCapability and then you hide the possibility of creating such evidence.

Something like this: Scastie - An interactive playground for Scala.

I may be simply misreading, but:

Why do you expect to be able to call .reconnect when it is not defined in either client class?

Is there a typo in your reduced example, or is this a misdiagnosis?

@sageserpent-open Yeah. It was a typo. My question is based on my company’s codebase. I’m not allowed to share it publicly. So, I included a sample code that outlines my current company code. While doing this, I ended up including a typo.

Thanks for bringing it to my attention. I’ve corrected it and updated my original question now.

The imp of the perverse wants me to mention that you still have this:

I’m a joy to work with as a PR reviewer. :sweat_smile:

Nitpicking aside, I think you are falling victim to being too worried about restricting access. I would start with having everything public, cooking the design a bit and then tightening up the access.

Nevertheless, given you are relying on package access to BaseClient.connect from BaseReconnectStrategy, why don’t you capture that channel of access and delegate up from the constructor of CustomReconnectStrategyCustomReconnectStrategy.reconnect with a supercall (to BaseReconnectStrategy.reconnect)? If that makes you feel uncomfortable, just mumble “DRY”, "Separation of concerns” and “Clocking off on time with a green build” to yourself a few times and it should be OK. The name BaseReconnectStrategy does suggest that sub classing is intended here…

PS: trivia note on consistency: Wikipedia:Emerson and Wilde on consistency - Wikipedia.

So my imp of the perverse is really a hobgoblin. I hope that is a saving grace.

PPS: edited the post again to correct my own typo. That will teach me. :grin: