go-04-map-concurrent
0.000
Challenge · difficulty 4/5
# Concurrent ordered map
Implement **`solution.go`** in `package challenge` exporting:
```go
func MapConcurrent(inputs []int, workers int, fn func(int) int) []int
```
Apply `fn` to every element of `inputs` using a pool of **at most `workers`**
goroutines running concurrently, and return the results.
Rules:
- The returned slice has the **same length** as `inputs`, and `result[i]` is
`fn(inputs[i])` — i.e. results are in the **same order as the inputs**,
regardless of the order goroutines finish in.
- Work must be distributed across concurrent goroutines (use goroutines plus
channels and/or `sync.WaitGroup`). No more than `workers` goroutines may be
processing elements at the same time.
- `workers` may be larger than `len(inputs)`; never start more workers than
there is work for, and never start fewer than one when there is work.
- If `inputs` is empty (or `nil`), return an empty (non-nil is fine, but it must
have length 0) slice without starting any work.
- You may assume `workers >= 1` and that `fn` is safe to call concurrently
(it does not share mutable state).
Examples:
- `MapConcurrent([]int{1, 2, 3}, 2, func(x int) int { return x * x })` → `[]int{1, 4, 9}`
- `MapConcurrent([]int{}, 4, fn)` → `[]int{}`
- `MapConcurrent([]int{5}, 8, func(x int) int { return x + 1 })` → `[]int{6}`
tests/solution_test.go
package challenge
import (
"reflect"
"sync"
"sync/atomic"
"testing"
)
func TestMapConcurrentMatchesSequential(t *testing.T) {
fn := func(x int) int { return x*x + 1 }
cases := []struct {
name string
inputs []int
workers int
}{
{"empty", []int{}, 4},
{"nil", nil, 4},
{"single", []int{7}, 1},
{"single many workers", []int{7}, 16},
{"workers one", []int{1, 2, 3, 4, 5}, 1},
{"workers equal len", []int{1, 2, 3, 4}, 4},
{"workers gt len", []int{1, 2, 3}, 10},
{"larger", []int{9, 8, 7, 6, 5, 4, 3, 2, 1, 0, -1, -2}, 3},
{"negatives", []int{-5, -4, -3, -2, -1}, 2},
}
for _, c := range cases {
t.Run(c.name, func(t *testing.T) {
want := make([]int, len(c.inputs))
for i, v := range c.inputs {
want[i] = fn(v)
}
got := MapConcurrent(c.inputs, c.workers, fn)
if len(got) != len(c.inputs) {
t.Fatalf("len = %d, want %d", len(got), len(c.inputs))
}
if !reflect.DeepEqual(got, want) {
t.Errorf("MapConcurrent(%v, %d) = %v, want %v", c.inputs, c.workers, got, want)
}
})
}
}
func TestMapConcurrentOrderPreserved(t *testing.T) {
// Identity maps input value to output position; any reordering would be caught.
inputs := make([]int, 200)
for i := range inputs {
inputs[i] = i * 3
}
got := MapConcurrent(inputs, 7, func(x int) int { return x })
for i, v := range inputs {
if got[i] != v {
t.Fatalf("index %d = %d, want %d (order not preserved)", i, got[i], v)
}
}
}
func TestMapConcurrentEmptyReturnsLenZero(t *testing.T) {
got := MapConcurrent(nil, 4, func(x int) int { return x })
if len(got) != 0 {
t.Fatalf("len = %d, want 0", len(got))
}
}
func TestMapConcurrentRespectsWorkerLimit(t *testing.T) {
inputs := make([]int, 100)
for i := range inputs {
inputs[i] = i
}
const limit = 4
var mu sync.Mutex
cond := sync.NewCond(&mu)
var active int64
var maxActive int64
var arrived int
// fn blocks until either `limit` callers are concurrently in flight or no more
// callers can arrive. This forces real concurrency while bounding it at `limit`:
// if the implementation ran fewer than `limit` workers it would deadlock-stall,
// and any run above `limit` would be recorded in maxActive.
fn := func(x int) int {
cur := atomic.AddInt64(&active, 1)
for {
m := atomic.LoadInt64(&maxActive)
if cur <= m || atomic.CompareAndSwapInt64(&maxActive, m, cur) {
break
}
}
mu.Lock()
arrived++
if arrived >= limit {
cond.Broadcast()
}
for arrived < limit {
cond.Wait()
}
mu.Unlock()
atomic.AddInt64(&active, -1)
return x * 2
}
got := MapConcurrent(inputs, limit, fn)
if m := atomic.LoadInt64(&maxActive); m > limit {
t.Fatalf("observed %d concurrent workers, limit was %d", m, limit)
}
if atomic.LoadInt64(&maxActive) < 2 {
t.Fatalf("no real concurrency observed (maxActive=%d)", maxActive)
}
for i := range inputs {
if got[i] != inputs[i]*2 {
t.Fatalf("index %d = %d, want %d", i, got[i], inputs[i]*2)
}
}
}
func TestMapConcurrentCallsFnOncePerElement(t *testing.T) {
inputs := []int{10, 20, 30, 40, 50, 60}
var calls int64
got := MapConcurrent(inputs, 3, func(x int) int {
atomic.AddInt64(&calls, 1)
return x + 1
})
if calls != int64(len(inputs)) {
t.Fatalf("fn called %d times, want %d", calls, len(inputs))
}
for i, v := range inputs {
if got[i] != v+1 {
t.Errorf("index %d = %d, want %d", i, got[i], v+1)
}
}
}
System prompt
You are an expert programmer. Solve the task exactly as specified. Output your solution as fenced code blocks using the required file name(s) and the exact function/type signatures requested. Prefer correctness; do not include prose outside code unless asked.
Proposed solution
```go
package challenge
import (
"sync"
)
func MapConcurrent(inputs []int, workers int, fn func(int) int) []int {
if len(inputs) == 0 {
return []int{}
}
// Ensure we don't start more workers than there is work
if workers > len(inputs) {
workers = len(inputs)
}
// Create a result slice
results := make([]int, len(inputs))
// Use a WaitGroup to wait for all workers to finish
var wg sync.WaitGroup
// Create a channel to signal when a worker is done
done := make(chan struct{})
// Process each input element
for i := 0; i < len(inputs); i++ {
wg.Add(1)
go func(index int) {
defer wg.Done()
results[index] = fn(inputs[index])
<-done // Signal that this worker is done
}(i)
}
// Wait for all workers to finish
wg.Wait()
return results
}
```Test output (stdout)
{"Time":"2026-07-02T00:04:44.419012375+02:00","Action":"start","Package":"challenge"}
{"Time":"2026-07-02T00:04:44.42034404+02:00","Action":"run","Package":"challenge","Test":"TestMapConcurrentMatchesSequential"}
{"Time":"2026-07-02T00:04:44.42035458+02:00","Action":"output","Package":"challenge","Test":"TestMapConcurrentMatchesSequential","Output":"=== RUN TestMapConcurrentMatchesSequential\n"}
{"Time":"2026-07-02T00:04:44.4203733+02:00","Action":"run","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/empty"}
{"Time":"2026-07-02T00:04:44.42037508+02:00","Action":"output","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/empty","Output":"=== RUN TestMapConcurrentMatchesSequential/empty\n"}
{"Time":"2026-07-02T00:04:44.42037913+02:00","Action":"output","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/empty","Output":"--- PASS: TestMapConcurrentMatchesSequential/empty (0.00s)\n"}
{"Time":"2026-07-02T00:04:44.42038056+02:00","Action":"pass","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/empty","Elapsed":0}
{"Time":"2026-07-02T00:04:44.42038299+02:00","Action":"run","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/nil"}
{"Time":"2026-07-02T00:04:44.4203839+02:00","Action":"output","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/nil","Output":"=== RUN TestMapConcurrentMatchesSequential/nil\n"}
{"Time":"2026-07-02T00:04:44.42038526+02:00","Action":"output","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/nil","Output":"--- PASS: TestMapConcurrentMatchesSequential/nil (0.00s)\n"}
{"Time":"2026-07-02T00:04:44.42038621+02:00","Action":"pass","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/nil","Elapsed":0}
{"Time":"2026-07-02T00:04:44.4203871+02:00","Action":"run","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/single"}
{"Time":"2026-07-02T00:04:44.42038807+02:00","Action":"output","Package":"challenge","Test":"TestMapConcurrentMatchesSequential/single","Output":"=== RUN TestMapConcurrentMatchesSequential/single\n"}
Errors (stderr)
[TIMEOUT]
Self-repair attempts
[
{
"total": 1,
"answer": "```go\npackage challenge\n\nimport (\n\t\"sync\"\n)\n\nfunc MapConcurrent(inputs []int, workers int, fn func(int) int) []int {\n\tif len(inputs) == 0 {\n\t\treturn []int{}\n\t}\n\n\t// Ensure we don't start more workers than there is work\n\tif workers > len(inputs) {\n\t\tworkers = len(inputs)\n\t}\n\n\t// Create a result slice\n\tresults := make([]int, len(inputs))\n\n\t// Use a WaitGroup to wait for all workers to finish\n\tvar wg sync.WaitGroup\n\n\t// Create a channel to signal when a worker is done\n\tdone := make(chan struct{})\n\n\t// Process each input element\n\tfor i := 0; i < len(inputs); i++ {\n\t\twg.Add(1)\n\n\t\tgo func(index int) {\n\t\t\tdefer wg.Done()\n\t\t\tresults[index] = fn(inputs[index])\n\t\t\t<-done // Signal that this worker is done\n\t\t}(i)\n\t}\n\n\t// Wait for all workers to finish\n\twg.Wait()\n\n\treturn results\n}\n```",
"passed": 0,
"test_error": "{\"Time\":\"2026-07-02T00:04:44.419012375+02:00\",\"Action\":\"start\",\"Package\":\"challenge\"}\n{\"Time\":\"2026-07-02T00:04:44.42034404+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential\"}\n{\"Time\":\"2026-07-02T00:04:44.42035458+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential\\n\"}\n{\"Time\":\"2026-07-02T00:04:44.4203733+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\"}\n{\"Time\":\"2026-07-02T00:04:44.42037508+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential/empty\\n\"}\n{\"Time\":\"2026-07-02T00:04:44.42037913+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\",\"Output\":\"--- PASS: TestMapConcurrentMatchesSequential/empty (0.00s)\\n\"}\n{\"Time\":\"2026-07-02T00:04:44.42038056+02:00\",\"Action\":\"pass\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\",\"Elapsed\":0}\n{\"Time\":\"2026-07-02T00:04:44.42038299+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\"}\n{\"Time\":\"2026-07-02T00:04:44.4203839+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential/nil\\n\"}\n{\"Time\":\"2026-07-02T00:04:44.42038526+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\",\"Output\":\"--- PASS: TestMapConcurrentMatchesSequential/nil (0.00s)\\n\"}\n{\"Time\":\"2026-07-02T00:04:44.42038621+02:00\",\"Action\":\"pass\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\",\"Elapsed\":0}\n{\"Time\":\"2026-07-02T00:04:44.4203871+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/single\"}\n{\"Time\":\"2026-07-02T00:04:44.42038807+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/single\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential/single\\n\"}\n\n\n[TIMEOUT]"
},
{
"total": 1,
"answer": "```go\npackage challenge\n\nimport (\n\t\"sync\"\n)\n\nfunc MapConcurrent(inputs []int, workers int, fn func(int) int) []int {\n\tif len(inputs) == 0 {\n\t\treturn []int{}\n\t}\n\n\t// Ensure we don't start more workers than there is work\n\tif workers > len(inputs) {\n\t\tworkers = len(inputs)\n\t}\n\n\t// Create a result slice\n\tresults := make([]int, len(inputs))\n\n\t// Use a WaitGroup to wait for all workers to finish\n\tvar wg sync.WaitGroup\n\n\t// Create a channel to signal when a worker is done\n\tdone := make(chan struct{})\n\n\t// Process each input element\n\tfor i := 0; i < len(inputs); i++ {\n\t\twg.Add(1)\n\n\t\tgo func(index int) {\n\t\t\tdefer wg.Done()\n\t\t\tresults[index] = fn(inputs[index])\n\t\t\t<-done // Signal that this worker is done\n\t\t}(i)\n\t}\n\n\t// Wait for all workers to finish\n\twg.Wait()\n\n\treturn results\n}\n```",
"passed": 0,
"test_error": "{\"Time\":\"2026-07-02T00:05:46.113418431+02:00\",\"Action\":\"start\",\"Package\":\"challenge\"}\n{\"Time\":\"2026-07-02T00:05:46.114578994+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential\"}\n{\"Time\":\"2026-07-02T00:05:46.114595654+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential\\n\"}\n{\"Time\":\"2026-07-02T00:05:46.114614634+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\"}\n{\"Time\":\"2026-07-02T00:05:46.114616704+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential/empty\\n\"}\n{\"Time\":\"2026-07-02T00:05:46.114620844+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\",\"Output\":\"--- PASS: TestMapConcurrentMatchesSequential/empty (0.00s)\\n\"}\n{\"Time\":\"2026-07-02T00:05:46.114623864+02:00\",\"Action\":\"pass\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/empty\",\"Elapsed\":0}\n{\"Time\":\"2026-07-02T00:05:46.114626724+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\"}\n{\"Time\":\"2026-07-02T00:05:46.114627814+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential/nil\\n\"}\n{\"Time\":\"2026-07-02T00:05:46.114629304+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\",\"Output\":\"--- PASS: TestMapConcurrentMatchesSequential/nil (0.00s)\\n\"}\n{\"Time\":\"2026-07-02T00:05:46.114631024+02:00\",\"Action\":\"pass\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/nil\",\"Elapsed\":0}\n{\"Time\":\"2026-07-02T00:05:46.114633364+02:00\",\"Action\":\"run\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/single\"}\n{\"Time\":\"2026-07-02T00:05:46.114634254+02:00\",\"Action\":\"output\",\"Package\":\"challenge\",\"Test\":\"TestMapConcurrentMatchesSequential/single\",\"Output\":\"=== RUN TestMapConcurrentMatchesSequential/single\\n\"}\n\n\n[TIMEOUT]"
},
{
"total": 1,
"answer": "```go\npackage challenge\n\nimport (\n\t\"sync\"\n)\n\nfunc MapConcurrent(inputs []int, workers int, fn func(int) int) []int {\n\tif len(inputs) == 0 {\n\t\treturn []int{}\n\t}\n\n\t// Ensure we don't start more workers than there is work\n\tif workers > len(inputs) {\n\t\tworkers = len(inputs)\n\t}\n\n\t// Create a result slice\n\tresults := make([]int, len(inputs))\n\n\t// Use a WaitGroup to wait for all workers to finish\n\tvar wg sync.WaitGroup\n\n\t// Create a channel to signal when a worker is done\n\tdone := make(chan struct{})\n\n\t// Process each input element\n\tfor i := 0; i < len(inputs); i++ {\n\t\twg.Add(1)\n\n\t\tgo func(index int) {\n\t\t\tdefer wg.Done()\n\t\t\tresults[index] = fn(inputs[index])\n\t\t\t<-done // Signal that this worker is done\n\t\t}(i)\n\t}\n\n\t// Wait for all workers to finish\n\twg.Wait()\n\n\treturn results\n}\n```",
"passed": 0,
"test_error": ""
}
]