Escribir código para tres tareas


8

Cuando ejecuta su programa primero, debe imprimir una secuencia de enteros. La segunda ejecución debería imprimir la canción 'Mary tenía un corderito'. La tercera ejecución debería producir un archivo de imagen ASCII.

Puntuación

Su puntaje será la suma de sus bytes de código y cualquier contenido de archivo al comienzo de cada una de las 3 ejecuciones. La puntuación más baja gana.

Por ejemplo, myprogtiene 1500 bytes y no cambia su tamaño, pero escribe un archivo de 4 bytes en cada ejecución para que sepa qué hacer en la próxima ejecución. La puntuación sería 1500 + (1500 + 4) + (1500 + 4) = 4508.

Salida

La primera salida es esta secuencia de 133 enteros, separados por espacios individuales:

0 89 98 179 188 197 269 278 287 296 359 368 377 386 395 449 458 467 476 485
494 539 548 557 566 575 584 593 629 638 647 656 665 674 683 692 719 728
737 746 755 764 773 782 791 809 818 827 836 845 854 863 872 881 890 908
917 926 935 944 953 962 971 980 1079 1088 1097 1169 1178 1187 1196 1259
1268 1277 1286 1295 1349 1358 1367 1376 1385 1394 1439 1448 1457 1466 1475
1484 1493 1529 1538 1547 1556 1565 1574 1583 1592 1619 1628 1637 1646 1655
1664 1673 1682 1691 1709 1718 1727 1736 1745 1754 1763 1772 1781 1790 1808
1817 1826 1835 1844 1853 1862 1871 1880 1907 1916 1925 1934 1943 1952 1961
1970

El segundo resultado es este texto exactamente (incluyendo mayúsculas, puntuación y espacios en blanco; la puntuación inconsistente es intencional):

Mary had a little lamb,
Little lamb, little lamb,
Mary had a little lamb,
Its fleece was white as snow

And everywhere that Mary went,
Mary went, Mary went,
Everywhere that Mary went
The lamb was sure to go

It followed her to school one day
School one day, school one day
It followed her to school one day
Which was against the rules.

It made the children laugh and play,
Laugh and play, laugh and play,
It made the children laugh and play
To see a lamb at school

El tercer resultado es esta imagen en formato PPM ASCII (P3) que queda en un archivo llamado blue.ppm. Esta es la imagen en formato PNG:

imagen objetivo

El archivo PPM debería verse así:

P3 50 50 255
255 255 255 255 255 255 255 255 255 255 255 255 ...
...
255 255 255

Hay una copia del archivo PPM completo aquí .

Ejemplo ejecutado (en Linux)

# ls
myprog
# interpreter myprog
0 89 98 179 188 197 269 278 287 296 359 ...
# interpreter myprog
Mary had a little lamb,
Little ...
# interpreter myprog
# ls
blue.ppm
myprog
# cat blue.ppm
P3 50 50 255
255 255 255 255 255 ...

Reglas

  • Su código puede modificarse a sí mismo.
  • Ejecutar el programa por cuarta vez es un comportamiento indefinido.
  • Puede escribir en un archivo en el directorio actual, pero el tamaño de este archivo se agregará a su puntaje al comienzo de cada ejecución. El archivo PPM final está excluido de esto.
  • Las dos primeras salidas son stdout (o similar). La última salida se escribe en un archivo, sin nada enviado a stdout.
  • Las dos primeras salidas pueden tener opcionalmente un carácter de espacio en blanco final (por ejemplo, espacio o avance de línea).
  • No puede usar la compresión o las bibliotecas de imágenes. Puede usar sus propias funciones de compresión o imagen, pero este código debe incluirse en su programa.
  • Las lagunas estándar están prohibidas.

1
Gran problema! Aprendí algunas cosas nuevas mientras lo resolvía.
Anmol Singh Jaggi

¿El archivo PPM necesita tener 256 pasos de color o debería ser capaz de generar la misma imagen?
Oberon

¿El archivo de salida de la imagen tiene que ser exactamente como el mencionado en la pregunta o puede haber diferencias en los espacios en blanco? Esto es vital para mi respuesta, ya que he tenido especial cuidado de que las nuevas líneas se inserten en el lugar correcto cada vez.
Anmol Singh Jaggi

@Oberon y Anmol Singh Jaggi, el PPM debe estar en formato P3 y mostrar exactamente la misma imagen que el PNG cuando se ve. La profundidad de bits, el espacio en blanco interno depende de usted, pero debe ser un archivo válido.
Logic Knight

Respuestas:


9

Python (897 + 814 + 386 = 2097)

Respuesta simple auto modificable. Tendré que refinar la segunda carrera más tarde.

def p(x):print x,
map(p,[0]+filter(lambda x:sum(map(int,`x`))==17,range(1971)))
#l="ittle lamb,";M="Mary ";w=M+"went";e="verywhere that "+w;s="chool one day";I="It followed her to s"+s+"\n";j="augh and play";i="It made the children l"+j;print M+"had a l"+l+"\nL"+l+" l"+l+"\n"+M+"had a l"+l+"\nIts fleece was white as snow\n\n"+"And e"+e+",\n"+w+", "+w+",\nE"+e+"\nThe lamb was sure to go\n\n"+I+"S"+s+", s"+s+"\n"+I+"Which was against the rules.\n\n"+i+",\nL"+j+", l"+j+",\n"+i+"\nTo see a lamb at school"
#print>>file("blue.ppm","w"),"P3 50 50 255"+' '.join(" %i"%(255-255*any(a<y<b and x==c or y==d and e<x<f for a,b,c,d,e,f in[(2,48,2,47,2,37),(9,48,36,9,27,37),(9,41,28,40,9,29),(0,41,10,1,9,49),(0,37,48,36,14,49),(2,37,15,3,2,16)]))*2+" 255"for x in range(50)for y in range(50))
import sys;x=file(sys.argv[0],"r+");y='#'.join(x.read().split('#')[1:]);x.seek(0);x.truncate();x.write(y)

Además, si se ejecuta por cuarta vez, se degrada con seguridad a un NOP:

'.join(x.read().split('#')[1:]);x.seek(0);x.truncate();x.write(y)

Eso va a ser difícil de superar. Me preguntaba cuándo aparecería un código furtivo de auto-modificación ;-)
Logic Knight

11

Rubí, 1356 1268 * 3 = 3804 bytes

if File.file?(?b)
s='P3
50 50
255
'
d=e=0
[153,44,6,1,42,1,6,1,42,1,6,1,42,1,6,1,42,1,6,1,42,1,6,1,42,1,4,40,5,1,4,1,1,1,36,1,5,1,4,1,1,1,36,1,5,1,4,1,1,1,36,1,5,1,4,1,1,1,36,1,5,1,4,1,1,34,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,34,1,3,1,5,1,4,1,8,31,5,1,4,1,8,1,25,1,9,1,4,1,8,1,25,1,9,1,4,1,8,1,25,1,9,1,4,1,8,1,25,1,9,1,4,1,8,1,25,1,9,1,4,1,8,1,25,1,9,1,4,1,8,1,25,1,9,1,4,1,8,37,4,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,1,34,1,14,36,63].map{|n|n.times{s+=[" 255 255 255"," 0 0 255"][d];s+=$/if (e+=1)%50==0};d=1-d}
File.write 'blue.ppm',s
elsif File.file?(?a)
puts [["","Mary had a ","little lamb",?,,?,,"Its fleece was white as snow"],["And ","everywhere that ","Mary went",?,,'',"The lamb was sure to go"],["","It followed her to ","school one day",'','',"Which was against the rules."],["","It made the children ","laugh and play",?,,'',"To see a lamb at school"]].map{|a,b,c,d,e,f|[a+b+c+d,c.capitalize+', '+c+d,b.capitalize+c+e,f]*$/}*"

"
`touch b`
else
1971.times{|i|$><<"#{i} "if i.to_s.chars.map(&:to_i).reduce(:+)%17<1}
`touch a`
end

Para realizar un seguimiento de la invocación, estoy creando un archivo vacío aen la primera ejecución y un archivo vacío ben la segunda ejecución.

  • El primer programa usa cierta propiedad de los números.
  • El segundo programa usa un montón de partes reutilizables de los versos, pero actualmente solo ahorra 100 bytes sobre la impresión ingenua de todo.
  • El tercer programa actualmente usa solo una simple codificación de longitud de ejecución. Más adelante investigaré la codificación de la ruta de la línea.

1
Buena solución, especialmente los números (que es más un problema matemático). Podría guardar algunos bytes si lo reemplaza i.to_s.chars.map(&:to_i).reduce(:+)con(j=i.to_s).sum-48*j.size
blutorange

@blutorange Oh, eso está bien. :) Todavía quiero volver a visitar la generación de imágenes. Trabajaré en tu sugerencia cuando lo haga.
Martin Ender

Solo necesita mantener una parte de su archivo fuente para la segunda y tercera ejecución, pero tal vez no sea tanta ganancia, ya que la parte más grande es la parte 3 actualmente. Con la codificación de longitud de ejecución, las filas codificadas de longitud de ejecución podría mejorar mucho la parte 3.
randomra

2

Python 3, 1027 + 1028 + 1029 = 3084

0 or print('0 '+' '.join(str(i)for i in range(1971)if sum(map(int,str(i)))==17))
1 or print("""+$,
L$, l$,
+$,
Its fleece was white as snow

And e(@,
@, @,
E(@
The!was sure to go

#&
S&, s&
#&
Which was against the rules.

%l/,
L/, l/,
%l/
To see a!at school""".translate({43:'Mary had a l',36:'ittle lamb',40:'verywhere that ',64:'Mary went',35:'It followed her to s',38:'chool one day',47:'augh and play',37:'It made the children ',33:' lamb '}))
q=[153,44]+[e-63for e in b'E@i@E@i@E@i@E@i@E@i@E@i@CgD@C@@@c@D@C@@@c@D@C@@@c@D@C@@@c@D@C@@aB@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@a@B@D@C@G^D@C@G@X@H@C@G@X@H@C@G@X@H@C@G@X@H@C@G@X@H@C@G@X@H@C@G@X@H@C@GdC@a@M@a@M@a@M@a@M@a@M@a@M@a@M@a@M@a@M@a@M@a@Mc~?']
1 or open("blue.ppm","w").write("P3\n50 50\n255\n"+' '.join(q[r*2]*'255 255 255 '+q[r*2+1]*'0 0 255 'for r in range(140)).strip())
f=open(__file__);l=f.read();f.close();open(__file__,"w").write(l.replace("0 or","0 and",1).replace("1 or","0 or",1))

Utiliza OEIS A166370 para los números, algo de compresión escrita a mano para Mary's Lamb y bytes ASCII para almacenar las cantidades de números en el PPM.

(Ab) usa la función de Python para evaluar solo el segundo valor para operadores lógicos si el resultado aún no se conoce (cortocircuito): 1 or codey 0 and codeno evalúa code. Luego usa algunos reemplazos para cambiar la 0 orejecución del código de habilitación a ay al 0 andlado 1 orde 0 or.

La tercera ejecución mata al segundo reemplazo, ya que no hay más 1 or. La cuarta ejecución mata al primer reemplazo, después de lo cual el código básicamente no hace nada.


2

C ++, 1204 * 3 = 3612 bytes

Compilador utilizado - GCC 4.9.2 con -std=c++14bandera

