-
Notifications
You must be signed in to change notification settings - Fork 17.8k
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
Proposal: Make Go2 better implement the Object-capability model for security. #23157
Comments
The first part of this proposal seems closely related to #6386. I'm not sure I understand the second part, but it seems to just be a compiler optimization. That does not need to be a proposal at all. The proposal process is primarily for user visible changes. |
Hmm, in some cases compiler optimizations enable the user to write visibly different code, and this is such a case. Like how tail-recursion optimization or lightweight-goroutines change user behavior. Whether Go2 decides to adopt the |
The way to refactor with confidence is to ensure your code has good unit test coverage, not insist that the langauge implements a syntax change before you start to write. I checked Wikipedia for what capabilities based security means and it looks like most use cases are in hardware, intel iAXP432, Burroughs B5000, etc, or an operating system concept like the capabilities mask on processes in Linux. I can see why this would be useful, but as this proposal reads today you’re not asking for capabilities on references (I’m guessing this is how it works at a programming language level), instead your asking for much requested of features like immutability and generic types. Given that this proposal cannot be successful unless you get all the things you ask for in the proposal, and bluntly, when you ask for more than one thing, you risk getting none of them. My suggestion would be to abandon this proposal and instead contribute to the discussion for generic types and immutablility. |
I'm not sure that I understand why the word security is used in the proposal. What is your definition of security here? |
For the second proposal you did point out the existing solution:
For the third proposal, there are cases where modifying the stack copy of the struct in a method makes sense, such as for changing parameters before calling another method without effects on the original struct. If you are providing a struct and you don’t want somebody to edit the contents then the private field plus getter/setter methods is still a solution. The empty interface case is interesting because assigning the type assertion to a variable gives you a copy and it can be passed as a reference instead of the full struct. Doesn’t An approach may be to have a wrapper struct pointing at your base struct that has no public fields and only has a Copy() method. This way you pass just the pointer down the stack but the person using it can’t get at the original memory. I still think that running untrusted plugins in a separate process is the right solution. Here's some discussion on how to implement communication: https://stackoverflow.com/questions/9366550/go-inter-process-communication |
This is a great idea. Since Javascript is being written everywhere with the addition of Node.js, why not Go everywhere? We'd need a jQuery syntax. Cancellation is a problem the context package handles, but it requires the context to be explicitly checked during processing. Since you are able to make additions to the arbitrary Go scripts you could have the plugin IPC handler trigger a context cancellation when a cancel is sent and have context checks somehow added to the script. |
One or more dependencies may be malicious. I'd like to defend my process against malicious dependencies. In a secure language ecosystem, if a module or a function argument passed in or a value returned provides the user with the capability to do something, it's because that is the intent. Ability is equivalent to right. Please take a look at the top post, I've updated it with clarifications.
If it works, it's because it's the same var. Once you assign/copy it to another var, mutating the other var doesn't affect my var.
Yes. My point is that the ability to do that is not sufficient to make the whole ecosystem "secure". If everyone did the above for every structure as needed, it wouldn't be a problem. But we don't, because there's no motivation to, because there is no commitment starting from the language spec and implementation. It's a chicken and egg problem, but I'm not in a position to be hacking on Golang right now, so here is my contribution. :)
Please check the top post, I showed why that solution isn't used, and pointed out the example w/ crypto/rand. Examples aren't hard to find, and it's just a matter of time until someone exploits it somewhere.
I'll make appropriate sub-proposals, but for reasons stated this proposal should exist on its own. That is, each proposal in isolation isn't very motivating.
You're right about that. I think the plugin argument isn't needed to make a case for this proposal (since dependencies can always be malicious) so I will remove that from the proposal. |
I'll split this proposal up as suggested, but I'd like to keep this issue as a reference to the whole concept of making Golang more secure against malicious dependencies. |
I think the problem of mutable package level state is better solved by not having package level state rather than finding ways to make package level state immutable.
… On 18 Dec 2017, at 08:24, Jae Kwon ***@***.***> wrote:
I'll split this proposal up as suggested, but I'd like to keep this issue as a reference to the whole concept of making Golang more secure against malicious dependencies.
—
You are receiving this because you commented.
Reply to this email directly, view it on GitHub, or mute the thread.
|
The language could enforce that no module can have exposed variables. That would be an acceptable solution. The proposal here is not to make module level state "immutable". Consts can still be mutable internally, but the pointers that they refer to cannot be swapped out for another pointer. I'd like for Golang to have these mutable consts (they're so convenient), and to disallow exported vars entirely. |
My mistake, I pressed the wrong button. |
@jaekwon : I'm confused as to what your threat model is that you're trying to defend against. I think there are two categories: buggy or otherwise inadvertent mistakes, and nefarious code. I think it is reasonable to provide some help to programmers to avoid the former. I think it is a much taller order to do anything about the latter. Go just isn't very well suited to defending against adversaries that can inject code. A few reasons:
|
The threat model is reality, with software such as go-ethereum, tendermint, or docker with complex external dependencies and not enough time, money, or human-capital to audit everything. The threat model is our financial infrastructure being exploited by hackers for personal gain through whatever means possible.
Lets do the hard work of making it resilient against nefarious code, and make this be a guiding principle for the design of Go2. It doesn't matter how it's done as long as it's done with the spirit of Go1 -- "simple", "explicit", "orthogonal", "concurrent". I'm suggesting we add "secure". It doesn't have to be perfect, but it should work towards it, and make decisions on language changes and classification of bug/feature accordingly. Why not? Don't you want Go to be the Go2 language for everyone? It's awesome! |
Right, if you're willing to just say no unsafe altogether, then fine. But that means no
This is one I don't know of any solution for. Except sandboxing a whole process. Which coincidentally also solves point 1 and 3.
A concrete proposal about how to be more sure of the correctness of the compiler/runtime/stdlib would be welcome. But they are all moving targets, so it would be complicated. Probably not just technically complicated, but change process complicated. In any case, without solving point 2, it is all kinda moot. No point in locking your windows when the door is open. |
I like the whitelist idea. Yes, protobuf seems worthy to audit, so it could be whitelisted too. I imagine most libraries otherwise don't use
Yeah, confirmed that it's a problem on my mac. Hmm... |
Disclosure: I work at Google, but not related to Go team. Kenton Varda and I had a proposal on this topic long before Go was first released. I think the original proposal is still valid to these days, so I share it here for discussion purpose. The proposal (nickname gocap) is to introduce the concept of stateless package. A stateless package cannot have mutable state and can only depend on stateless packages. A stateless package is essentially pure functional: it can modify inputs, generate output, consume cpu and memory, but it cannot create side effect besides modification to inputs. This proposal only needs a tiny language change: a simple declarative marks a package as stateless, and the linker verifies it. However, we found the proposal cannot work because Go doesn't have good constant support. Without constants, core packages, such as If Go native supports constants, something equivalent to C++ Regarding read-only slice et al, I think it is only needed for performance reasons. The caller can always duplicate the slice and not worry about malicious modification by the callee. In Java, this can be achieved by using immutable collections without direct language support. To introduce read-only data structure at language level, we need to be aware that caller and the callee don't really trust each other. For example, a caller can pass a read-only slice to a callee, but the caller can create a go routine to continuously modify the slice. So the callee sees a read-only but mutable by someone else slice, which seems a bad idea. We can solve the mutual trust problem by using Rust-like borrow check plus deep immutability to make sure an immutable reference can not be interfered by a mutable reference. This would make the language and language usage significantly more complicated. It also requires massive rewrite of all core libraries to correctly mark every pointer parameter. I don't think it is worthwhile and it is also against Go's philosophy. Summary: if Go natively supports constants, we can use stateless package to achieve capability-based security with very little effort. A stateless package can only damage the input data structures and consume cpu+memory. |
That doesn't make sense to me. Of the things being proposed, Proposal_0 and Proposal_1 are handled by rules of the language, and is not dependent on machine architecture or platform. Proposal_2 is in a sense already implemented as demonstrated, we just need syntactic support. And these proposals would help make any large codebase secure regardless of the architecture or platform. The data races issue might be solvable by packing things into 64 bits. You'd have to run the code in a 64-bit machine for complete security, but those machines are widely available. See slide 3: https://talks.golang.org/2015/go-gc.pdf |
Yes, we need it for performance. Do you always duplicate slices before passing it to any interface object? Often there's an implicit contract that the callee should not modify something. The proposal makes that implicit contract (which would otherwise have to be written as documentation in english) an explicit contract. It's just making the compiler do more useful work to help let you enforce whatever contract that you're supposed to set. It sounds like a bad idea not to think about that contract. See the
That depends on the context. It might be perfectly ok. It's the job of the code that created the mutable data-structure (or received it from elsewhere) to ensure that it is only accessible for writing in ways such that reading is also safe. Again, see the
We already have that to some degree w/ non-pointer structs (+ private fields on pointer structs). See Proposal_2, Compiler optimizations for immutable struct assignment. Embedded structs, even when the field type is an interface, as long as the structs are non-pointers, are already deeply immutable unless you hold a pointer to it. Proposal_0 extends this to slices. The whole point of Proposal_0 and Proposal_2 is that Rust-like control over mutability is not too difficult to achieve (Proposal_0 for slices; Already works for structs, but Proposal_2 makes it better).
These proposals do solve the "mutual trust" problem, and you can see for yourself how simple it is. The only complication is from the introduction of |
If we agree this is for performance, I recommend we punt it for a couple of
years, and solve the constant issue first.
We can look at the complexity of Rust. I estimate it will take at least 2
years to support immutable values. We have to refactor existing libraries
to use it, which is very unlikely to happen without breaking change.
Building a language feature is easy. Adopting it in a large ecosystem is
thousand times or million times harder. We force work on many developers,
and most of them will just ignore it.
My original proposal is much closer to Go language nature, and has
practical chance of success. It also help Go runtime by putting large data
structure, like Unicode table, into data segment, and eliminate the
initialization code at link time, and avoid GC cost on it.
|
Note that the current implementation of the math package has internal state (and modifies it during init) by figuring out at runtime which cpu capabilities are present (which can not be known at compile time) and choosing optimized function implementations accordingly (which might slightly differ in output within some range of precision too). for example: |
Thanks for pointing out explicitly. I am aware of it. Go linker needs to
whitelist several core packages to be stateless even they use asm code and
keep internal state. Such state cannot be observed by Go applications, so
it is fine.
|
I'm having a hard time understanding the benefit of this proposal.
Now I'm arguing the other side... why not just use a
How do you address the static initialization fiasco? If you have a specific proposal for constexpr-like consts, how does it compare to #23161 ?
The caller could duplicate the slice, but sometimes the caller doesn't want to. Yes, for performance. It's already the case today that developers do not always duplicate slices before passing it onto an unknown object (e.g an Interface, who knows what it'll do) for performance, and they're either doing so with documentation-level contracts (e.g. "CONTRACT: caller must duplicate slice before passing"), or worse, not having a clear contract documented anywhere, because the programmer was too busy breeding cryptokittles. So yes, we need it for performance, but we need it yesterday for both performance and security.
That's a great point. We could implement a custom XYZSlice type without setters, maybe that will work. But who would actually use this thing without language-level support? type ROByteSlice struct {
s []byte
}
func (r ROByteSlice) Get(i int) byte {...}
func (r ROByteSlice) Subslice(s int, e int) ROByteSlice {...}
func (r ROByteSlice) Bytes() []byte {...} // copies
func (r ROByteSlice) Len() int {...} |
The benefit of my proposal is it has minimum complexity to achieve practical goal of running untrusted code. Without deep immutability, the caller has to protect inputs from being destroyed by callees. This is an intentional trade off to make the proposal practical. I am not against solving the mutability issue, such as readonly slice or even deep mutual immutability. I just think it is a very hard problem, and I don't think it is really needed. This issue should be decided by the Go community. I don't have strong take on this. Regarding the static initializer fiasco, it doesn't exist with stateless package. For stateless package, all constants are resolved by compiler at compile time. If compiler can not handle it, it is build error. There is no static initializer in stateless package. PS: I was involved with Java static initialization logic. I understand the complexity, which is why I would eliminate it entirely for stateless package. |
@wora do you mind creating a |
Sure. I will do it after Christmas. Happy holidays!
|
thanks, looking forward to it. happy hols to you too. |
math depends on unsafe: https://godoc.org/math?import-graph If we're going to talk about performance then we need benchmarks based on real systems to prove there's a problem.
Concurrent algorithms can be done in a stateless way from the caller's perspective, this is a big restriction to place on Go library code. We don't know what performance needs there will be in the future and concurrent algorithms as a possible solution is a strength of Go. I don't understand a software development model of using untrusted code directly linked into a security-critical program, are there examples that can be more deeply described? |
I created a dedicated issue #23267 for Gocap. Thanks everyone for your valuable feedback. |
There is no clear proposal here on which to make a decision. And as @randall77 says above, "Go just isn't very well suited to defending against adversaries that can inject code." It's almost certainly a mistake to try to retrofit security into a language that wasn't designed with it as a first principle. |
Please reconsider, @ianlancetaylor . I've studied a few languages and I think Go is very well suited to be a secure language in the way proposed here. All that we need to make code injection safe is to parse it and filter it, e.g. ensure that only certain modules are imported. Even filtering for go-routine calls, while it would be prohibitive for Golang in general, can work very well for many use-cases where logic can be plugged in. We're determined to make the next iteration of this language be safe in the way described here, and would really prefer not to fork the language. Please reconsider, I'm convinced that it's worth discussing. |
Follow this for discussions of a Go fork: https://twitter.com/jaekwon/status/985577190300508160 |
@jaekwon You didn't address the most important point: there is no clear proposal here on which to make a decision. |
@ianlancetaylor This largely depends on whether the Go community wants to provide in-process security and at what development cost. If there is no sufficient desire, then a concrete proposal wouldn't make too much difference. I don't know how to sense the desire on this topic. |
That's circular logic. No one can tell if someone wants to make it happen if there's no proper proposal to evaluate. |
@wora I'm not sure precisely what you mean by in-process security, but see @randall77 's comment above: #23157 (comment) . Certainly in-process security is not going to be a top priority for Go. That is, we are not willing to do it, whatever it is, at any cost. You want to know whether there is "sufficient desire," but it seems impossible to know whether there is sufficient desire without knowing how much it would cost. |
In that case, this is a chicken and egg problem. The best way to solve it is to create a fork for Secure Go, and see what we can achieve. Many years ago, I tried to explore the concept of Gocap. Since Go doesn't support Much like other design challenges, a convincing prototype is a good way to find the right answer. |
UPDATE
We're forking to HOG-lang. https://github.com/HogLang/hog
Original proposal and motivation
This proposal is a combination of three proposals to evolve Golang towards a more secure language that is robust against malicious dependencies.
Design principle: Implement the object-capability model
0. Read-only slices
Slices are used everywhere in Golang, but there's no native language feature to help with safety. It's not always immediately obvious whether it's safe to pass a slice, or whether that slice needs to be copied before being passed in or returned. You need to think about the performance tradeoffs of copying or not, and often that blinds you to the concern of security/mutability.
Read the read-only slices proposal here.
It's an extension to Brad's original proposal, and it introduces
any
andreadonly
modifiers to slices.1. Secure modules w/ the const-mutable type
With most module-level vars today, it's clear that you're not really supposed to update them. Or rather, it wouldn't be secure to allow arbitrary code to update them. For example, see the var in https://golang.org/pkg/crypto/rand/ : Reader could be replaced with something bad by malicious code, and the next user of crypto/rand.Reader wouldn't know.
See the const-mutable proposal.
2. Pointerized structs
The same problem exists for pointers as for slices. One could argue that the coder should keep fields private and just use getters and setters for permissions, but public/private (exposed/unexposed) fields are not really used today in the security context of malicious code. You expose a field because the user might need read access to it and it's convenient to expose rather than writing a getter, or, because the user is expected to write to the field in a non-malicious way.
The proposed solution is to support structs that are always passed by reference.
Read the pointerized struct proposal here.
More discussions and proposals
Several people have brought up other proposals worth mentioning here. I'll continue to compile these proposals and link to them from here.
UPDATE
We're forking to HOG-lang. https://github.com/HogLang/hog
The text was updated successfully, but these errors were encountered: