def sample_counterfree(alphabet_size, num_operations, p_union, p_complement, p_concat, generator):
    alphabet = set([Sym(str(i)) for i in range(alphabet_size)])
    
    # automata that recognize exactly single symbol 
    def unary_automaton(symbol):
        fsa = FSA()
        fsa.set_I(0)
        fsa.set_F(1)
        fsa.set_arc(0, symbol, 1)
        for elem in alphabet:
            fsa.set_arc(1, elem, 2)
        return fsa
    
    # automaton that recognizes only empty string
    empty_fsa = FSA()
    empty_fsa.set_I(0)
    empty_fsa.set_F(1)
    for elem in alphabet:
        empty_fsa.set_arc(0, elem, 1)
    
    # set of automata after iteratively applying union, complement, concat
    closure = set()
    for symbol in alphabet:
        closure.add(unary_automaton(symbol))
    closure.add(empty_fsa)

    fsa = empty_fsa

    # iteratively, randomly apply union, complement, concat
    for _ in range(num_operations):
        if generator.random() <= p_union:
            r = random.choice(closure)
            fsa = fsa.union(r)
            closure.add(fsa)

        if generator.random() <= p_complement:
            fsa = fsa.complement()
            closure.add(fsa)

        if generator.random() <= p_concat:
            r = random.choice(closure)
            fsa = fsa.concatenate(r)
            closure.add(fsa)    
    
    # translate rayuela fsa to nnch fsa
    M = FiniteAutomatonContainer.from_rayuela(fsa)
    
    return M