#include<bits/stdc++.h>
int i,j,c;
using namespace std;int main(){fstream f("F"),g("G");if(!f){while(i<=1970){j=i,c=0;while(j)c+=j%10,j/=10;if(c==17||i==0)cout<<i<<" ";i++;}ofstream f("F");}else if(f&&!g){cout<<R"(Mary had a little lamb,
Little lamb, little lamb,
Mary had a little lamb,
Its fleece was white as snow

And everywhere that Mary went,
Mary went, Mary went,
Everywhere that Mary went
The lamb was sure to go

It followed her to school one day
School one day, school one day
It followed her to school one day
Which was against the rules.

It made the children laugh and play,
Laugh and play, laugh and play,
It made the children laugh and play
To see a lamb at school)";ofstream g("G");}else{ofstream F("blue.ppm");F<<"P3 50 50 ";vector<int> L={1,2,4,10,13,16,19,25,28,43,76,103,109,127,190,460};vector<string> R={"1","?","43","10","6","161=","1","14","39","10","4","1514121<","1","151412","33","10","12","1315141;","1","13151417","30","10","1","1514171","7",":1814171","36","01","1","41","11",";191","35","01","1",">"},C={"255","0"};string r;i=j=0;while(i<R.size()){stringstream S(R[i]);S>>c;while(c--)r+=R[i+1];i+=2;}i=0;while(i<r.size()){c=L[r[i]-48];while(c--){F<<C[j]<<" ";}j=1-j;i++;}}}

Explicación -:

#include<bits/stdc++.h>
using namespace std;

int main()
{
    fstream f( "F" ), g( "G" ); // Creating empty files so that the program can decide when to run which task

    /**************************** TASK 1 ****************************/

    if ( !f ) // Print all the numbers within the range whose sum of digits is 17
    {
        for ( int i = 0; i <= 1970; i++ )
        {
            int j = i, c = 0;
            while ( j )
            {
                c += j % 10, j /= 10;
            }
            if ( c == 17 || i == 0 )
            {
                cout << i << " ";
            }
        }       
        ofstream f( "F" );
    }

    /**************************** TASK 2 ****************************/

    else if ( f && !g )  // Print the second task as it is using raw literals
    {
        cout << R"(Mary had a little lamb,
Little lamb, little lamb,
Mary had a little lamb,
Its fleece was white as snow

And everywhere that Mary went,
Mary went, Mary went,
Everywhere that Mary went
The lamb was sure to go

It followed her to school one day
School one day, school one day
It followed her to school one day
Which was against the rules.

It made the children laugh and play,
Laugh and play, laugh and play,
It made the children laugh and play
To see a lamb at school)";

        ofstream g( "G" );
    }

    /**************************** TASK 3 ****************************/

    else // Third task using Run-Length encoding
    {
        ofstream F( "blue.ppm" );
        F << "P3 50 50 ";

        /*
        L = set of all the lengths found during run-length encoding.
        R = vector containing the run-length encoding of "255" and "0" alternatingly.
                {1,"?") means 1 times print '255'/'0' "x" number of times where x = L[ASCII(?)-48] = L[15] = 469
        C = vector containing the two strings to be printed alternatingly.
        */

        vector<int> L = {1, 2, 4, 10, 13, 16, 19, 25, 28, 43, 76, 103, 109, 127, 190, 460};
        vector<string> R = {"1", "?", "43", "10", "6", "161=", "1", "14", "39", "10", "4", "1514121<", "1", "151412", "33", "10", "12", "1315141;", "1", "13151417", "30", "10", "1", "1514171", "7", ":1814171", "36", "01", "1", "41", "11", ";191", "35", "01", "1", ">"};
        vector<string> C = {"255", "0"};
        string r; // Contains the normal/expanded version of R
        i = j = 0;

        while ( i < R.size() ) // Expanding "R" into normal form and putting it into "r"
        {
            stringstream S( R[i] );
            S >> c; // Convert R[i] to its integer representation and put it in "c" with the help of stringstream.
            while ( c-- )
            {
                r += R[i + 1];
            }
            i += 2;
        }

        i = 0;
        while ( i < r.size() )
        {
            c = L[r[i] - 48];  // "c" is the number of times the current character has to be printed consecutively.
            while ( c-- )
            {
                F << C[j] << " ";
            }
            j = 1 - j; // Alternate "255" and "0"
            i++;
        }
    }
}

1

PHP (1552 * 3) +2 = 4658

Hay MUCHO golf que se puede hacer.

El código escribe 1 byte en otro archivo llamado z, para verificar el estado.

Aquí está el código:

<?$Y=base64_decode;$Z=gzdecode;$r=file_put_contents;switch(include z){case 2:$M=Mary;$l=lamb;$L=ittle;$H="$M had a l$L $l,";echo"$H
L$L $l, l$L $l,
$H
Its fleece was white as snow
";$W="$M went";$H="verywhere that $W";echo"
And e$H,
$W, $W,
E$H
The $l was sure to go
";$S=chool;$O="$S one day";$I="
It followed her to s$O";echo"$I
S$O, s$O$I
Which was against the rules.";$L="augh and play";$I="
It made the children l$L";echo"$I,
L$L, l$L,$I
To see a $l at s$S";$r(z,3);break;case 3:$r($Z($Y("H4sIAAAAAAAAA+3ay63CMBSE4X2qSAkRKH3QfzXAmkf8mBOfiX9dRXcBfIJ47Ewwj/uyb+u+Lbd9fx+r4gACAgICApoO2l5/Jf+X0icCfT4gG7ka0DSQQMdPOopB9bv5BpqeG6D2F8tjQK6AImJAroAiYkCugCJiQK6AemKQ51bWF+oaxe6Rk2DhAQUKjIIEIU9Xg8gTkBIiT0BKKG+e8nXMcVBRDFr3oyQIq4ADVLW1JNnfbEYkafYcJR+oatZLtrebEfluPVAAVDNKkn3yZoQfazhAvXkaW4WNTvQkEIUcSAlRyIGUEIUcSApRyIGUEIUcSAlRyIGUEIUcSAlRyH+/w/H7qudDXUHMf+nhpq4NGlM28196KOFtx5iy6XPpyTdiyaEhZdPg3p0S3gaN+baOAnVViAKV6bz4QxSoTOflAhAF6v95yfdVTT7otOEPCSNQ+Sgl/1jc9RhAhrOfux8HyG/2Gy6xE0J+v+AyXGInhAxnv98SOyFkOPspWA6Q3+w3XGInhChYQBGQ4ez3W2InhAxnPwXLAfKb/RfYvQqH8o0aENBJ0BPXPoC/PXEAAA==")),'blue.ppm');$r(z,1);break;default:echo$Z($Y("H4sIAAAAAAAAAxXSuQHAMAwCwFUYIehF+y8WXCWWaDj7gw4ncA+Uv7eIOcQKIf/fIPuQI+QuUj5fo+pQLdQsagclz67QeegSuhc9g95Gy/NLTBwmhanF9GCmMVsYeXeB5WFD2FxsDbYbO4XdxMr7I/QdREGxUA5UDXVBk9AGJGfuw33CcXExuGxcFa4TN4FbuvIHfq/091p/rk26N+nipN7R1RnuznB5htsz9GbuzzQA0wJMEzBtwNRbWIFlBpYdWIZgWYJlCpbe1hhsa7DNwbYH2yBsi7BNwtaLGIVjFY5ZOHbhGIZjGY5pOLbh6OWswzUPl+8+DcS1ENdEXBtxjcS1kku+sJ0oQ1GWokxF2YoyFmUtylzG4HsbDt/n3NG5C+cunbty7tq5G7738/3WJLxlVAIAAA=="));$r(z,2);}

Sé que este es un código MUY horrible y puedo intentar jugar golf en la lista.

Pero bueno, no puedo encontrar un patrón en esos: /


Puedo ver que has comprimido la imagen en la última línea y minimizado el tamaño de la canción de cordero en las otras líneas, pero no puedo ver dónde está la secuencia entera. Necesito un poco de orientación
Logic Knight

@CarpetPython Más compresión. Es la ultima parte.
Ismael Miguel

Gracias. Si está buscando el patrón en la secuencia de enteros, algunas de las otras respuestas pueden darle algunas pistas. Por ejemplo: los dígitos de los números tienen una propiedad interesante.
Logic Knight

@CarpetPython No soy un imitador. Pero sí eché un vistazo a las otras respuestas, pero aún no pude encontrar el patrón.
Ismael Miguel

1

Rubí, 691 + 612 + 225 = 1528 bytes

Otra respuesta auto modificable, esta vez en rubí. Ayer me topé con esto y la compresión del poema me llamó la atención. Solo pude guardar 92 bytes del texto sin procesar pensado.

puts (0..1970).select{|i|eval(i.to_s.gsub /\B/,'+')%17<1}*' '
File.write $0,%@
m='lamb:little 0,:Mary :2had a 1:2went:laugh and play:It made the children 5:It followed her to 10: was : that :11 one day:school:everywhere:3
1 1
3
Its fleece8white as snow

And 1294,
4, 4,
1294
The 08sure to go

7
10, 10
7
Which8against the rules.

6,
5, 5,
6
To see a 0 at 11'
12.downto(0){|i|m.gsub! i.to_s,m.split(':')[i]}
puts m.split(':')[13].gsub /^./,&:upcase
File.write $0,%!a=[1]
s=[16,13]
File.write 'blue.ppm',(' P3
50 50 '+[460,a*43,[19,127]*6,13,a*39,[s,4,109]*4,s,4,a*33,10,s,[103,10,s]*12,25,a*30,s,25,[76,28,13,25]*7,a*36,13,[103,43]*11,a*35,190].flatten.map{|i|('255 '*i)+'0 0 '}*' ')[1..-6]!@

Cada ejecución del programa reescribe el programa para la siguiente ejecución. No descubrí la propiedad especial de los números hasta que examiné las otras respuestas. Y tomé la misma inspiración de la respuesta de Martin, pero no utilicé la sugerencia de blutorange porque no podía entender por qué funcionó.

Probablemente hay algunos bytes más para buscar del poema, pero no mucho. Básicamente se trata de subcapsular en secuencias comunes y está usando el orden de la secuencia como la parte para reemplazar.

Para la parte final, la longitud codifiqué todos los 255 bytes en una matriz y los pegué junto con los ceros. Y luego codifiqué las secuencias repetidas en esa matriz.


0

JavaScript + HTML, 3909 bytes

Esta es una aplicación HTML / JavaScript que utiliza la recarga de página para volver a ejecutar archivos. Me burlé al analizar el PPM. Si sabe cómo leer archivos PPM, hágamelo saber. Stack Exchange no permite el almacenamiento local. Entonces tendrá que ejecutar JSFiddle.

De nuevo, sé que bombardeé el archivo ppm.

Pero al final:

JavaScript : 1,232 bytes * 3 = 3696

HTML : 71 bytes * 3 = 213

Total : 3909

JSFiddle

var mary = '</br>\
Mary had a little lamb,</br>\
Little lamb, little lamb,</br>\
Mary had a little lamb,</br>\
Its fleece was white as snow</br>\
</br>\
And everywhere that Mary went,</br>\
Mary went, Mary went,</br>\
Everywhere that Mary went</br>\
The lamb was sure to go</br>\
</br>\
It followed her to school one day</br>\
School one day, school one day</br>\
It followed her to school one day</br>\
Which was against the rules.</br>\
</br></br>\
It made the children laugh and play,</br>\
Laugh and play, laugh and play,</br>\
It made the children laugh and play</br>\
To see a lamb at school\
';
var img = '<img src="http://i.stack.imgur.com/sHCba.png"/>'

function run() {
  var step = localStorage.getItem("step");
  step++;
  localStorage.setItem("step", step);
  $('#step').text(step);
  var r = $('#results')
  switch (step) {
    case 1:
      for (var i = 0; i < 133; i++) {
        r.append(i + 1 + ' ');
      }
      break;
    case 2:
      r.html(mary);
      break;
    case 3:
      r.html(img);
      localStorage.setItem("step", 0);
      break;
    default:
      localStorage.setItem("step", 0);
      break;
  }
}
run();
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
<label>Step: <span id='step'></span>
</label>
<div id='results'></div>


Hola @Neoaptt y bienvenido a PPCG. Es posible que deba leer la pregunta detenidamente para asegurarse de que su respuesta cumpla con los requisitos. Su secuencia de enteros no coincide con la secuencia requerida, y la imagen no se escribe en un archivo en formato PPM. El punto de la pregunta es desafiarlo a almacenar las tres salidas de forma comprimida dentro de su programa, explotando patrones. Tampoco es que la descarga de archivos de la red sea una de las 'lagunas estándar' prohibidas en la pregunta.
Logic Knight

Al usar nuestro sitio, usted reconoce que ha leído y comprende nuestra Política de Cookies y Política de Privacidad.
Licensed under cc by-sa 3.0 with attribution required.